From e071a3f6c06f41068ad17134189a4ac3073ef76b Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sun, 30 Aug 2009 03:46:39 +0000 Subject: [PATCH 001/133] First commit git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@1 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- base58.h | 201 + bignum.h | 498 ++ db.cpp | 614 ++ db.h | 420 ++ headers.h | 71 + irc.cpp | 314 + irc.h | 10 + key.h | 156 + libeay32.dll | Bin 0 -> 1306630 bytes license.txt | 19 + main.cpp | 2692 ++++++++ main.h | 1329 ++++ makefile | 83 + makefile.vc | 77 + market.cpp | 264 + market.h | 182 + mingwm10.dll | Bin 0 -> 11673 bytes net.cpp | 1100 ++++ net.h | 856 +++ rc/addressbook16.bmp | Bin 0 -> 1334 bytes rc/addressbook16mask.bmp | Bin 0 -> 126 bytes rc/addressbook20.bmp | Bin 0 -> 1478 bytes rc/addressbook20mask.bmp | Bin 0 -> 142 bytes rc/bitcoin.ico | Bin 0 -> 22486 bytes rc/check.ico | Bin 0 -> 766 bytes rc/send16.bmp | Bin 0 -> 1334 bytes rc/send16mask.bmp | Bin 0 -> 126 bytes rc/send16masknoshadow.bmp | Bin 0 -> 126 bytes rc/send20.bmp | Bin 0 -> 1478 bytes rc/send20mask.bmp | Bin 0 -> 142 bytes readme.txt | 76 + script.cpp | 1127 ++++ script.h | 597 ++ serialize.h | 1158 ++++ sha.cpp | 554 ++ sha.h | 177 + ui.cpp | 3290 ++++++++++ ui.h | 420 ++ ui.rc | 14 + uibase.cpp | 1825 ++++++ uibase.h | 723 +++ uint256.h | 750 +++ uiproject.fbp | 11860 ++++++++++++++++++++++++++++++++++++ util.cpp | 383 ++ util.h | 399 ++ 45 files changed, 32239 insertions(+) create mode 100644 base58.h create mode 100644 bignum.h create mode 100644 db.cpp create mode 100644 db.h create mode 100644 headers.h create mode 100644 irc.cpp create mode 100644 irc.h create mode 100644 key.h create mode 100644 libeay32.dll create mode 100644 license.txt create mode 100644 main.cpp create mode 100644 main.h create mode 100644 makefile create mode 100644 makefile.vc create mode 100644 market.cpp create mode 100644 market.h create mode 100644 mingwm10.dll create mode 100644 net.cpp create mode 100644 net.h create mode 100644 rc/addressbook16.bmp create mode 100644 rc/addressbook16mask.bmp create mode 100644 rc/addressbook20.bmp create mode 100644 rc/addressbook20mask.bmp create mode 100644 rc/bitcoin.ico create mode 100644 rc/check.ico create mode 100644 rc/send16.bmp create mode 100644 rc/send16mask.bmp create mode 100644 rc/send16masknoshadow.bmp create mode 100644 rc/send20.bmp create mode 100644 rc/send20mask.bmp create mode 100644 readme.txt create mode 100644 script.cpp create mode 100644 script.h create mode 100644 serialize.h create mode 100644 sha.cpp create mode 100644 sha.h create mode 100644 ui.cpp create mode 100644 ui.h create mode 100644 ui.rc create mode 100644 uibase.cpp create mode 100644 uibase.h create mode 100644 uint256.h create mode 100644 uiproject.fbp create mode 100644 util.cpp create mode 100644 util.h diff --git a/base58.h b/base58.h new file mode 100644 index 00000000..69f40bd2 --- /dev/null +++ b/base58.h @@ -0,0 +1,201 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + + +// +// Why base-58 instead of standard base-64 encoding? +// - Don't want 0OIl characters that look the same in some fonts and +// could be used to create visually identical looking account numbers. +// - A string with non-alphanumeric characters is not as easily accepted as an account number. +// - E-mail usually won't line-break if there's no punctuation to break at. +// - Doubleclicking selects the whole number as one word if it's all alphanumeric. +// + + +static const char* pszBase58 = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz"; + + +inline string EncodeBase58(const unsigned char* pbegin, const unsigned char* pend) +{ + CAutoBN_CTX pctx; + CBigNum bn58 = 58; + CBigNum bn0 = 0; + + // Convert big endian data to little endian + // Extra zero at the end make sure bignum will interpret as a positive number + vector vchTmp(pend-pbegin+1, 0); + reverse_copy(pbegin, pend, vchTmp.begin()); + + // Convert little endian data to bignum + CBigNum bn; + bn.setvch(vchTmp); + + // Convert bignum to string + string str; + str.reserve((pend - pbegin) * 138 / 100 + 1); + CBigNum dv; + CBigNum rem; + while (bn > bn0) + { + if (!BN_div(&dv, &rem, &bn, &bn58, pctx)) + throw bignum_error("EncodeBase58 : BN_div failed"); + bn = dv; + unsigned int c = rem.getulong(); + str += pszBase58[c]; + } + + // Leading zeroes encoded as base58 zeros + for (const unsigned char* p = pbegin; p < pend && *p == 0; p++) + str += pszBase58[0]; + + // Convert little endian string to big endian + reverse(str.begin(), str.end()); + return str; +} + +inline string EncodeBase58(const vector& vch) +{ + return EncodeBase58(&vch[0], &vch[0] + vch.size()); +} + +inline bool DecodeBase58(const char* psz, vector& vchRet) +{ + CAutoBN_CTX pctx; + vchRet.clear(); + CBigNum bn58 = 58; + CBigNum bn = 0; + CBigNum bnChar; + while (isspace(*psz)) + psz++; + + // Convert big endian string to bignum + for (const char* p = psz; *p; p++) + { + const char* p1 = strchr(pszBase58, *p); + if (p1 == NULL) + { + while (isspace(*p)) + p++; + if (*p != '\0') + return false; + break; + } + bnChar.setulong(p1 - pszBase58); + if (!BN_mul(&bn, &bn, &bn58, pctx)) + throw bignum_error("DecodeBase58 : BN_mul failed"); + bn += bnChar; + } + + // Get bignum as little endian data + vector vchTmp = bn.getvch(); + + // Trim off sign byte if present + if (vchTmp.size() >= 2 && vchTmp.end()[-1] == 0 && vchTmp.end()[-2] >= 0x80) + vchTmp.erase(vchTmp.end()-1); + + // Restore leading zeros + int nLeadingZeros = 0; + for (const char* p = psz; *p == pszBase58[0]; p++) + nLeadingZeros++; + vchRet.assign(nLeadingZeros + vchTmp.size(), 0); + + // Convert little endian data to big endian + reverse_copy(vchTmp.begin(), vchTmp.end(), vchRet.end() - vchTmp.size()); + return true; +} + +inline bool DecodeBase58(const string& str, vector& vchRet) +{ + return DecodeBase58(str.c_str(), vchRet); +} + + + + + +inline string EncodeBase58Check(const vector& vchIn) +{ + // add 4-byte hash check to the end + vector vch(vchIn); + uint256 hash = Hash(vch.begin(), vch.end()); + vch.insert(vch.end(), (unsigned char*)&hash, (unsigned char*)&hash + 4); + return EncodeBase58(vch); +} + +inline bool DecodeBase58Check(const char* psz, vector& vchRet) +{ + if (!DecodeBase58(psz, vchRet)) + return false; + if (vchRet.size() < 4) + { + vchRet.clear(); + return false; + } + uint256 hash = Hash(vchRet.begin(), vchRet.end()-4); + if (memcmp(&hash, &vchRet.end()[-4], 4) != 0) + { + vchRet.clear(); + return false; + } + vchRet.resize(vchRet.size()-4); + return true; +} + +inline bool DecodeBase58Check(const string& str, vector& vchRet) +{ + return DecodeBase58Check(str.c_str(), vchRet); +} + + + + + + +static const unsigned char ADDRESSVERSION = 0; + +inline string Hash160ToAddress(uint160 hash160) +{ + // add 1-byte version number to the front + vector vch(1, ADDRESSVERSION); + vch.insert(vch.end(), UBEGIN(hash160), UEND(hash160)); + return EncodeBase58Check(vch); +} + +inline bool AddressToHash160(const char* psz, uint160& hash160Ret) +{ + vector vch; + if (!DecodeBase58Check(psz, vch)) + return false; + if (vch.empty()) + return false; + unsigned char nVersion = vch[0]; + if (vch.size() != sizeof(hash160Ret) + 1) + return false; + memcpy(&hash160Ret, &vch[1], sizeof(hash160Ret)); + return (nVersion <= ADDRESSVERSION); +} + +inline bool AddressToHash160(const string& str, uint160& hash160Ret) +{ + return AddressToHash160(str.c_str(), hash160Ret); +} + +inline bool IsValidBitcoinAddress(const char* psz) +{ + uint160 hash160; + return AddressToHash160(psz, hash160); +} + +inline bool IsValidBitcoinAddress(const string& str) +{ + return IsValidBitcoinAddress(str.c_str()); +} + + + + +inline string PubKeyToAddress(const vector& vchPubKey) +{ + return Hash160ToAddress(Hash160(vchPubKey)); +} diff --git a/bignum.h b/bignum.h new file mode 100644 index 00000000..8aa4e9c4 --- /dev/null +++ b/bignum.h @@ -0,0 +1,498 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include +#include +#include + + + + + +class bignum_error : public std::runtime_error +{ +public: + explicit bignum_error(const std::string& str) : std::runtime_error(str) {} +}; + + + +class CAutoBN_CTX +{ +protected: + BN_CTX* pctx; + BN_CTX* operator=(BN_CTX* pnew) { return pctx = pnew; } + +public: + CAutoBN_CTX() + { + pctx = BN_CTX_new(); + if (pctx == NULL) + throw bignum_error("CAutoBN_CTX : BN_CTX_new() returned NULL"); + } + + ~CAutoBN_CTX() + { + if (pctx != NULL) + BN_CTX_free(pctx); + } + + operator BN_CTX*() { return pctx; } + BN_CTX& operator*() { return *pctx; } + BN_CTX** operator&() { return &pctx; } + bool operator!() { return (pctx == NULL); } +}; + + + +class CBigNum : public BIGNUM +{ +public: + CBigNum() + { + BN_init(this); + } + + CBigNum(const CBigNum& b) + { + BN_init(this); + if (!BN_copy(this, &b)) + { + BN_clear_free(this); + throw bignum_error("CBigNum::CBigNum(const CBigNum&) : BN_copy failed"); + } + } + + explicit CBigNum(const std::string& str) + { + BN_init(this); + SetHex(str); + } + + CBigNum& operator=(const CBigNum& b) + { + if (!BN_copy(this, &b)) + throw bignum_error("CBigNum::operator= : BN_copy failed"); + return (*this); + } + + ~CBigNum() + { + BN_clear_free(this); + } + + CBigNum(char n) { BN_init(this); if (n >= 0) setulong(n); else setint64(n); } + CBigNum(short n) { BN_init(this); if (n >= 0) setulong(n); else setint64(n); } + CBigNum(int n) { BN_init(this); if (n >= 0) setulong(n); else setint64(n); } + CBigNum(long n) { BN_init(this); if (n >= 0) setulong(n); else setint64(n); } + CBigNum(int64 n) { BN_init(this); setint64(n); } + CBigNum(unsigned char n) { BN_init(this); setulong(n); } + CBigNum(unsigned short n) { BN_init(this); setulong(n); } + CBigNum(unsigned int n) { BN_init(this); setulong(n); } + CBigNum(unsigned long n) { BN_init(this); setulong(n); } + CBigNum(uint64 n) { BN_init(this); setuint64(n); } + explicit CBigNum(uint256 n) { BN_init(this); setuint256(n); } + + explicit CBigNum(const std::vector& vch) + { + BN_init(this); + setvch(vch); + } + + void setulong(unsigned long n) + { + if (!BN_set_word(this, n)) + throw bignum_error("CBigNum conversion from unsigned long : BN_set_word failed"); + } + + unsigned long getulong() const + { + return BN_get_word(this); + } + + unsigned int getuint() const + { + return BN_get_word(this); + } + + int getint() const + { + unsigned long n = BN_get_word(this); + if (!BN_is_negative(this)) + return (n > INT_MAX ? INT_MAX : n); + else + return (n > INT_MAX ? INT_MIN : -(int)n); + } + + void setint64(int64 n) + { + unsigned char pch[sizeof(n) + 6]; + unsigned char* p = pch + 4; + bool fNegative = false; + if (n < (int64)0) + { + n = -n; + fNegative = true; + } + bool fLeadingZeroes = true; + for (int i = 0; i < 8; i++) + { + unsigned char c = (n >> 56) & 0xff; + n <<= 8; + if (fLeadingZeroes) + { + if (c == 0) + continue; + if (c & 0x80) + *p++ = (fNegative ? 0x80 : 0); + else if (fNegative) + c |= 0x80; + fLeadingZeroes = false; + } + *p++ = c; + } + unsigned int nSize = p - (pch + 4); + pch[0] = (nSize >> 24) & 0xff; + pch[1] = (nSize >> 16) & 0xff; + pch[2] = (nSize >> 8) & 0xff; + pch[3] = (nSize) & 0xff; + BN_mpi2bn(pch, p - pch, this); + } + + void setuint64(uint64 n) + { + unsigned char pch[sizeof(n) + 6]; + unsigned char* p = pch + 4; + bool fLeadingZeroes = true; + for (int i = 0; i < 8; i++) + { + unsigned char c = (n >> 56) & 0xff; + n <<= 8; + if (fLeadingZeroes) + { + if (c == 0) + continue; + if (c & 0x80) + *p++ = 0; + fLeadingZeroes = false; + } + *p++ = c; + } + unsigned int nSize = p - (pch + 4); + pch[0] = (nSize >> 24) & 0xff; + pch[1] = (nSize >> 16) & 0xff; + pch[2] = (nSize >> 8) & 0xff; + pch[3] = (nSize) & 0xff; + BN_mpi2bn(pch, p - pch, this); + } + + void setuint256(uint256 n) + { + unsigned char pch[sizeof(n) + 6]; + unsigned char* p = pch + 4; + bool fLeadingZeroes = true; + unsigned char* pbegin = (unsigned char*)&n; + unsigned char* psrc = pbegin + sizeof(n); + while (psrc != pbegin) + { + unsigned char c = *(--psrc); + if (fLeadingZeroes) + { + if (c == 0) + continue; + if (c & 0x80) + *p++ = 0; + fLeadingZeroes = false; + } + *p++ = c; + } + unsigned int nSize = p - (pch + 4); + pch[0] = (nSize >> 24) & 0xff; + pch[1] = (nSize >> 16) & 0xff; + pch[2] = (nSize >> 8) & 0xff; + pch[3] = (nSize >> 0) & 0xff; + BN_mpi2bn(pch, p - pch, this); + } + + uint256 getuint256() + { + unsigned int nSize = BN_bn2mpi(this, NULL); + if (nSize < 4) + return 0; + std::vector vch(nSize); + BN_bn2mpi(this, &vch[0]); + if (vch.size() > 4) + vch[4] &= 0x7f; + uint256 n = 0; + for (int i = 0, j = vch.size()-1; i < sizeof(n) && j >= 4; i++, j--) + ((unsigned char*)&n)[i] = vch[j]; + return n; + } + + void setvch(const std::vector& vch) + { + std::vector vch2(vch.size() + 4); + unsigned int nSize = vch.size(); + vch2[0] = (nSize >> 24) & 0xff; + vch2[1] = (nSize >> 16) & 0xff; + vch2[2] = (nSize >> 8) & 0xff; + vch2[3] = (nSize >> 0) & 0xff; + reverse_copy(vch.begin(), vch.end(), vch2.begin() + 4); + BN_mpi2bn(&vch2[0], vch2.size(), this); + } + + std::vector getvch() const + { + unsigned int nSize = BN_bn2mpi(this, NULL); + if (nSize < 4) + return std::vector(); + std::vector vch(nSize); + BN_bn2mpi(this, &vch[0]); + vch.erase(vch.begin(), vch.begin() + 4); + reverse(vch.begin(), vch.end()); + return vch; + } + + CBigNum& SetCompact(unsigned int nCompact) + { + unsigned int nSize = nCompact >> 24; + std::vector vch(4 + nSize); + vch[3] = nSize; + if (nSize >= 1) vch[4] = (nCompact >> 16) & 0xff; + if (nSize >= 2) vch[5] = (nCompact >> 8) & 0xff; + if (nSize >= 3) vch[6] = (nCompact >> 0) & 0xff; + BN_mpi2bn(&vch[0], vch.size(), this); + return *this; + } + + unsigned int GetCompact() const + { + unsigned int nSize = BN_bn2mpi(this, NULL); + std::vector vch(nSize); + nSize -= 4; + BN_bn2mpi(this, &vch[0]); + unsigned int nCompact = nSize << 24; + if (nSize >= 1) nCompact |= (vch[4] << 16); + if (nSize >= 2) nCompact |= (vch[5] << 8); + if (nSize >= 3) nCompact |= (vch[6] << 0); + return nCompact; + } + + void SetHex(const std::string& str) + { + // skip 0x + const char* psz = str.c_str(); + while (isspace(*psz)) + psz++; + bool fNegative = false; + if (*psz == '-') + { + fNegative = true; + psz++; + } + if (psz[0] == '0' && tolower(psz[1]) == 'x') + psz += 2; + while (isspace(*psz)) + psz++; + + // hex string to bignum + static char phexdigit[256] = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,1,2,3,4,5,6,7,8,9,0,0,0,0,0,0, 0,0xa,0xb,0xc,0xd,0xe,0xf,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0xa,0xb,0xc,0xd,0xe,0xf,0,0,0,0,0,0,0,0,0 }; + *this = 0; + while (isxdigit(*psz)) + { + *this <<= 4; + int n = phexdigit[*psz++]; + *this += n; + } + if (fNegative) + *this = 0 - *this; + } + + unsigned int GetSerializeSize(int nType=0, int nVersion=VERSION) const + { + return ::GetSerializeSize(getvch(), nType, nVersion); + } + + template + void Serialize(Stream& s, int nType=0, int nVersion=VERSION) const + { + ::Serialize(s, getvch(), nType, nVersion); + } + + template + void Unserialize(Stream& s, int nType=0, int nVersion=VERSION) + { + vector vch; + ::Unserialize(s, vch, nType, nVersion); + setvch(vch); + } + + + bool operator!() const + { + return BN_is_zero(this); + } + + CBigNum& operator+=(const CBigNum& b) + { + if (!BN_add(this, this, &b)) + throw bignum_error("CBigNum::operator+= : BN_add failed"); + return *this; + } + + CBigNum& operator-=(const CBigNum& b) + { + *this = *this - b; + return *this; + } + + CBigNum& operator*=(const CBigNum& b) + { + CAutoBN_CTX pctx; + if (!BN_mul(this, this, &b, pctx)) + throw bignum_error("CBigNum::operator*= : BN_mul failed"); + return *this; + } + + CBigNum& operator/=(const CBigNum& b) + { + *this = *this / b; + return *this; + } + + CBigNum& operator%=(const CBigNum& b) + { + *this = *this % b; + return *this; + } + + CBigNum& operator<<=(unsigned int shift) + { + if (!BN_lshift(this, this, shift)) + throw bignum_error("CBigNum:operator<<= : BN_lshift failed"); + return *this; + } + + CBigNum& operator>>=(unsigned int shift) + { + if (!BN_rshift(this, this, shift)) + throw bignum_error("CBigNum:operator>>= : BN_rshift failed"); + return *this; + } + + + CBigNum& operator++() + { + // prefix operator + if (!BN_add(this, this, BN_value_one())) + throw bignum_error("CBigNum::operator++ : BN_add failed"); + return *this; + } + + const CBigNum operator++(int) + { + // postfix operator + const CBigNum ret = *this; + ++(*this); + return ret; + } + + CBigNum& operator--() + { + // prefix operator + CBigNum r; + if (!BN_sub(&r, this, BN_value_one())) + throw bignum_error("CBigNum::operator-- : BN_sub failed"); + *this = r; + return *this; + } + + const CBigNum operator--(int) + { + // postfix operator + const CBigNum ret = *this; + --(*this); + return ret; + } + + + friend inline const CBigNum operator-(const CBigNum& a, const CBigNum& b); + friend inline const CBigNum operator/(const CBigNum& a, const CBigNum& b); + friend inline const CBigNum operator%(const CBigNum& a, const CBigNum& b); +}; + + + +inline const CBigNum operator+(const CBigNum& a, const CBigNum& b) +{ + CBigNum r; + if (!BN_add(&r, &a, &b)) + throw bignum_error("CBigNum::operator+ : BN_add failed"); + return r; +} + +inline const CBigNum operator-(const CBigNum& a, const CBigNum& b) +{ + CBigNum r; + if (!BN_sub(&r, &a, &b)) + throw bignum_error("CBigNum::operator- : BN_sub failed"); + return r; +} + +inline const CBigNum operator-(const CBigNum& a) +{ + CBigNum r(a); + BN_set_negative(&r, !BN_is_negative(&r)); + return r; +} + +inline const CBigNum operator*(const CBigNum& a, const CBigNum& b) +{ + CAutoBN_CTX pctx; + CBigNum r; + if (!BN_mul(&r, &a, &b, pctx)) + throw bignum_error("CBigNum::operator* : BN_mul failed"); + return r; +} + +inline const CBigNum operator/(const CBigNum& a, const CBigNum& b) +{ + CAutoBN_CTX pctx; + CBigNum r; + if (!BN_div(&r, NULL, &a, &b, pctx)) + throw bignum_error("CBigNum::operator/ : BN_div failed"); + return r; +} + +inline const CBigNum operator%(const CBigNum& a, const CBigNum& b) +{ + CAutoBN_CTX pctx; + CBigNum r; + if (!BN_mod(&r, &a, &b, pctx)) + throw bignum_error("CBigNum::operator% : BN_div failed"); + return r; +} + +inline const CBigNum operator<<(const CBigNum& a, unsigned int shift) +{ + CBigNum r; + if (!BN_lshift(&r, &a, shift)) + throw bignum_error("CBigNum:operator<< : BN_lshift failed"); + return r; +} + +inline const CBigNum operator>>(const CBigNum& a, unsigned int shift) +{ + CBigNum r; + if (!BN_rshift(&r, &a, shift)) + throw bignum_error("CBigNum:operator>> : BN_rshift failed"); + return r; +} + +inline bool operator==(const CBigNum& a, const CBigNum& b) { return (BN_cmp(&a, &b) == 0); } +inline bool operator!=(const CBigNum& a, const CBigNum& b) { return (BN_cmp(&a, &b) != 0); } +inline bool operator<=(const CBigNum& a, const CBigNum& b) { return (BN_cmp(&a, &b) <= 0); } +inline bool operator>=(const CBigNum& a, const CBigNum& b) { return (BN_cmp(&a, &b) >= 0); } +inline bool operator<(const CBigNum& a, const CBigNum& b) { return (BN_cmp(&a, &b) < 0); } +inline bool operator>(const CBigNum& a, const CBigNum& b) { return (BN_cmp(&a, &b) > 0); } diff --git a/db.cpp b/db.cpp new file mode 100644 index 00000000..f38f3ffa --- /dev/null +++ b/db.cpp @@ -0,0 +1,614 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" + + + + + + +// +// CDB +// + +static CCriticalSection cs_db; +static bool fDbEnvInit = false; +DbEnv dbenv(0); +static map mapFileUseCount; + +class CDBInit +{ +public: + CDBInit() + { + } + ~CDBInit() + { + if (fDbEnvInit) + { + dbenv.close(0); + fDbEnvInit = false; + } + } +} +instance_of_cdbinit; + + +CDB::CDB(const char* pszFile, const char* pszMode, bool fTxn) : pdb(NULL) +{ + int ret; + if (pszFile == NULL) + return; + + bool fCreate = strchr(pszMode, 'c'); + bool fReadOnly = (!strchr(pszMode, '+') && !strchr(pszMode, 'w')); + unsigned int nFlags = DB_THREAD; + if (fCreate) + nFlags |= DB_CREATE; + else if (fReadOnly) + nFlags |= DB_RDONLY; + if (!fReadOnly || fTxn) + nFlags |= DB_AUTO_COMMIT; + + CRITICAL_BLOCK(cs_db) + { + if (!fDbEnvInit) + { + string strAppDir = GetAppDir(); + string strLogDir = strAppDir + "\\database"; + _mkdir(strLogDir.c_str()); + printf("dbenv.open strAppDir=%s\n", strAppDir.c_str()); + + dbenv.set_lg_dir(strLogDir.c_str()); + dbenv.set_lg_max(10000000); + dbenv.set_lk_max_locks(10000); + dbenv.set_lk_max_objects(10000); + dbenv.set_errfile(fopen("db.log", "a")); /// debug + ///dbenv.log_set_config(DB_LOG_AUTO_REMOVE, 1); /// causes corruption + ret = dbenv.open(strAppDir.c_str(), + DB_CREATE | + DB_INIT_LOCK | + DB_INIT_LOG | + DB_INIT_MPOOL | + DB_INIT_TXN | + DB_THREAD | + DB_PRIVATE | + DB_RECOVER, + 0); + if (ret > 0) + throw runtime_error(strprintf("CDB() : error %d opening database environment\n", ret)); + fDbEnvInit = true; + } + + strFile = pszFile; + ++mapFileUseCount[strFile]; + } + + pdb = new Db(&dbenv, 0); + + ret = pdb->open(NULL, // Txn pointer + pszFile, // Filename + "main", // Logical db name + DB_BTREE, // Database type + nFlags, // Flags + 0); + + if (ret > 0) + { + delete pdb; + pdb = NULL; + CRITICAL_BLOCK(cs_db) + --mapFileUseCount[strFile]; + strFile = ""; + throw runtime_error(strprintf("CDB() : can't open database file %s, error %d\n", pszFile, ret)); + } + + if (fCreate && !Exists(string("version"))) + WriteVersion(VERSION); + + RandAddSeed(); +} + +void CDB::Close() +{ + if (!pdb) + return; + if (!vTxn.empty()) + vTxn.front()->abort(); + vTxn.clear(); + pdb->close(0); + delete pdb; + pdb = NULL; + dbenv.txn_checkpoint(0, 0, 0); + + CRITICAL_BLOCK(cs_db) + --mapFileUseCount[strFile]; + + RandAddSeed(); +} + +void DBFlush(bool fShutdown) +{ + // Flush log data to the actual data file + // on all files that are not in use + printf("DBFlush(%s)\n", fShutdown ? "true" : "false"); + CRITICAL_BLOCK(cs_db) + { + dbenv.txn_checkpoint(0, 0, 0); + map::iterator mi = mapFileUseCount.begin(); + while (mi != mapFileUseCount.end()) + { + string strFile = (*mi).first; + int nRefCount = (*mi).second; + if (nRefCount == 0) + { + dbenv.lsn_reset(strFile.c_str(), 0); + mapFileUseCount.erase(mi++); + } + else + mi++; + } + if (fShutdown) + { + char** listp; + if (mapFileUseCount.empty()) + dbenv.log_archive(&listp, DB_ARCH_REMOVE); + dbenv.close(0); + fDbEnvInit = false; + } + } +} + + + + + + +// +// CTxDB +// + +bool CTxDB::ReadTxIndex(uint256 hash, CTxIndex& txindex) +{ + assert(!fClient); + txindex.SetNull(); + return Read(make_pair(string("tx"), hash), txindex); +} + +bool CTxDB::UpdateTxIndex(uint256 hash, const CTxIndex& txindex) +{ + assert(!fClient); + return Write(make_pair(string("tx"), hash), txindex); +} + +bool CTxDB::AddTxIndex(const CTransaction& tx, const CDiskTxPos& pos, int nHeight) +{ + assert(!fClient); + + // Add to tx index + uint256 hash = tx.GetHash(); + CTxIndex txindex(pos, tx.vout.size()); + return Write(make_pair(string("tx"), hash), txindex); +} + +bool CTxDB::EraseTxIndex(const CTransaction& tx) +{ + assert(!fClient); + uint256 hash = tx.GetHash(); + + return Erase(make_pair(string("tx"), hash)); +} + +bool CTxDB::ContainsTx(uint256 hash) +{ + assert(!fClient); + return Exists(make_pair(string("tx"), hash)); +} + +bool CTxDB::ReadOwnerTxes(uint160 hash160, int nMinHeight, vector& vtx) +{ + assert(!fClient); + vtx.clear(); + + // Get cursor + Dbc* pcursor = GetCursor(); + if (!pcursor) + return false; + + unsigned int fFlags = DB_SET_RANGE; + loop + { + // Read next record + CDataStream ssKey; + if (fFlags == DB_SET_RANGE) + ssKey << string("owner") << hash160 << CDiskTxPos(0, 0, 0); + CDataStream ssValue; + int ret = ReadAtCursor(pcursor, ssKey, ssValue, fFlags); + fFlags = DB_NEXT; + if (ret == DB_NOTFOUND) + break; + else if (ret != 0) + return false; + + // Unserialize + string strType; + uint160 hashItem; + CDiskTxPos pos; + ssKey >> strType >> hashItem >> pos; + int nItemHeight; + ssValue >> nItemHeight; + + // Read transaction + if (strType != "owner" || hashItem != hash160) + break; + if (nItemHeight >= nMinHeight) + { + vtx.resize(vtx.size()+1); + if (!vtx.back().ReadFromDisk(pos)) + return false; + } + } + return true; +} + +bool CTxDB::ReadDiskTx(uint256 hash, CTransaction& tx, CTxIndex& txindex) +{ + assert(!fClient); + tx.SetNull(); + if (!ReadTxIndex(hash, txindex)) + return false; + return (tx.ReadFromDisk(txindex.pos)); +} + +bool CTxDB::ReadDiskTx(uint256 hash, CTransaction& tx) +{ + CTxIndex txindex; + return ReadDiskTx(hash, tx, txindex); +} + +bool CTxDB::ReadDiskTx(COutPoint outpoint, CTransaction& tx, CTxIndex& txindex) +{ + return ReadDiskTx(outpoint.hash, tx, txindex); +} + +bool CTxDB::ReadDiskTx(COutPoint outpoint, CTransaction& tx) +{ + CTxIndex txindex; + return ReadDiskTx(outpoint.hash, tx, txindex); +} + +bool CTxDB::WriteBlockIndex(const CDiskBlockIndex& blockindex) +{ + return Write(make_pair(string("blockindex"), blockindex.GetBlockHash()), blockindex); +} + +bool CTxDB::EraseBlockIndex(uint256 hash) +{ + return Erase(make_pair(string("blockindex"), hash)); +} + +bool CTxDB::ReadHashBestChain(uint256& hashBestChain) +{ + return Read(string("hashBestChain"), hashBestChain); +} + +bool CTxDB::WriteHashBestChain(uint256 hashBestChain) +{ + return Write(string("hashBestChain"), hashBestChain); +} + +CBlockIndex* InsertBlockIndex(uint256 hash) +{ + if (hash == 0) + return NULL; + + // Return existing + map::iterator mi = mapBlockIndex.find(hash); + if (mi != mapBlockIndex.end()) + return (*mi).second; + + // Create new + CBlockIndex* pindexNew = new CBlockIndex(); + if (!pindexNew) + throw runtime_error("LoadBlockIndex() : new CBlockIndex failed"); + mi = mapBlockIndex.insert(make_pair(hash, pindexNew)).first; + pindexNew->phashBlock = &((*mi).first); + + return pindexNew; +} + +bool CTxDB::LoadBlockIndex() +{ + // Get cursor + Dbc* pcursor = GetCursor(); + if (!pcursor) + return false; + + unsigned int fFlags = DB_SET_RANGE; + loop + { + // Read next record + CDataStream ssKey; + if (fFlags == DB_SET_RANGE) + ssKey << make_pair(string("blockindex"), uint256(0)); + CDataStream ssValue; + int ret = ReadAtCursor(pcursor, ssKey, ssValue, fFlags); + fFlags = DB_NEXT; + if (ret == DB_NOTFOUND) + break; + else if (ret != 0) + return false; + + // Unserialize + string strType; + ssKey >> strType; + if (strType == "blockindex") + { + CDiskBlockIndex diskindex; + ssValue >> diskindex; + + // Construct block index object + CBlockIndex* pindexNew = InsertBlockIndex(diskindex.GetBlockHash()); + pindexNew->pprev = InsertBlockIndex(diskindex.hashPrev); + pindexNew->pnext = InsertBlockIndex(diskindex.hashNext); + pindexNew->nFile = diskindex.nFile; + pindexNew->nBlockPos = diskindex.nBlockPos; + pindexNew->nHeight = diskindex.nHeight; + pindexNew->nVersion = diskindex.nVersion; + pindexNew->hashMerkleRoot = diskindex.hashMerkleRoot; + pindexNew->nTime = diskindex.nTime; + pindexNew->nBits = diskindex.nBits; + pindexNew->nNonce = diskindex.nNonce; + + // Watch for genesis block and best block + if (pindexGenesisBlock == NULL && diskindex.GetBlockHash() == hashGenesisBlock) + pindexGenesisBlock = pindexNew; + } + else + { + break; + } + } + + if (!ReadHashBestChain(hashBestChain)) + { + if (pindexGenesisBlock == NULL) + return true; + return error("CTxDB::LoadBlockIndex() : hashBestChain not found\n"); + } + + if (!mapBlockIndex.count(hashBestChain)) + return error("CTxDB::LoadBlockIndex() : blockindex for hashBestChain not found\n"); + pindexBest = mapBlockIndex[hashBestChain]; + nBestHeight = pindexBest->nHeight; + printf("LoadBlockIndex(): hashBestChain=%s height=%d\n", hashBestChain.ToString().substr(0,14).c_str(), nBestHeight); + + return true; +} + + + + + +// +// CAddrDB +// + +bool CAddrDB::WriteAddress(const CAddress& addr) +{ + return Write(make_pair(string("addr"), addr.GetKey()), addr); +} + +bool CAddrDB::LoadAddresses() +{ + CRITICAL_BLOCK(cs_mapIRCAddresses) + CRITICAL_BLOCK(cs_mapAddresses) + { + // Load user provided addresses + CAutoFile filein = fopen("addr.txt", "rt"); + if (filein) + { + try + { + char psz[1000]; + while (fgets(psz, sizeof(psz), filein)) + { + CAddress addr(psz, NODE_NETWORK); + if (addr.ip != 0) + { + AddAddress(*this, addr); + mapIRCAddresses.insert(make_pair(addr.GetKey(), addr)); + } + } + } + catch (...) { } + } + + // Get cursor + Dbc* pcursor = GetCursor(); + if (!pcursor) + return false; + + loop + { + // Read next record + CDataStream ssKey; + CDataStream ssValue; + int ret = ReadAtCursor(pcursor, ssKey, ssValue); + if (ret == DB_NOTFOUND) + break; + else if (ret != 0) + return false; + + // Unserialize + string strType; + ssKey >> strType; + if (strType == "addr") + { + CAddress addr; + ssValue >> addr; + mapAddresses.insert(make_pair(addr.GetKey(), addr)); + } + } + + //// debug print + printf("mapAddresses:\n"); + foreach(const PAIRTYPE(vector, CAddress)& item, mapAddresses) + item.second.print(); + printf("-----\n"); + + // Fix for possible bug that manifests in mapAddresses.count in irc.cpp, + // just need to call count here and it doesn't happen there. The bug was the + // pack pragma in irc.cpp and has been fixed, but I'm not in a hurry to delete this. + mapAddresses.count(vector(18)); + } + + return true; +} + +bool LoadAddresses() +{ + return CAddrDB("cr+").LoadAddresses(); +} + + + + +// +// CReviewDB +// + +bool CReviewDB::ReadReviews(uint256 hash, vector& vReviews) +{ + vReviews.size(); // msvc workaround, just need to do anything with vReviews + return Read(make_pair(string("reviews"), hash), vReviews); +} + +bool CReviewDB::WriteReviews(uint256 hash, const vector& vReviews) +{ + return Write(make_pair(string("reviews"), hash), vReviews); +} + + + + + + + +// +// CWalletDB +// + +bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) +{ + vchDefaultKeyRet.clear(); + + //// todo: shouldn't we catch exceptions and try to recover and continue? + CRITICAL_BLOCK(cs_mapKeys) + CRITICAL_BLOCK(cs_mapWallet) + { + // Get cursor + Dbc* pcursor = GetCursor(); + if (!pcursor) + return false; + + loop + { + // Read next record + CDataStream ssKey; + CDataStream ssValue; + int ret = ReadAtCursor(pcursor, ssKey, ssValue); + if (ret == DB_NOTFOUND) + break; + else if (ret != 0) + return false; + + // Unserialize + // Taking advantage of the fact that pair serialization + // is just the two items serialized one after the other + string strType; + ssKey >> strType; + if (strType == "name") + { + string strAddress; + ssKey >> strAddress; + ssValue >> mapAddressBook[strAddress]; + } + else if (strType == "tx") + { + uint256 hash; + ssKey >> hash; + CWalletTx& wtx = mapWallet[hash]; + ssValue >> wtx; + + if (wtx.GetHash() != hash) + printf("Error in wallet.dat, hash mismatch\n"); + + //// debug print + //printf("LoadWallet %s\n", wtx.GetHash().ToString().c_str()); + //printf(" %12I64d %s %s %s\n", + // wtx.vout[0].nValue, + // DateTimeStr(wtx.nTime).c_str(), + // wtx.hashBlock.ToString().substr(0,14).c_str(), + // wtx.mapValue["message"].c_str()); + } + else if (strType == "key") + { + vector vchPubKey; + ssKey >> vchPubKey; + CPrivKey vchPrivKey; + ssValue >> vchPrivKey; + + mapKeys[vchPubKey] = vchPrivKey; + mapPubKeys[Hash160(vchPubKey)] = vchPubKey; + } + else if (strType == "defaultkey") + { + ssValue >> vchDefaultKeyRet; + } + else if (strType == "setting") /// or settings or option or options or config? + { + string strKey; + ssKey >> strKey; + if (strKey == "fGenerateBitcoins") ssValue >> fGenerateBitcoins; + if (strKey == "nTransactionFee") ssValue >> nTransactionFee; + if (strKey == "addrIncoming") ssValue >> addrIncoming; + } + } + } + + printf("fGenerateBitcoins = %d\n", fGenerateBitcoins); + printf("nTransactionFee = %I64d\n", nTransactionFee); + printf("addrIncoming = %s\n", addrIncoming.ToString().c_str()); + + return true; +} + +bool LoadWallet() +{ + vector vchDefaultKey; + if (!CWalletDB("cr").LoadWallet(vchDefaultKey)) + return false; + + if (mapKeys.count(vchDefaultKey)) + { + // Set keyUser + keyUser.SetPubKey(vchDefaultKey); + keyUser.SetPrivKey(mapKeys[vchDefaultKey]); + } + else + { + // Create new keyUser and set as default key + RandAddSeed(true); + keyUser.MakeNewKey(); + if (!AddKey(keyUser)) + return false; + if (!SetAddressBookName(PubKeyToAddress(keyUser.GetPubKey()), "Your Address")) + return false; + CWalletDB().WriteDefaultKey(keyUser.GetPubKey()); + } + + return true; +} diff --git a/db.h b/db.h new file mode 100644 index 00000000..1139f768 --- /dev/null +++ b/db.h @@ -0,0 +1,420 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include +class CTransaction; +class CTxIndex; +class CDiskBlockIndex; +class CDiskTxPos; +class COutPoint; +class CUser; +class CReview; +class CAddress; +class CWalletTx; + +extern map mapAddressBook; +extern bool fClient; + + +extern DbEnv dbenv; +extern void DBFlush(bool fShutdown); + + + + +class CDB +{ +protected: + Db* pdb; + string strFile; + vector vTxn; + + explicit CDB(const char* pszFile, const char* pszMode="r+", bool fTxn=false); + ~CDB() { Close(); } +public: + void Close(); +private: + CDB(const CDB&); + void operator=(const CDB&); + +protected: + template + bool Read(const K& key, T& value) + { + if (!pdb) + return false; + + // Key + CDataStream ssKey(SER_DISK); + ssKey.reserve(1000); + ssKey << key; + Dbt datKey(&ssKey[0], ssKey.size()); + + // Read + Dbt datValue; + datValue.set_flags(DB_DBT_MALLOC); + int ret = pdb->get(GetTxn(), &datKey, &datValue, 0); + memset(datKey.get_data(), 0, datKey.get_size()); + if (datValue.get_data() == NULL) + return false; + + // Unserialize value + CDataStream ssValue((char*)datValue.get_data(), (char*)datValue.get_data() + datValue.get_size(), SER_DISK); + ssValue >> value; + + // Clear and free memory + memset(datValue.get_data(), 0, datValue.get_size()); + free(datValue.get_data()); + return (ret == 0); + } + + template + bool Write(const K& key, const T& value, bool fOverwrite=true) + { + if (!pdb) + return false; + + // Key + CDataStream ssKey(SER_DISK); + ssKey.reserve(1000); + ssKey << key; + Dbt datKey(&ssKey[0], ssKey.size()); + + // Value + CDataStream ssValue(SER_DISK); + ssValue.reserve(10000); + ssValue << value; + Dbt datValue(&ssValue[0], ssValue.size()); + + // Write + int ret = pdb->put(GetTxn(), &datKey, &datValue, (fOverwrite ? 0 : DB_NOOVERWRITE)); + + // Clear memory in case it was a private key + memset(datKey.get_data(), 0, datKey.get_size()); + memset(datValue.get_data(), 0, datValue.get_size()); + return (ret == 0); + } + + template + bool Erase(const K& key) + { + if (!pdb) + return false; + + // Key + CDataStream ssKey(SER_DISK); + ssKey.reserve(1000); + ssKey << key; + Dbt datKey(&ssKey[0], ssKey.size()); + + // Erase + int ret = pdb->del(GetTxn(), &datKey, 0); + + // Clear memory + memset(datKey.get_data(), 0, datKey.get_size()); + return (ret == 0 || ret == DB_NOTFOUND); + } + + template + bool Exists(const K& key) + { + if (!pdb) + return false; + + // Key + CDataStream ssKey(SER_DISK); + ssKey.reserve(1000); + ssKey << key; + Dbt datKey(&ssKey[0], ssKey.size()); + + // Exists + int ret = pdb->exists(GetTxn(), &datKey, 0); + + // Clear memory + memset(datKey.get_data(), 0, datKey.get_size()); + return (ret == 0); + } + + Dbc* GetCursor() + { + if (!pdb) + return NULL; + Dbc* pcursor = NULL; + int ret = pdb->cursor(NULL, &pcursor, 0); + if (ret != 0) + return NULL; + return pcursor; + } + + int ReadAtCursor(Dbc* pcursor, CDataStream& ssKey, CDataStream& ssValue, unsigned int fFlags=DB_NEXT) + { + // Read at cursor + Dbt datKey; + if (fFlags == DB_SET || fFlags == DB_SET_RANGE || fFlags == DB_GET_BOTH || fFlags == DB_GET_BOTH_RANGE) + { + datKey.set_data(&ssKey[0]); + datKey.set_size(ssKey.size()); + } + Dbt datValue; + if (fFlags == DB_GET_BOTH || fFlags == DB_GET_BOTH_RANGE) + { + datValue.set_data(&ssValue[0]); + datValue.set_size(ssValue.size()); + } + datKey.set_flags(DB_DBT_MALLOC); + datValue.set_flags(DB_DBT_MALLOC); + int ret = pcursor->get(&datKey, &datValue, fFlags); + if (ret != 0) + return ret; + else if (datKey.get_data() == NULL || datValue.get_data() == NULL) + return 99999; + + // Convert to streams + ssKey.SetType(SER_DISK); + ssKey.clear(); + ssKey.write((char*)datKey.get_data(), datKey.get_size()); + ssValue.SetType(SER_DISK); + ssValue.clear(); + ssValue.write((char*)datValue.get_data(), datValue.get_size()); + + // Clear and free memory + memset(datKey.get_data(), 0, datKey.get_size()); + memset(datValue.get_data(), 0, datValue.get_size()); + free(datKey.get_data()); + free(datValue.get_data()); + return 0; + } + + DbTxn* GetTxn() + { + if (!vTxn.empty()) + return vTxn.back(); + else + return NULL; + } + +public: + bool TxnBegin() + { + if (!pdb) + return false; + DbTxn* ptxn = NULL; + int ret = dbenv.txn_begin(GetTxn(), &ptxn, 0); + if (!ptxn || ret != 0) + return false; + vTxn.push_back(ptxn); + return true; + } + + bool TxnCommit() + { + if (!pdb) + return false; + if (vTxn.empty()) + return false; + int ret = vTxn.back()->commit(0); + vTxn.pop_back(); + return (ret == 0); + } + + bool TxnAbort() + { + if (!pdb) + return false; + if (vTxn.empty()) + return false; + int ret = vTxn.back()->abort(); + vTxn.pop_back(); + return (ret == 0); + } + + bool ReadVersion(int& nVersion) + { + nVersion = 0; + return Read(string("version"), nVersion); + } + + bool WriteVersion(int nVersion) + { + return Write(string("version"), nVersion); + } +}; + + + + + + + + +class CTxDB : public CDB +{ +public: + CTxDB(const char* pszMode="r+", bool fTxn=false) : CDB(!fClient ? "blkindex.dat" : NULL, pszMode, fTxn) { } +private: + CTxDB(const CTxDB&); + void operator=(const CTxDB&); +public: + bool ReadTxIndex(uint256 hash, CTxIndex& txindex); + bool UpdateTxIndex(uint256 hash, const CTxIndex& txindex); + bool AddTxIndex(const CTransaction& tx, const CDiskTxPos& pos, int nHeight); + bool EraseTxIndex(const CTransaction& tx); + bool ContainsTx(uint256 hash); + bool ReadOwnerTxes(uint160 hash160, int nHeight, vector& vtx); + bool ReadDiskTx(uint256 hash, CTransaction& tx, CTxIndex& txindex); + bool ReadDiskTx(uint256 hash, CTransaction& tx); + bool ReadDiskTx(COutPoint outpoint, CTransaction& tx, CTxIndex& txindex); + bool ReadDiskTx(COutPoint outpoint, CTransaction& tx); + bool WriteBlockIndex(const CDiskBlockIndex& blockindex); + bool EraseBlockIndex(uint256 hash); + bool ReadHashBestChain(uint256& hashBestChain); + bool WriteHashBestChain(uint256 hashBestChain); + bool LoadBlockIndex(); +}; + + + + + +class CReviewDB : public CDB +{ +public: + CReviewDB(const char* pszMode="r+", bool fTxn=false) : CDB("reviews.dat", pszMode, fTxn) { } +private: + CReviewDB(const CReviewDB&); + void operator=(const CReviewDB&); +public: + bool ReadUser(uint256 hash, CUser& user) + { + return Read(make_pair(string("user"), hash), user); + } + + bool WriteUser(uint256 hash, const CUser& user) + { + return Write(make_pair(string("user"), hash), user); + } + + bool ReadReviews(uint256 hash, vector& vReviews); + bool WriteReviews(uint256 hash, const vector& vReviews); +}; + + + + + +class CMarketDB : public CDB +{ +public: + CMarketDB(const char* pszMode="r+", bool fTxn=false) : CDB("market.dat", pszMode, fTxn) { } +private: + CMarketDB(const CMarketDB&); + void operator=(const CMarketDB&); +}; + + + + + +class CAddrDB : public CDB +{ +public: + CAddrDB(const char* pszMode="r+", bool fTxn=false) : CDB("addr.dat", pszMode, fTxn) { } +private: + CAddrDB(const CAddrDB&); + void operator=(const CAddrDB&); +public: + bool WriteAddress(const CAddress& addr); + bool LoadAddresses(); +}; + +bool LoadAddresses(); + + + + + +class CWalletDB : public CDB +{ +public: + CWalletDB(const char* pszMode="r+", bool fTxn=false) : CDB("wallet.dat", pszMode, fTxn) { } +private: + CWalletDB(const CWalletDB&); + void operator=(const CWalletDB&); +public: + bool ReadName(const string& strAddress, string& strName) + { + strName = ""; + return Read(make_pair(string("name"), strAddress), strName); + } + + bool WriteName(const string& strAddress, const string& strName) + { + mapAddressBook[strAddress] = strName; + return Write(make_pair(string("name"), strAddress), strName); + } + + bool EraseName(const string& strAddress) + { + mapAddressBook.erase(strAddress); + return Erase(make_pair(string("name"), strAddress)); + } + + bool ReadTx(uint256 hash, CWalletTx& wtx) + { + return Read(make_pair(string("tx"), hash), wtx); + } + + bool WriteTx(uint256 hash, const CWalletTx& wtx) + { + return Write(make_pair(string("tx"), hash), wtx); + } + + bool EraseTx(uint256 hash) + { + return Erase(make_pair(string("tx"), hash)); + } + + bool ReadKey(const vector& vchPubKey, CPrivKey& vchPrivKey) + { + vchPrivKey.clear(); + return Read(make_pair(string("key"), vchPubKey), vchPrivKey); + } + + bool WriteKey(const vector& vchPubKey, const CPrivKey& vchPrivKey) + { + return Write(make_pair(string("key"), vchPubKey), vchPrivKey, false); + } + + bool ReadDefaultKey(vector& vchPubKey) + { + vchPubKey.clear(); + return Read(string("defaultkey"), vchPubKey); + } + + bool WriteDefaultKey(const vector& vchPubKey) + { + return Write(string("defaultkey"), vchPubKey); + } + + template + bool ReadSetting(const string& strKey, T& value) + { + return Read(make_pair(string("setting"), strKey), value); + } + + template + bool WriteSetting(const string& strKey, const T& value) + { + return Write(make_pair(string("setting"), strKey), value); + } + + bool LoadWallet(vector& vchDefaultKeyRet); +}; + +bool LoadWallet(); + +inline bool SetAddressBookName(const string& strAddress, const string& strName) +{ + return CWalletDB().WriteName(strAddress, strName); +} diff --git a/headers.h b/headers.h new file mode 100644 index 00000000..7bd68a19 --- /dev/null +++ b/headers.h @@ -0,0 +1,71 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#ifdef _MSC_VER +#pragma warning(disable:4786) +#pragma warning(disable:4804) +#pragma warning(disable:4717) +#endif +#ifdef _WIN32_WINNT +#undef _WIN32_WINNT +#endif +#define _WIN32_WINNT 0x0400 +#define WIN32_LEAN_AND_MEAN 1 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#define BOUNDSCHECK 1 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#pragma hdrstop +using namespace std; +using namespace boost; + + + +#include "serialize.h" +#include "uint256.h" +#include "util.h" +#include "key.h" +#include "bignum.h" +#include "base58.h" +#include "script.h" +#include "db.h" +#include "net.h" +#include "irc.h" +#include "main.h" +#include "market.h" +#include "uibase.h" +#include "ui.h" diff --git a/irc.cpp b/irc.cpp new file mode 100644 index 00000000..3518df24 --- /dev/null +++ b/irc.cpp @@ -0,0 +1,314 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" + + +map, CAddress> mapIRCAddresses; +CCriticalSection cs_mapIRCAddresses; + + + + +#pragma pack(push, 1) +struct ircaddr +{ + int ip; + short port; +}; +#pragma pack(pop) + +string EncodeAddress(const CAddress& addr) +{ + struct ircaddr tmp; + tmp.ip = addr.ip; + tmp.port = addr.port; + + vector vch(UBEGIN(tmp), UEND(tmp)); + return string("u") + EncodeBase58Check(vch); +} + +bool DecodeAddress(string str, CAddress& addr) +{ + vector vch; + if (!DecodeBase58Check(str.substr(1), vch)) + return false; + + struct ircaddr tmp; + if (vch.size() != sizeof(tmp)) + return false; + memcpy(&tmp, &vch[0], sizeof(tmp)); + + addr = CAddress(tmp.ip, tmp.port); + return true; +} + + + + + + +static bool Send(SOCKET hSocket, const char* pszSend) +{ + if (strstr(pszSend, "PONG") != pszSend) + printf("SENDING: %s\n", pszSend); + const char* psz = pszSend; + const char* pszEnd = psz + strlen(psz); + while (psz < pszEnd) + { + int ret = send(hSocket, psz, pszEnd - psz, 0); + if (ret < 0) + return false; + psz += ret; + } + return true; +} + +bool RecvLine(SOCKET hSocket, string& strLine) +{ + strLine = ""; + loop + { + char c; + int nBytes = recv(hSocket, &c, 1, 0); + if (nBytes > 0) + { + if (c == '\n') + continue; + if (c == '\r') + return true; + strLine += c; + } + else if (nBytes <= 0) + { + if (!strLine.empty()) + return true; + // socket closed + printf("IRC socket closed\n"); + return false; + } + else + { + // socket error + int nErr = WSAGetLastError(); + if (nErr != WSAEMSGSIZE && nErr != WSAEINTR && nErr != WSAEINPROGRESS) + { + printf("IRC recv failed: %d\n", nErr); + return false; + } + } + } +} + +bool RecvLineIRC(SOCKET hSocket, string& strLine) +{ + loop + { + bool fRet = RecvLine(hSocket, strLine); + if (fRet) + { + if (fShutdown) + return false; + vector vWords; + ParseString(strLine, ' ', vWords); + if (vWords[0] == "PING") + { + strLine[1] = 'O'; + strLine += '\r'; + Send(hSocket, strLine.c_str()); + continue; + } + } + return fRet; + } +} + +bool RecvUntil(SOCKET hSocket, const char* psz1, const char* psz2=NULL, const char* psz3=NULL) +{ + loop + { + string strLine; + if (!RecvLineIRC(hSocket, strLine)) + return false; + printf("IRC %s\n", strLine.c_str()); + if (psz1 && strLine.find(psz1) != -1) + return true; + if (psz2 && strLine.find(psz2) != -1) + return true; + if (psz3 && strLine.find(psz3) != -1) + return true; + } +} + +bool Wait(int nSeconds) +{ + if (fShutdown) + return false; + printf("Waiting %d seconds to reconnect to IRC\n", nSeconds); + for (int i = 0; i < nSeconds; i++) + { + if (fShutdown) + return false; + Sleep(1000); + } + return true; +} + + + +void ThreadIRCSeed(void* parg) +{ + SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL); + int nErrorWait = 10; + int nRetryWait = 10; + + while (!fShutdown) + { + CAddress addrConnect("216.155.130.130:6667"); + struct hostent* phostent = gethostbyname("chat.freenode.net"); + if (phostent && phostent->h_addr_list && phostent->h_addr_list[0]) + addrConnect = CAddress(*(u_long*)phostent->h_addr_list[0], htons(6667)); + + SOCKET hSocket; + if (!ConnectSocket(addrConnect, hSocket)) + { + printf("IRC connect failed\n"); + nErrorWait = nErrorWait * 11 / 10; + if (Wait(nErrorWait += 60)) + continue; + else + return; + } + + if (!RecvUntil(hSocket, "Found your hostname", "using your IP address instead", "Couldn't look up your hostname")) + { + closesocket(hSocket); + nErrorWait = nErrorWait * 11 / 10; + if (Wait(nErrorWait += 60)) + continue; + else + return; + } + + string strMyName = EncodeAddress(addrLocalHost); + + if (!addrLocalHost.IsRoutable()) + strMyName = strprintf("x%u", GetRand(1000000000)); + + + Send(hSocket, strprintf("NICK %s\r", strMyName.c_str()).c_str()); + Send(hSocket, strprintf("USER %s 8 * : %s\r", strMyName.c_str(), strMyName.c_str()).c_str()); + + if (!RecvUntil(hSocket, " 004 ")) + { + closesocket(hSocket); + nErrorWait = nErrorWait * 11 / 10; + if (Wait(nErrorWait += 60)) + continue; + else + return; + } + Sleep(500); + + Send(hSocket, "JOIN #bitcoin\r"); + Send(hSocket, "WHO #bitcoin\r"); + + int64 nStart = GetTime(); + string strLine; + while (!fShutdown && RecvLineIRC(hSocket, strLine)) + { + if (strLine.empty() || strLine.size() > 900 || strLine[0] != ':') + continue; + printf("IRC %s\n", strLine.c_str()); + + vector vWords; + ParseString(strLine, ' ', vWords); + if (vWords.size() < 2) + continue; + + char pszName[10000]; + pszName[0] = '\0'; + + if (vWords[1] == "352" && vWords.size() >= 8) + { + // index 7 is limited to 16 characters + // could get full length name at index 10, but would be different from join messages + strcpy(pszName, vWords[7].c_str()); + printf("GOT WHO: [%s] ", pszName); + } + + if (vWords[1] == "JOIN" && vWords[0].size() > 1) + { + // :username!username@50000007.F000000B.90000002.IP JOIN :#channelname + strcpy(pszName, vWords[0].c_str() + 1); + if (strchr(pszName, '!')) + *strchr(pszName, '!') = '\0'; + printf("GOT JOIN: [%s] ", pszName); + } + + if (pszName[0] == 'u') + { + CAddress addr; + if (DecodeAddress(pszName, addr)) + { + CAddrDB addrdb; + if (AddAddress(addrdb, addr)) + printf("new "); + else + { + // make it try connecting again + CRITICAL_BLOCK(cs_mapAddresses) + if (mapAddresses.count(addr.GetKey())) + mapAddresses[addr.GetKey()].nLastFailed = 0; + } + addr.print(); + + CRITICAL_BLOCK(cs_mapIRCAddresses) + mapIRCAddresses.insert(make_pair(addr.GetKey(), addr)); + } + else + { + printf("decode failed\n"); + } + } + } + closesocket(hSocket); + + if (GetTime() - nStart > 20 * 60) + { + nErrorWait /= 3; + nRetryWait /= 3; + } + + nRetryWait = nRetryWait * 11 / 10; + if (!Wait(nRetryWait += 60)) + return; + } +} + + + + + + + + + + +#ifdef TEST +int main(int argc, char *argv[]) +{ + WSADATA wsadata; + if (WSAStartup(MAKEWORD(2,2), &wsadata) != NO_ERROR) + { + printf("Error at WSAStartup()\n"); + return false; + } + + ThreadIRCSeed(NULL); + + WSACleanup(); + return 0; +} +#endif diff --git a/irc.h b/irc.h new file mode 100644 index 00000000..91c3ffe6 --- /dev/null +++ b/irc.h @@ -0,0 +1,10 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +extern bool RecvLine(SOCKET hSocket, string& strLine); +extern void ThreadIRCSeed(void* parg); +extern bool fRestartIRCSeed; + +extern map, CAddress> mapIRCAddresses; +extern CCriticalSection cs_mapIRCAddresses; diff --git a/key.h b/key.h new file mode 100644 index 00000000..8b0b54e4 --- /dev/null +++ b/key.h @@ -0,0 +1,156 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + + +// secp160k1 +// const unsigned int PRIVATE_KEY_SIZE = 192; +// const unsigned int PUBLIC_KEY_SIZE = 41; +// const unsigned int SIGNATURE_SIZE = 48; +// +// secp192k1 +// const unsigned int PRIVATE_KEY_SIZE = 222; +// const unsigned int PUBLIC_KEY_SIZE = 49; +// const unsigned int SIGNATURE_SIZE = 57; +// +// secp224k1 +// const unsigned int PRIVATE_KEY_SIZE = 250; +// const unsigned int PUBLIC_KEY_SIZE = 57; +// const unsigned int SIGNATURE_SIZE = 66; +// +// secp256k1: +// const unsigned int PRIVATE_KEY_SIZE = 279; +// const unsigned int PUBLIC_KEY_SIZE = 65; +// const unsigned int SIGNATURE_SIZE = 72; +// +// see www.keylength.com +// script supports up to 75 for single byte push + + + +class key_error : public std::runtime_error +{ +public: + explicit key_error(const std::string& str) : std::runtime_error(str) {} +}; + + +// secure_allocator is defined is serialize.h +typedef vector > CPrivKey; + + + +class CKey +{ +protected: + EC_KEY* pkey; + +public: + CKey() + { + pkey = EC_KEY_new_by_curve_name(NID_secp256k1); + if (pkey == NULL) + throw key_error("CKey::CKey() : EC_KEY_new_by_curve_name failed"); + } + + CKey(const CKey& b) + { + pkey = EC_KEY_dup(b.pkey); + if (pkey == NULL) + throw key_error("CKey::CKey(const CKey&) : EC_KEY_dup failed"); + } + + CKey& operator=(const CKey& b) + { + if (!EC_KEY_copy(pkey, b.pkey)) + throw key_error("CKey::operator=(const CKey&) : EC_KEY_copy failed"); + return (*this); + } + + ~CKey() + { + EC_KEY_free(pkey); + } + + void MakeNewKey() + { + if (!EC_KEY_generate_key(pkey)) + throw key_error("CKey::MakeNewKey() : EC_KEY_generate_key failed"); + } + + bool SetPrivKey(const CPrivKey& vchPrivKey) + { + const unsigned char* pbegin = &vchPrivKey[0]; + if (!d2i_ECPrivateKey(&pkey, &pbegin, vchPrivKey.size())) + return false; + return true; + } + + CPrivKey GetPrivKey() const + { + unsigned int nSize = i2d_ECPrivateKey(pkey, NULL); + if (!nSize) + throw key_error("CKey::GetPrivKey() : i2d_ECPrivateKey failed"); + CPrivKey vchPrivKey(nSize, 0); + unsigned char* pbegin = &vchPrivKey[0]; + if (i2d_ECPrivateKey(pkey, &pbegin) != nSize) + throw key_error("CKey::GetPrivKey() : i2d_ECPrivateKey returned unexpected size"); + return vchPrivKey; + } + + bool SetPubKey(const vector& vchPubKey) + { + const unsigned char* pbegin = &vchPubKey[0]; + if (!o2i_ECPublicKey(&pkey, &pbegin, vchPubKey.size())) + return false; + return true; + } + + vector GetPubKey() const + { + unsigned int nSize = i2o_ECPublicKey(pkey, NULL); + if (!nSize) + throw key_error("CKey::GetPubKey() : i2o_ECPublicKey failed"); + vector vchPubKey(nSize, 0); + unsigned char* pbegin = &vchPubKey[0]; + if (i2o_ECPublicKey(pkey, &pbegin) != nSize) + throw key_error("CKey::GetPubKey() : i2o_ECPublicKey returned unexpected size"); + return vchPubKey; + } + + bool Sign(uint256 hash, vector& vchSig) + { + vchSig.clear(); + unsigned char pchSig[10000]; + unsigned int nSize = 0; + if (!ECDSA_sign(0, (unsigned char*)&hash, sizeof(hash), pchSig, &nSize, pkey)) + return false; + vchSig.resize(nSize); + memcpy(&vchSig[0], pchSig, nSize); + return true; + } + + bool Verify(uint256 hash, const vector& vchSig) + { + // -1 = error, 0 = bad sig, 1 = good + if (ECDSA_verify(0, (unsigned char*)&hash, sizeof(hash), &vchSig[0], vchSig.size(), pkey) != 1) + return false; + return true; + } + + static bool Sign(const CPrivKey& vchPrivKey, uint256 hash, vector& vchSig) + { + CKey key; + if (!key.SetPrivKey(vchPrivKey)) + return false; + return key.Sign(hash, vchSig); + } + + static bool Verify(const vector& vchPubKey, uint256 hash, const vector& vchSig) + { + CKey key; + if (!key.SetPubKey(vchPubKey)) + return false; + return key.Verify(hash, vchSig); + } +}; diff --git a/libeay32.dll b/libeay32.dll new file mode 100644 index 0000000000000000000000000000000000000000..3bc745c4c0f4751cab5195d9fbdf19ea42113016 GIT binary patch literal 1306630 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

v_FCS|;eZTiM@xO9LEn)+yikJAf}4kzOb< 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�z=(6g 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

x8tM1U&vI%f85M-kh&a< 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

g+4142`ZBK&MBTSqQdq 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Ν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 )kSS+GRQxR@w-N8N-a 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 z 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%0D1655rzvs+Oyh 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�FamToW_y{zIxy}Wcg$2rD}8-LinOxJC2xMys9ZK5Un)6mb*LD_Zz7w+RA z$=G-Xcw6_fKi3s36qdjXm%S*`r&Q%i*hG#j^i-81b=}zP}R}PWX=C4h(#M8vD z?CJhOhyGwsS7}e_ErIoojRoH~T}NJUd%RkW543u7eyl9no*F`e*Ht} z&>!sAFWRr^c%QiKSuOl>Pi{UFu7XXHZ9JtrK{A3oMK-7S_2^ZBs+pqkE_~fdw6d+E zk#Pe4oUpUdp+7hpTkpobJhpOb)A+(gf%7)Jh@RSfG*-eUb$X$vsuX#Oe-&(P=Ba-K zG)Q|v-v^o6;1;ZHK1U06jyaXn-KMeQCfDeZLWE8LJ1Lx;p|h7Iua%&ljBlg%GM(;% zYiF_Vt~E)gB-$fLH;}YXjAMMSmnnN=xa#3T zJipo5y9xj7kvG6^_N)~Y|EE9rtbR_9)Awx~Kf{mOjmIgzS5Y~A?9}Np#rr1FJCl^B z#b#rw_)yqH*oJ^}GKeckEl5(3OF=S%+yUZD_OcyD- z3L@qo6U8~}n%NhGTX1GO6X4me4>eJ2Jo}B+QN_GbJ7s*~y7$W@?)ySGy_J<$L#KO? zErNK*?}GG!O@e(d;ssSS0yasu@oFATH48EsBrC{F5U0+JJRQG>FkRUIBLN?SPd=lNPnAs##`P5Z<*hgLtmT#8M1FD3d!*yNxO?xfyw61` zZ?~NEeJ`~p*$Ji6IzCj<0U#McjskHmH2dOokfTwm2xcq@c+EqO7k z`8eMNn=spW0#ATMHQ^jn^^ou9z)Hnmx5r=G{LjAotHBQpzmPpx!RJ?l^wek^+&DUh zUL%&F&05_fAA&4?oM#(CpM$vUSCQT!n=S9hti?72oV`F2f*b{sx{h;$Y)%Eq2yz}s z)+bw#i8B=>dOhbeksCo`H<;vJDm9x4JoUa|9lVjXxG$6!)E7^*$IBI{GQHV(7`J@2 z^RV=IkLx^?>lzZ|Jp5GmS5wjT#tZN(?0bRbi?y*|%7*;913@dk*GqGs7OTgf$D%D5 zUuo^v<#v1gJX~k^<-H{*pN3wUd%wSmw1V`+|tQVQNMDZ{;GYa?@3K@#~OQlvfLA1xHp&fo&8caOtO!S4&p+7h?hwJ#zcmJ*MID5j2;4{jLQmf^G zuu1ziDl?A(iKf|i(xY6>z3A6Nz@c1CylfK6)zlJ`P_E{dnS^pR_=ZW&q&WLTsZ;i* z-ZX6}dn0d|q#C7CJ~6V_^MWF_5rcUjF%sK4dy~3%#-WoscroD*?ww~Don zg%`p1lwYN?m!8Azyzkq10{4Qrns8>SddT-Z5Jq`+yRCf>Zp)du3%jg;vhNeL?{qeS zoR>SoXW>O;Mf1MD1DhDzcnbdk$qKT?gNVilW_)%4Nej{!B>16ea}Y>Ykdw&fBhzLQ zNKBBcsnkl-WLeiD}d2FBmC;>I*f=+)>fE4L(%7#By#-$AS=6iJKJt`T zaUOXR#9ifkDn_~zW#5$z`92c=*h>A=|FnJYhyN5Dr?g^y7}dTH6*}|>`+h}%?-PvU z6ta7Rd~e1tvROsZU>gF?S0GtU*mqSA`M$SjWqz>lIgVPmX+GNJU*UqX)pEFlXv9t97JY?(53a>3mQL*a#P zk5|XhAi=MEzlvlFd#Y^6QyS~i(9U@p?*%q~&d2uK{5J(s+SAg<4o?YB>)jf>;=*+H z4EPmlfnUmo{MyoPv%m5TFWmh7B+dHekL4Fdo2w^}uaZw1n1nWmThL}@Lv5CGDs?FL zMO(G`UD4(XYo{D2Z{@j^+RX8x1~@l@M4It}Z0?3l)VEPSJOq;XTIHc4PlLq&Ym(PM z+?+{11c`lPlK+52)|q4jkqt~ZpH&MfhAX#moUzm>z1cCOd@fjvwj!T1I-lu$o!ofj zt`c#!zsbJc!|+|id3TWD6Xsd913*F>m8Zqlz3jWPA>T{yyR}vzgMmCmG@8#@HJkJP z8@BSE-}Oi(!*y8?V%LAkL4R&z05nmE)$4cif+t@RR@hb~V*+ zMXBV^Y(o{z2XQf*dv~FDH9rcH+fosA%=j`$cq@~9Nc9Mk14#?A^&{BBYQkBkJf+_G z`?k0$?J4pHk2oElK%D=jf3_y;bmRk<>+<{yXI+;N_h4OhMP>2#r0WcKDo+?$K{Z4> z_|cHNH?MN`RN0WH6ToS|lHb_VOsn#+$g05Kr|c0jy{MkhSa3YJ8~P&MOq;1?6|l*5 zHhY<#nGxR3BqxDH1vwiexINpj?@Eh&zaA52!JO&uPg~-<+k$n1%NZW%OKymbdwF0; z#|N*ptPlsw1n51NDXvF9DKzT)Rcuu9lR{m#arjB0i99*#^eEMyvf`s^rud9UdvO}p z*jketEs2lztC=T-4i9mB==fqS<}2;ll;&9?&thby3JF~ z%TYR#O=xqb1#MO~zfGG@)oresGQmDHcdn&2^PJ#$@+TT!i_HD|BOuup_@!*fuhRQ> ztreJvss1<}JNkuZ+Lq*4a0lMM# zJQd`xAZhlr$P7Tmo`+4is~N+8fTRTZ6vW-ZwD}e!)XgNFQ&1PA4~WxU*{D7}1SI4W zwMsbw#1&*bNK%lCLDGWU1d||;k2@(?I6p*kWRUi>T=)Rt)Aa{Vo1bGs~6(j=^7v$d{2|+f1BsJj-SD8Q=j?*eD%b_N!?inFH%f0(L1G|DLFz$LnsDS*3;*}X>u59+c^#i-BL8!iKxzj zO>kE;J_|r{f;zZ@d4{ev`?xgcMv)fs43On)Y3Av0ABkw#PgPn9 zFH*gHFN$bU0{sUz8MYzdY=d(xIYIUZadzWqkj;r8K|v;fWHsT4DRqk2mWbGnR8((m z9WiHTUXRJ1Awgahd>P_7lk$h&0U-zPJYiSK2`=NXWkAg@zB zz1fE2qiUx3bi)B-*i+;WJ3h49k=@Aa)s|#bNXLiHKolIG`o|Z6Js?J9n*bLr~k*laISU{ zeKCG)!RZa>c6i|o@V$^}!BsgAlg$CjM#cGQkW83;C&T3+(SauU8YH0!$46;VeAc3> zK3ao*X-#~xoGnG#%<;+S_?#$oJS{(0_tx>D^$#xl%AxFZ#xbSV@@k>=PiPnO*s2F? za_ni5_E51fY=VQ#7#JvYr^R~EihNMiL4WXgPwB|_v~B$H{+y0{eO052d+g;UL}w$s$Z<5w)IkyoUM|vK6tdG`;|wwLrxqk6$fY0=L2d?#3DSUj z0;+}j{jiA(nBqzukkl?|l7aKvsf^=GpHVZ;e z{+}bXdcwvPHUmN8f*b>q6y#ixv>=TjSwZO8eQBX}J#3ueW}EK<2@3K!NLY}iAW=a+ z25|+UC-torTI*nw5H=m2gBODI1W5}r5F{(e(ICzUv(5Bg;2wLKc{m0(Az?EGBrM1k zAW=c)fVhJE86+VHJ%4PZ(0Ud&DPi*_NLrA8gJcEy9>h7s^rF-AXtN;nJ=V!WYj@a0 zgv|hus33=exPqJrk`UxPkdz?wEz!9`s~$EPVRIEoR*< zK84l^G6``TL_Xynhd@hSv1F5PstoTK7Hq71VFmdv z;`22~w6~dc9bQ0=^)X2=5Lb|aASpqP0a-1`IaJTyW<3ocNkOg!$p|tJB-+<3^*BgE z5ZXJe6l5uEV%$3=_U|9S#$_8XdS6n_g8U4U5@h={S||wZFFN)!Tev4|a>8aXh;x{k zQOAOW1UVNZBFF_GF+pg5Q6aRhflXZ4+y;^m>AkTti1bG7_CkX8?qCzVRn_$HB zA_o!@B=91<5Tq+eOpv`m;)2lrVusKf44b5|83mFOWDH0~5EmpT$V`yn;bxm@e=%2R z-42_uuz3I^BFG|;m>^3*;)1LINeV*yizPzq3)rNE%}*d1LAHAdUI?-WNbm^Liwcmi zAhf?&E3`(zCMs;s1c?b!3lbOPVvwXDH-MxCq5VYyU68OKYe1rc(EehS(AoqWSJ-rU8D0p|7bGdjAs}f%P6f#dLi>wqp>;lNoRMam zF98V(awAAskh?*mf;MY5NeMC< zBzUZ8Qwx$2lBmh z2oe!we~`E!M}s5;sRT(1av?}ckl%x(1$i7KBgp$8Swa2-5VoVK zk`v@8kjQAWp7THwf?Noa6=XI@_%yTBgCKE1o&pJ-ZrZ#85a$n>L+a zMK24o3rOl5(`EojT98p7;W4I714vGgc~sB2rp=2W(eq65Ih7Km^HQ|A(zF=_5{{YV z9FUA4SA)1?O`F@P)HstoOr->Q79?I}+Pnr5sy4}oASprCkxh+hv-NAp1VMHO35_>x zhJa)QISM2)!L&IQBrQlCh%=FGc!pH74b71E?SxO3aY6;b}s9N&>eNIiGze@(@VKRiqsO=S7gzRMTcT zNJfxVAn|Ez!!cCtp%`x5&T$5Nwed$=$1tU1NID%@#{0=Jd{oD9YHhK-%n$HF9|3tK zJN_N{bD_8`AaOw&s2)LP zfrMw6rS1aB2(l0)c&TagcaYR&Ciy2wR*-K&BA1&s9bU(H7o;ahMv#FZxwu*CND$`= zlbj9`6r>I$EXbuGDW9mUyPis2$@Ng2=}K|@Dkc=MTghgoNge`;2=XjQOpw<>;)47e zBq7K~kd!8*j<1?Ik*iIuj>|BU2(lAMMv%QhvVsf)314H@a~w!akg*_fK`sPI3UU)j zT9A7|vVuGV;#_NLWk7<0d)BHxfj1F(L3RO23UVMwPLLz0)D31mRa8n7j-hG~#c(L5 zvh7JwZ>^cJgZ-O1kK|uV`a#FXyOvZlz2M^dmE;Atuta?Bf=zI?>DRL$VL?`aLGW z2~rJ`733-qCt+&M2MG!CGDt*_H6SrT0`EXwkX|53L57251UU=Dx!F{200|3n4M^e- zrp?_%ZZXM|MCO=eDM;jxCiws)DaaQfsas8(_WuC6%_O}*qH~$>SXN_)#`4qKTIXr` zsV2BeM{Aa1D@MMrsoAmYf6p$muYXP^urq3t8_$QGv5r>nYoq(VsxKJbvKFhjire^jFNY zlcwg69Z2;nc(H7MGxBskDl5p=%Q4>P`|&9;8;4*MVjIt$VIUDfq98FrCW6ETxdJ5V z6SW$=9mKhp`-0=F+Cy=sXN>ps6#0YWoZ;Lk(&pC0Il|Eh{1+0PSjPLwaSrP^dwaBr z1v@n7F?f;lz1X5id#JJ{u+eSyYJUqPcAuG<{{l(eZ<6&OsRc|pK1z$?Q`Hgu;MK+- zc6?6ef9apCiq8O^0rt`5`4^7QF)}_?1vh>=TfGZUANIW{5ucv0$*_$lPyrHsgncJ; zJV@wKlT?BvA2Z2RkhCDb2Z{XEv`K=*1WAF!1xbVC1o;5OecY@kN2L}r;oec6Qtv#0 z-Z_qg<^AN|3F9t-wwRw?-8(&m4*kKsGe-9geNU=(LQTPqr(3><8GzSyUM)L-gr4;M zD$!p9U=#6e)L1?NB*vw@dd7pK1-S|&{1iuwjP3?W2=X*YN{~0G9zn7op+#H|M_#p= zBHsnU-N9=Ef0P&bAQ#s^`L3#jj=Y?o$dDuN<|9w%cBVEJ42<>gBEnvji1T*uV-&EB z*NT21t{_7|a)KNM5_y{AL&Zjeqy(t~i9Tc6Oa;jZavexc6OOZLGsSrfx}NrP`K>i^ z4sq5M@txzG(s3@m0-34f?9I=O4b$X)@K$({eAf50h!(Z}G1#Qp#&hR&kgOnCiiRNT zKq8CH7;duyqd<^7Kyrc{L^jWxrA`9LXu@$;ZKgO!x5RbYe30>5TgN%mf;itMsAggFEYB z=xe?gMY5OuQa0q5ZN%A<;i%ds`un-f`sE+vKc?I4owXj{P$NfN@B{Qc^HuIgIJu-+_L^SgkKO?bS-lgzg>8lZ7CZgE&1*ri^2y!9F20^X^iHn-&P$@y?gCqrc z9wa5m+aPH{av&K&w*3h2rWESCfH=6vE+4}_AngSSgLD*RI7nwfjs^(|5(Vid$k`x0 z1gQq0I~VinQ$Rw3Ob6*L$dw@d1epzz;89RyEK|qd4VzrX9AA%6skcq?3`p`_lPm?v zzGsr|Brv+RY8=; zPGBuUNzu%2Jg;udk3pQ5u%G3%5nhBvRJU7+y{sU6fVhLrBj^Dj!-UN-AR`1h3ncNe zichKI!6{_(PmUqA{xXo1APJC+Aoqae1oIbAD7jAA*%5H(0#%23EU2@W^%UXz6kc#=ZLD@%DVa~#vybAPN0WTAjkE1g_pvw zgzs0W7_Nj(l5Mp?PtZ1*wdx~$(9p3MM|j37sY~1ji%j^h??th`Xu^p%)TeC7uVMJB;5<@W`Q@*^ z2I)4_liS^4h2PFQ1Z{S5n{TtSp*GX_E0_V=xXnlFHczjdTt8)knPDENbjRy{M{691 z?$cYtJ?K;(f6|G^RMs|yoAHZE(HYLpL(Licctj(@rO3oN9VF@#HDAR*TtOy*#08lO z68=U-wVi$+qG}Jt@Kt074%3Xy9?L!P-vXOf%(PjKMxYY)v^SLU`FxI!v$ry@HkY22 zO=Bkv!fOs9B0iVHFX!9R*+Mcm!Y0Hvp6(qWvF|tA8VsoR7WhZ3=+lH{Ixpr zxYyBs)S}Y|Ug-9C4IK=Ue9l}AjsyvAAMA`fm zBpNhrUI9trYvz<=Dx=;9$!Nm)quN6;yb}TJB^=`zhF)x&y>RjmdX7@1$WvTYD)Hu#|JhUDi(9o9-de(XIMhyA z%%VFq@5Mf_N#dam`5rtNB-Gs`CxXNUIS0hqk!`q_l@|HF5u;!rO!8Y>_j2%P?&baS zBK~hYQXbXu!M$qsL5VXd3r{)gys{U9gnKj}v8!Mc@om~Ul}$|-Rn|{!ilZ5D+Mj7} zhD}P_06TYsB(#lJs&!Ue(GsL6wVg1sTx%4xP6&$oK<@$&Nm=J zoxOJQhaG3i9~v|Htu=8@vtNM-Dk!MOpCMQ3IHSpv8tToH=y>y)8a&6CE7)#`PtNzN z*s(Bss%*&9KV!#-FdG|t8si>BEDE>QcMLia@O#?gYsS@)$_uCDUdJ zNJ5bJK{BG|RUlbG)`83srG5dKE6CRWcAR;FbOT8WvOAG|n;%UF!Y0(8bDms15+p51 z6eN0pX;TRj3!7vzNPM74E(Hk>F-Zc%8E%sKL=IuXd8pbNg zo&W9qwDT|`{!f4Kj2+_q37oNo>bblr&uqIJc{pXN9C2j*6ud|uYmT_3L|!s8;U6G5 zK|TXnFUSvMBQjx&Rn$~-FT6cSP>?<#8$>;WK|(K^>Z3qnf}9O9OOVMRAyM-sAhU(d z)gVFcFVQdgsLg>*ifufc1j!uAy-es4kn~|Dc?KjKG07WbbGS*eAh9D9G2R2^X;HSk zj&=C#d_(XXXN&HO0PVYrUysv~4@5*A^ap24oO?Mi9^P*u>^NHn=d%T8guHiAiXGuP zco7zHc2>jpBYnS0M70NOa%|)2hC!T>W_(Tn2?|mR67q>^ZJcb5HcQP1aebm}7K3Dk z%?gn4F&uf4{~9Ev36EUm1@#5JxZ+^1o&3Rlk$lyY#p;zgiAd9%o9vK3%SddX55z(jTfkXwF0umGC zYLK+3=T?xcAb+7!OU$wS9LOR;G9a#~`3sP=u-OQb5H{^U!;CG+jvz@v_5?`@G6*Cs z$k8AfLCyfl3NjHSC&+~$-Ci;Mx&fqzAoqau6y&cUAwgaS=`F}RAh}V^kEYeIiT}#6ySdg?WjAVERi0?C|V+N=VJpT&enu8J6q+|@zH*-wm3d*sf-|NDs{$0Ikwy&Oo3 z|I;5li-z>beX_lEuG?SbPVd8OKOE;4IUDu^yoiW?@301b39=(d*7vlzb+57^-$$aA zTZwkF?^-MH0RCTaZ~0f`H8AxKJ)+d;B|JPr~&$Jg@ec?TqXu1UTD zanCbJmoJdvf*b^ri+W z8zd#jH6Xc(rcDaOsWZuIATie@{{~4nn&dl>$i*h<_$BJO+$6h!Lh;yw;{tObDZIU!dR1=;7RQ^!*9yzGgs@wV7=J(w#83j6fy{n45d0E6C zJPKaTj{>|<5?9s=p3wgRd2xz43ce)YxrJVJKY)Y<*=8-S(F@WGM7P;1br499$FfI` z0tpE+24tA1rw$}0$Q2-ILFRzu1o<;aSk(L!NJNkiL85|u53*j=6Zi^80fKY}*(AtL zAaU+@YxeORb@qdebCb%Jb{-lIk`!bVNG@U8oI&JflT?EQZ!t*|+1zT9Ss;nqm~bAd z7E&G_xQ*jHt(xV3@@%YoIpDw#c}|M=M7mD{F;NHo!LxCY^C$2D>M6dSe^=+BcShRU z7tMng&a381<`E)GP4WUr+V`~B{LH>9o96dKS}QOYZ4Q$=dSl;XoGnFqnSIZl%6p<^ zs-tx+h>yP)UZ;Jps-H3*H(nH6Dt-@sX)Uk(7a%UHd*m08;9S+q#e65DE?;AO-RawS zrS=9%2{Ifcc(-YDJV-{6N)R_`+Drq93vv@kLXZbRl7cJ&NeS{XNLrBZKr(`C`(MY& z3ep!OC&=L-&OK%eV<1UErh~-(k8LizKK7(`P2K_(&g>xgYH?Pp>=JdMy!k6A$2QL!yl^4ZUL{#5A^4+&lqwg`0G?&s> zpj18N`#~_1$9t{uJ$}IES7^~Q*>@y=!N$ICuYLE%MdMgk9O64K!3(YB)$txk=03kq zi}Vf|eFhu%{=$A&rO5Y(V6&Z98-JANI~Mo_8~1x$`@U;|MSf1$P5bWMZ#-qZbluqq zFA@is=d8BNAsW74r7~e}*ku3WXN#vD0SP};7!6eqMdNiWwM(2~Xu&AZEo|ltLsCcM z!jjS0M@Iw4DN~!QQw(Rqizs_hqTg#_lVcm5^QshiT8DBN^2XMheot~=6jA4VK3jWH zQ#rP#Aeo_i1zH$8i`QCYLT^mcvyIoX1Z*-Kd5_!!vRsf9NLG-SLE^8O`^tAgmJ0GY zi1xy(`Dc(6`{fb3Q+Jstwaqt}NrcTFAQ?f1f@B3b79=OgSs=8V^^WX3tqCAOP5fxo zhcqFh%U}~0^TtNDsp%`iMCypCT_FGszN=j394<-jzAj?2pK~{k@3GxfbbV0g&hi6&}asWtN zkl%sK6l5I8EI}>-nJq{HBq7LrkU4@p1rm8yja)hUD7V%225hp6&C&f&D)qccz6D7M zvgP;ai?nGI0*Sq7lA$2smrQa3NLG+>Ac>bvn~Om*e=|vfO1);12SI|bo8%dgT*f5t zfP~&L$!8!*O?Zw{@u4~DteqX_DzU%jIZDq7-qGIx4xRUtkN%>Z&)(6oFwsZ8*>iZ6 zOLxU|5Jo&l-LL1Usv>tbWA#s7ELVON>r?hi*^pn|p(77A*{>vLi||@COZ&y6shEv_ z#G2A=u9=1x5mXf}Pxtr%^F|BWtZb;wJE9QNd4Fdq9aDYxdvnr_q#{tfh-mz1(Fuz z6_C&}bMNp0NKBBgK%$&24%VvmRDDbn`fJM#m_-HI38Y%oa{x%LnaEQ;BViMM&m7%n zgQNwi1BnZ*tEkipv(#LW&<7@Y2qf~MNuCFZYQp17wUGLK$X1RsjT)sldwgN}?d+ug zPrumXD+yoif9{~IJN$T}i#n!Vg^Biiwn zNp=JYf2875Z2qBk_km4>Z3sAnK%zcTbJ3w7DM6wjSwUhT!IfrTG=M||xg5k5t{luiuWKhMIH19??HamnNU|#S6MY#zK_Ro{tYjJz8A%Mko{6N zku{Qgw_q<=W5x2QVUp}ver{cI(wQ^o_oONikTW!v4-G4-X3F3YC zM38-9<2K_dMQu22;%q~}IRPXoNF_*GkPASvg3uk3p&GM=GhyR=rLsls%l;1}D9DQ- zVL`GWQ9*tMaW&!IQGQYH?1->-^;*s!oOLbfon#BLF3eF4bcR-Emhpaa4-U{hh;FZ} zp3pE>-io^WPcY|f@k$;5l3>5=)bMN$gH4icJaQsP)+Z__&jJa5&7+cxYN(VT4Ir`q znl_h##J@Gk^&pXTW`EsBHiA3{lJ!ac`$WnM>Wj0msN3IbCx6&|(GUMBSo`sDZVUP% z$~hJ|6G?Rq%XmMzFHXq!1#Zi=URvn91}{<*%v^mBBqzu!kl;kqCI`}Ake@-Kf^^!1 z7>Zur6(qObk7|itJ`gseMX8Y>&Ua>Ci~*S_Z0bOQ!scp_8N%jPkdUx>1SBDBUH}OT zoA*H?f~)}vZZO-k5ybt;B%OXnn*$x4{5K$W1qpRz!XsBjgR*5cb_A1T5ArwoxP^gg z{2vDk#>V@LmW*88%YoHK4eD3UmYB{KJZq@2sa~EFaU6O}U`q6IfB2=_WwR}+rBY9U4b3PiqOZR6*2OCqnM;T=hN z4c8wW`83C;naC$~8?P15fdscTB zVm=gPUl3Q2V?h#voDY%`WHv}fkVimrf@DB~+nMe89waQtu9%CWf*c0o3NisCA;^s& zDM6k9$q4c;NKTNSK%DK(_Usl2I6*;<00{{)79=dll^_v8?gfbo@;8u}AZtKeLAJxv zHZI74APGUv07(jRIY>&7`#{oyyaAHYghzqOC>jN$Fbr`>Xl$*Sf6_cB6v<2;1sOdG zb}>}cuRNM1xTY?PM_zk(OhT66!t6)qDd1fdo6N>9i%(cQI}Hg5(4_93-);X>$%psHaJ$f}{kw1tikTw0RUHBgit4*lwoH zry$PmCiw*RsHf~*3G_AzZXfMf*;c0ix*ZQAq$3HCL~!5~pVP5^QGnKt7l}cjoLeW43a(7EcHD|XtYU!+n_H5=?#+Ag!d_G zex`lO^@lso*6NVPpQ+n=pOQM?-@^u);K-w5f52;izIvZRXVvP);#=NTrr~gSF@1`; zrydQG5rozN9qUb-Nw9HF_v3?Ki*2+9=q*ZJhEhQ;aPq) zP7{v2Y7a$z5~99`Xi{58KFsk690~8?sIhV64-%2LS6`pPi=b;pl_GY$AP!Epxo0*X z=bc~^XB)2_2Y@6683mH|iHczi#5qgVoIi`Dv?zvM5yM`FS2(xaisQJJtY8zI8-at- zl&z(fzapBUV~A&K;C9D>GI^cp@FIAD8N-`EQi9w|HVvlD(_|ya8)VaH+N=VJ2(k{u z6=dseF;WEC7o>BOSau`TTkW)c2g3z207FrWwlM^-{o$#C@WAc@;>uR@JP?uOOidOsyS)XpbO$L6Qxo%|Rg1sU|rNBs|?D=YoVTGD$s1 zT97zM;$pVpnO}_^nzgpu16OkpjQrNtvsT9S`#mrNr(~mfA^gTO{~wH5Yhsb(xSPo@ zKF0TIxE~~WiSL(ae_r4P*reFT!z(~CK2g2%HIW&r9z}iz$qBM8PL9Sd$JABk<<@F9_)JE*z&ZqF}tu+~y;SpD)m-(z(LT41+F(o(7 zsHHliaIFMq^$SidJNv^6t>snWT|MENJ#r9ilI&@b_K?xZuu1ziYAn}*WVw`A&s8Ad z*=7vy1W5?;B$X26EfD8Mvs4x&sELler$v#!0tL!HtFNQc%vt>u$0vYOaRnPkKC2_& zIC1RYfrHG&VC8WDRPI7^zx)mS3bS7&;`|H6hiyFNZMq?bH>voj$oB?uzcC{=h-}uG zBm$BWA2_T6*EG3QoqDd26Fivjj3pcLHT*r> zaDORJslRr?Fp=klTQgF0djhW4KJ$Y|N?P|Qksjpy z(fuAcUFgssoIgA2{BbXsS~IoaT?LNwA9#`K<$F;?i|YCgHaWH-;Ly!7!QI$*BHMyQ z1la{7CdfV@IYEYiIJ=wm903v%assgjyog+4jxRdj(j>^fut~9}McPBs z!(fwU8?PwXlf|VBgtORf;^l808RnV{0={eJ^HeFYqMX zN<72Xe0(Q#=c0vaZea;8{sNoO0qg}e_G$7|U8;Nk6OGKl?u9!)DoBS07Rf_yN4%6>mUTyqg_bJVO z1^Zn8KG>FIZ&GLP{w$$?@(d8zmybc{>D*HrrWLFrcP1~m9UqT^mxDx?nB-O}^|DDG2Fbi=k|ji5Fv&+Cspm}c z4cRO)l1iGSiEQpR z$qgXxT_*WIkl>vrd75ngWReWo++mW>K~i%~@(bDAW|Ho^BZhx8Nq>;cEhagVZ2n-9 zvq9X1NiHCpn@lnbBzJ>J?xs@LndC{3%rz!?gUD4T`GD%V(j=dPLh{{;+z5*A^Kt&$UMG_sf&p{xY#B6*D5i!r#fy@y5f>|K*#2#cm5wVwf9%S){rq%~UJ~GLA zkPNR6y**lo-WaD9{tBeT-h3~z8CJMTQKe{aeg+on`{}(a{la^5tyS#IZ7cTXdM_N< zDTZ6##j=3+=G*JNxr?(9O?bAvc@7xQrGXc&??t;3o*o06g!Ys?IgiLeg}$qL$oDR| z?r=OA>WzKRa-553wdz=gd(e9eRR^s@zp(GSYu|A;6~}sYhVSa{Mkl<|%!CWbuOpl9 z_c&~VBiVP-yPZlMYmx^+(wguJR@F@LX@{lziBwc?Z5cdZ$p7vNVi+I)PL!Nxh+jK&8bF`uY24BvvJs!bcGPrwO{H_3J&5kY!^ z#3!%~pXb->Uhi4}wK{3osOtrBJ~~!5gDRZZM6QQ|(x238nQ(t8PpQBD*u#2~s()`! z>+(55Vq)WC^DxIb5V1xE_tz-hUvy(n<%AmJh^%Q!Q=N+|A4%B{|4+a0 z44}Pe=K7cWm7kg3clvPSzOiw{Ds{x@7}mZ&+=Uk@9Wkol8X_0@o?1PjB$BX6vyF$I zp=kI-W%%1fE>`s@@-GnQ5|exj;`&5Q-oJum1nIIjY-X5|?*$UM)Fgw+Mvze;&Sh-F z`K!fCk7cbDSc)zycshZS+V(SX-UMir_DTN_Ye7q;cW%(*y_pkRg;We*7kkT zCEOmbp}jzQ2(k}|ZjWbkFi3{`)Fa1%M6XoQC^bW#1sm75Q6qN>NLJJ{10*=pug9x7 z0g~2)V_Wpl9ndi%PKz9 z%SYobIPbm-3WB%sXwp&j=BVxY-^#|Hf1<-T|GbNw?&V>k4*G-VpQm*%(^VdVWXHTk3Z>@<&R7c}rG`-+@w?CFw=xEUT$3Bn!97RL;{x(SJ#^!y`!Y1w8sEB zdj--JUL-|)`ha8w848k_TRLLwyRsqQORqq*R$wBUi4YlEYjPv4egD{~LH)`zK+f=e zVv!p?Pl8{e+c)o*vLV0DhkzX7+^3-rH@{A3b_VeK>nEeljg`|H3Le+wj76Ih&9s>! z&;Xkp+vt(2N|6`e!lvNa5xhQZBQLhZX(9EkRXpH89>!4voj7@M$L1riY^cpIVEB*l z%JN5BN4~pma~<8n*;x3jp?PR?@>gE3R@P3bZ9-F`^Zgk>riItY%Be2mTI0BE<5Z5X zoicVZzUi+C8QzbYQ$ESROH0*EzK@4Z!HTG@`paL>Yrg`cpjD6?erE2a<5N>LvBfF86?6P<&oteY4)^8#CRrAHspKhnM7-O=f?`#+=^M0#x2e9?E>d`Z8Sf4 zCON_IeNc(Fz{I#7uYxb(Y4G0C8BRu9?2Cx8jaRA{NKBA}K@x(T3z8FLI@NQ(ujSP< z7bNv(lPm%WKVXs%K%#saxih=O%y1H6a?{ITPeyapu2eidmU8Qlq+q$u@QkPMgd>iIiJ zXbZFEPe5|Q<_D0hu<1Mi+53VS`Q9Mec4j>XfrN$4$sn;UO`Gu`QDJi#*>o^%ZUJ$H z%|m2^$1r*`iRVQcB%y7%msNbIm+_FSf)m$m-OIXn0wH-L4V1Z;8+0$%S2l_32c8qo z3V4xXFN*Z3_RF&&zXqV7G;A}!{5{bXy3JMf`wt#AWI&^8D}Sy*o3qWdne2XmO?Zi! zEn6LcK7GX`-9eH|nXsp-X7UshTEW&H|pO~Uh6cl0wVAlE+hf9}ECjA-D!Fk{_X zc|6z~UL;=gy^tM4)p*-b*kswpBPWA||IWS>ssc#~GL>q6-L$zHBqzwNAkk&YMjc5# z1>$N_IdNAFl|17O+=7YASs{Z*GS4Q>WeHENb-)Kw(1K$vlQEo`=XB*A@hU# z;vwA^(}xYIa;M@wEFw3*C*R-nBUYj>I)(9%x6G*S3=$G#5J*grkst{{P6tT~az02_ z6V7MV9*W_^5LSJ}zO_{hsXof*4LYB1UZYm$(EU-}gwsm><3gX-zZgkz}MLwSfblwMXc3|I$d~$ARP~cmL?oArA`t19+h^X%JkOO5es(bi0!I1_!o}Y@gib$($ySuct<*U(arZA zEiSh03!89vj*s@#D@C5-kf6l(T-s`Ug7}^}Q&S8^@B(j>=Ot zibsP)dYI%w5NCIe2F2?JkX*J+iVw{e^UMX1|4H=;UXrPs8(;BZr2 zGgJQQ1&4zOt&$4f|e{ir7o=G|D!fNB;mxSiY-z$oDs}cPRc0uC~e+SNmS% zIHjp(I$iZ}dQLJVb)t34KOMSMb9xh9r9(*o+Ut+1xw$wwG zdZaA%_LllmOFdz!hb{GJS?ZgteotHKvn=%S`~3k+ zeTJo;DocHhmGcWM^(ISwNhjSxf2K{BrM}uyf51|AE%gPKdZsM(^_Kb~OFeF>Z?e?0 zWvO?x`hA(DKF3lYVfA~iEcKwJzT8ruXQ_{|)SW)%?BCZ~kxyCb(=GMoR^)?asrR#d zUu&r^wA7Q9@1e5PJ6q}*OMR}Te!Qg~E=xUPskgW0`DK=R)>4m@rCwpFZ?Jq%Tk4A~ z^=Mh@{VnyJrM}2gf51|Ym8G7ua(=ChZp&6;*_y-YUUf6gd)a zvm-=xZn^G`9ql+DmloIioiGB^~U!ai-?ktQk>Ypuc5MS@f;g^D0U=5d>~dXh7$ zRqFjL_4bzfGE4mdOFdPVdVfp3qotm))E8Om>9W+jS?W2?7M=5ST}4JdYN=<+Qm?So zgO=|rEcF$ZdbTX}VU~I~OMRuKzQ$6|m8IUpQeSUHezB!K#!`3oE@$l*w$wXYzAv}b zms;wKiQI)0X;lOFdMUddO1WWT`K))ahN{{wVP0`J0-V=XpPvA3MAsG{%=d zXNK8}-(WvTdq2;MX7+uw$Xu;NIYa~CP z7wR@osH!T`=1(rc{qw{9Z+qcwJz=js`aHJx*GBji@$$L2Uu4v|0@pOz#;L5t$@tnS zPDT^X4OI`FMSmEjunaXe+jr_y8A$v4DfkL_!N&ddy!PE~teIL}zkfpwzRj7xM6f&B zoZ}Ws8Lviev$CN!_X9^V85_6RZNXi^F^+RFEx*6Q19%-yA_t>|`HkBX>g(?v%zyf3 z<+TTsLy_%7s3-VJg!lhS5TzbMG>Q}fx(YU7uZ6JjOy+{P?7K%EBEk{#$de#(Ve>M` zbYb%*NJ7~B8zd>nI;uyM+Hx?iO8f1}-;b1!VRtmL_?0Z(Lhn0bUNiLv=b`p15YhkR zU-bP*97FGW$Nb;?&ppt>$iCF4#m5)-yRxA?{D&2L?)L=yZdqVG@86xC;)oTop;q-r zdqVv;-yUT{?b*w!sjb@MHq?)+L7PtBU*-H(PIdhh0#2PK9G|F9+66H13%0gu4{OD> zR)G-y{Qst9*KGGGukeR1;XN9Y!E=!5V$EMM2K^?;kTh<#@(CaPv8Sz4&sge7OMRlH zzS2@pmZiSbQlD$7$1L?_mU^lz^+lHYEK5CVsn4_2(`BhIwA5!>>c?B^2}?awmijVF zeV(OWZK*G})U#!&FSgWYTk4}N^#?5VTv_UAOFdz!pJ}Nt%&YrrfQViLls*@i??FBW zF?!h_xz2#iuM_^){e^!HZ4iAoLaoC~pNorgG)k|-D@JjAirM(_c?(~)R(E~a@!|RT zH}t(?a&v9w^N^eOeduz|pMYE!(;mk1t5vgcmTT7gf(6_jCKXNW1(P9b@|o#pRFE^O zRqEZg)%z4b8d*#I0ZSdP@hR97h00P7Tk4xE^(B^i+ENdfrM}jR{31&|ZmD;()FWl7 zA8+~I1BX;%k6)3s)H~vCQ@I9+mZd)0QV&_`YbTX%;V=VQ4mU_=vea`{&M&sqXIko^Hd3dz5{e8T zyxvk@Y^kR#^<-JTkFeA`TK%4})K^&Qsj}32Sn8`S-xpZwcnK9p%%A7eWvOqn`kkI| zCiisPBa9NRGVDmp^Tx3|AdN)gbrKOJ7{c#I@-&19&Pqx&DS?Z1z`D$xLoi0m#rllUW)O%R!3oP|aS?cpF z^@)~x*iy&W5DOxoElWLNsgJSLdt2)G>VJWHt}OLgmilN*y{Dy)#|RauJK=KHHghfY zYD>MprM|{e50<4KveegF^ZY_<#LcrtT&OH{*HRy0`QG009gp-bh@+mioPx zx@)Nqv()h=>;m;vS?WeZHd*izqMsYfmKP+96DE%l(KzQR(^S&z_ZgOY$Gp0KHSm*W zt_JcQn@zeNzFKNr|Kjri>;-HKw>G<@L!Ad06or6y)Yi-$n-6d_id_j+^;Ed*S1}tu zK3n6KzRg`Njr?}LS44RxDb{BH`rs_i!_qcG7jyPDbG20a)vEKE%eiL0Dl~W=w}(kZ zHG9Ei$UJL4jB-?4rJk|W=UVD9OFeF>$I4P)W~tAy)W=xr)t0(jmikglJz=SzX{kr^ z>VD4e+)U2%oRF`F=LAE`pA+Kj#cwbtRJUMGn9b2BJuWInb9{=~`0-iVOnmrGgWu5i ziiyp&*`E`VoA-U_3eKNq<^=6mt48H?Yi||h`I%>ciXJ>0Ga0f3le~X+j?epXMzu;k zX{k@Q)FYO9+`7t;EK7a1rCx2R_qWu0TI#8?)SE2z<1O`Wmij^~^69eFEGmilr_JzAFfTua@x)Q4H>qpirt%2H2Q>Ju&Xu%+JLQg_Qz zpJ}O&vDAB8>Km-c$IDWmY^jg5)H_@1vn=&QS?Z%L_5POndTYFgEcIkr>SHYR3QK*H zr5>@=Q)Q{2X{m=T^$nJKe@i`GmU^|NKEhIOZ}of3QqPp79<|i_S?W2<_imPYwk-8g zmU_riUt_6nu)Z_9qM4%!KHAFH!$(_&mcKU6u@{JU;nvn?VEVCN&76Vh$9-$VSVnfUP0)^F&0#YoQQ(tZt|v3cKz-plzDz+qRxX8F~s4L91_fDrk_#Hf#t0I9QKQ~2Cuf%7g*{MOFdYY`XE?jPfCgSRr5I`7A$G9N=8l}4QX7>3x3-(Xb2E!C&ZjLH=pjndg$k<|5+9+f*c z6Ccjr-_ZApRC8_iN5PuS`#!WiXK%4l>E}it?R!-NUS2tMe8DU3hSE1cy?)1cc#HMB zl^eJuL5p%@YqJ^Ke<#H-orlz?Qj27dMVlQbvC8a5&C z335zc2ASp0tE!$1)$BcK#H$C7cON^s$%+1pTSyK12sWwtj;E|jtp!;r+Vc}gdqFx5 z0YT}!@4Z15`R|EP^&APpSGN^81*D^B;rSryNi6Ke#URnYnJv5ygdTXHqDpPP1%&eu z@xgm<>*}Y~xZd?auIEA6s3$8DshrYOKXG!T!qE|{bn&Ll`Ubp-f#z^7^%Q9dPnUp@ za^BN-Kvrm*%JEf`>l2?3C{tK|;C|dC_5Lz*#1uu`@`-&qGzJFNixI z?!#V@gFr&MRONWQ@E4EvSNHR?7stV-v$i3vaUk)0CN$MguD`IRAuJ+R4;v@<+1R|N zSAc91_1p-;#}2AJNf5qI4{=s0?HnX(F?bnknrMh*u=3vML;MsF)FJXCRZnaG;{t9(h}-f5WYu9 zsec8sP(*dBgE0T##vKI2&z7A)X6CgfHq}qTD2VG(07GYA*mU&w@2ckEAge`GM}efL znPc!w5Po~FDm5O2&l@W;1H{Q^T|>=S@=1N?nj`;v*aW+pZB9}tZ28DP0g~43sl+S2 zr&PHYse2u{o>ySwYQHMSH`dflavb$-3%1F4H3yBc`~?W#_odpr0i-FfMG z!;8sy&WF0olxyDkV6<7pZ~#b3_YU!G+@ z3=%t?vxTC8$B;Yf?niFVwXg|`ndFZk8Ic?FL6W+6sOHB(c-5`cUjyOq@G9~i$ZFA^ z&p@*HXFluJgE*g=q}}j}ofPBT9e!HMd9 zC3!IhHmUQ?dZvJ=v&0;qtN&kRZyRH2ww(tRp9vvIqW~f}VBJXmVN1!-RXyF)gJROv zRn^nw>8`3mJ7DUlx~PwJl!f%r^Q?YRK*5bk>|ixrTIBr;f3?OMt!5LX?VE|`Z-D2I z*ETqQE#KkQ^FN+re;Ygx|5{Ax{{zI|9zo8ZO{NzQH@A*T%Vi$^z>gyBMLUrn2I6aR znum{od^g4VSAqQA607@kIa{nYHXf$Ya{`_}U2VR+yuQU(@7^lu>blW#IbN*pq}uWd zvA&YpGKS_2JzwW3{c-T{4+e0&={A7yt9*}<~A%{Tn_`U%0y;6e6dDg?xk-fU0rTk6s{G}x4PXqa`y!kg6{9(kx7f)-f-vi`f z(*A=$zMZ_Le;LSEQXl>SkUP^Trvu{e1X|jGeDm)_3;zuu|7f!3TR?tysxAKkkiV3W zUjXuJDb_Cm@%2^J^*;l7>!+fw-vZ+I3zX;o1(M7CcR&1xHvWaA>j!{*C+YfOAb&0) z?*RFsf}mZD&`dY(=zeG8uZgkyvdV$uRbK}>+^Ycm9Zvye}Nm(8Oc`J<| z|Lit$6%S{?Y^#UAAIMdb^8kpSYEU^(fqeh}j5faj@%L#HojX*M|^;*#m2V_!m{{H#QJ_oDMw+t zo_hP0HG#2~;}u?{i?g=>3$cEy*wbCE#@E-VKZ)meK0+@F4b?mS7l9lV3#t7_fc!mV zOy${QEL%Cz9Qz2!w~K|0bq(YYslDzwF#c1g0QY$kEa`7OEw>Z=NF63l-N;nD!~~Lz5mSIVNZ1+#CL8b$3>PGVDT{2X;{{90I#`VNpgDW(6V@Tg|ACd@(3uQ2N` zl4*{m@=?~ zuI~V;<4$)uo1+nIyp?+6zXqPK{4X)q-vBaCDg6hOoOJzD`{)5uKlXz_9wra}Ujp(` ziuI#FzLm5;0W$wRUR$`1G?`w`9{$6QFcr_i^HWLA=Rk5h{gXg`G4cF6K)xtF0OkC7 zAitV;egVi^_T5sgy}tzHTd79x2jq_?o;yHlN}-*z{`(J8FZu{P-%GI$f#mt%$CQ&ikQtD_ zRbsJknNKb@ek=7Ge+)d|PpSBmK;9~`n5#bxa!`h4!-^Mm~td&ei+Pp$Nj8ZCE=gipJlf5L8v^=8Z&s}rzRzJkNJhK#)HW@ld? zpL?cahSLYevvU)tyx9Ky1}u}>^%xqls-}T+(Uze1$E%a^r?=zjxu?aGfYgQBWoRv4 z1GKtHZW&T5SB-PW>@%cRS_7o|iM^wX3l3k{X4LF926$R8*`T5##n^jhJd}&Y=9e}% zx{vPNc~5tSSGS9CCquNM{D$}B!l9_dxCDBBx^ue!Y%j6Bzjv^Ay!~|lkL>OCXb}+Q zA`G36X^j>*32ttr*5JW*(afc}>u$1od+QxbxwS=L8@1_|23i_bK_odIsg{N#Qm9qS zUDL96@cbDPd%Cxq6;(Vm;0TJk3WRDEz(y&$TZtC>#FV;<2c;2gW7lL>F6|#`N;ggE z;m+w^>57xuNC}EFEUF_EQk>u>&E8B7o

vzSOwr$JF8yftkS%wjj+^HfQnO2H+? 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 literal 0 HcmV?d00001 diff --git a/license.txt b/license.txt new file mode 100644 index 00000000..3844dfb4 --- /dev/null +++ b/license.txt @@ -0,0 +1,19 @@ +Copyright (c) 2009 Satoshi Nakamoto + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/main.cpp b/main.cpp new file mode 100644 index 00000000..97000db3 --- /dev/null +++ b/main.cpp @@ -0,0 +1,2692 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" +#include "sha.h" + + + + + +// +// Global state +// + +CCriticalSection cs_main; + +map mapTransactions; +CCriticalSection cs_mapTransactions; +unsigned int nTransactionsUpdated = 0; +map mapNextTx; + +map mapBlockIndex; +const uint256 hashGenesisBlock("0x000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f"); +CBlockIndex* pindexGenesisBlock = NULL; +int nBestHeight = -1; +uint256 hashBestChain = 0; +CBlockIndex* pindexBest = NULL; + +map mapOrphanBlocks; +multimap mapOrphanBlocksByPrev; + +map mapOrphanTransactions; +multimap mapOrphanTransactionsByPrev; + +map mapWallet; +vector > vWalletUpdated; +CCriticalSection cs_mapWallet; + +map, CPrivKey> mapKeys; +map > mapPubKeys; +CCriticalSection cs_mapKeys; +CKey keyUser; + +string strSetDataDir; +int nDropMessagesTest = 0; + +// Settings +int fGenerateBitcoins; +int64 nTransactionFee = 0; +CAddress addrIncoming; + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// mapKeys +// + +bool AddKey(const CKey& key) +{ + CRITICAL_BLOCK(cs_mapKeys) + { + mapKeys[key.GetPubKey()] = key.GetPrivKey(); + mapPubKeys[Hash160(key.GetPubKey())] = key.GetPubKey(); + } + return CWalletDB().WriteKey(key.GetPubKey(), key.GetPrivKey()); +} + +vector GenerateNewKey() +{ + CKey key; + key.MakeNewKey(); + if (!AddKey(key)) + throw runtime_error("GenerateNewKey() : AddKey failed\n"); + return key.GetPubKey(); +} + + + + +////////////////////////////////////////////////////////////////////////////// +// +// mapWallet +// + +bool AddToWallet(const CWalletTx& wtxIn) +{ + uint256 hash = wtxIn.GetHash(); + CRITICAL_BLOCK(cs_mapWallet) + { + // Inserts only if not already there, returns tx inserted or tx found + pair::iterator, bool> ret = mapWallet.insert(make_pair(hash, wtxIn)); + CWalletTx& wtx = (*ret.first).second; + bool fInsertedNew = ret.second; + if (fInsertedNew) + wtx.nTimeReceived = GetAdjustedTime(); + + //// debug print + printf("AddToWallet %s %s\n", wtxIn.GetHash().ToString().substr(0,6).c_str(), fInsertedNew ? "new" : "update"); + + if (!fInsertedNew) + { + // Merge + bool fUpdated = false; + if (wtxIn.hashBlock != 0 && wtxIn.hashBlock != wtx.hashBlock) + { + wtx.hashBlock = wtxIn.hashBlock; + fUpdated = true; + } + if (wtxIn.nIndex != -1 && (wtxIn.vMerkleBranch != wtx.vMerkleBranch || wtxIn.nIndex != wtx.nIndex)) + { + wtx.vMerkleBranch = wtxIn.vMerkleBranch; + wtx.nIndex = wtxIn.nIndex; + fUpdated = true; + } + if (wtxIn.fFromMe && wtxIn.fFromMe != wtx.fFromMe) + { + wtx.fFromMe = wtxIn.fFromMe; + fUpdated = true; + } + if (wtxIn.fSpent && wtxIn.fSpent != wtx.fSpent) + { + wtx.fSpent = wtxIn.fSpent; + fUpdated = true; + } + if (!fUpdated) + return true; + } + + // Write to disk + if (!wtx.WriteToDisk()) + return false; + + // Notify UI + vWalletUpdated.push_back(make_pair(hash, fInsertedNew)); + } + + // Refresh UI + MainFrameRepaint(); + return true; +} + +bool AddToWalletIfMine(const CTransaction& tx, const CBlock* pblock) +{ + if (tx.IsMine() || mapWallet.count(tx.GetHash())) + { + CWalletTx wtx(tx); + // Get merkle branch if transaction was found in a block + if (pblock) + wtx.SetMerkleBranch(pblock); + return AddToWallet(wtx); + } + return true; +} + +bool EraseFromWallet(uint256 hash) +{ + CRITICAL_BLOCK(cs_mapWallet) + { + if (mapWallet.erase(hash)) + CWalletDB().EraseTx(hash); + } + return true; +} + + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// mapOrphanTransactions +// + +void AddOrphanTx(const CDataStream& vMsg) +{ + CTransaction tx; + CDataStream(vMsg) >> tx; + uint256 hash = tx.GetHash(); + if (mapOrphanTransactions.count(hash)) + return; + CDataStream* pvMsg = mapOrphanTransactions[hash] = new CDataStream(vMsg); + foreach(const CTxIn& txin, tx.vin) + mapOrphanTransactionsByPrev.insert(make_pair(txin.prevout.hash, pvMsg)); +} + +void EraseOrphanTx(uint256 hash) +{ + if (!mapOrphanTransactions.count(hash)) + return; + const CDataStream* pvMsg = mapOrphanTransactions[hash]; + CTransaction tx; + CDataStream(*pvMsg) >> tx; + foreach(const CTxIn& txin, tx.vin) + { + for (multimap::iterator mi = mapOrphanTransactionsByPrev.lower_bound(txin.prevout.hash); + mi != mapOrphanTransactionsByPrev.upper_bound(txin.prevout.hash);) + { + if ((*mi).second == pvMsg) + mapOrphanTransactionsByPrev.erase(mi++); + else + mi++; + } + } + delete pvMsg; + mapOrphanTransactions.erase(hash); +} + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CTransaction +// + +bool CTxIn::IsMine() const +{ + CRITICAL_BLOCK(cs_mapWallet) + { + map::iterator mi = mapWallet.find(prevout.hash); + if (mi != mapWallet.end()) + { + const CWalletTx& prev = (*mi).second; + if (prevout.n < prev.vout.size()) + if (prev.vout[prevout.n].IsMine()) + return true; + } + } + return false; +} + +int64 CTxIn::GetDebit() const +{ + CRITICAL_BLOCK(cs_mapWallet) + { + map::iterator mi = mapWallet.find(prevout.hash); + if (mi != mapWallet.end()) + { + const CWalletTx& prev = (*mi).second; + if (prevout.n < prev.vout.size()) + if (prev.vout[prevout.n].IsMine()) + return prev.vout[prevout.n].nValue; + } + } + return 0; +} + +int64 CWalletTx::GetTxTime() const +{ + if (!fTimeReceivedIsTxTime && hashBlock != 0) + { + // If we did not receive the transaction directly, we rely on the block's + // time to figure out when it happened. We use the median over a range + // of blocks to try to filter out inaccurate block times. + map::iterator mi = mapBlockIndex.find(hashBlock); + if (mi != mapBlockIndex.end()) + { + CBlockIndex* pindex = (*mi).second; + if (pindex) + return pindex->GetMedianTime(); + } + } + return nTimeReceived; +} + + + + + + +int CMerkleTx::SetMerkleBranch(const CBlock* pblock) +{ + if (fClient) + { + if (hashBlock == 0) + return 0; + } + else + { + CBlock blockTmp; + if (pblock == NULL) + { + // Load the block this tx is in + CTxIndex txindex; + if (!CTxDB("r").ReadTxIndex(GetHash(), txindex)) + return 0; + if (!blockTmp.ReadFromDisk(txindex.pos.nFile, txindex.pos.nBlockPos, true)) + return 0; + pblock = &blockTmp; + } + + // Update the tx's hashBlock + hashBlock = pblock->GetHash(); + + // Locate the transaction + for (nIndex = 0; nIndex < pblock->vtx.size(); nIndex++) + if (pblock->vtx[nIndex] == *(CTransaction*)this) + break; + if (nIndex == pblock->vtx.size()) + { + vMerkleBranch.clear(); + nIndex = -1; + printf("ERROR: SetMerkleBranch() : couldn't find tx in block\n"); + return 0; + } + + // Fill in merkle branch + vMerkleBranch = pblock->GetMerkleBranch(nIndex); + } + + // Is the tx in a block that's in the main chain + map::iterator mi = mapBlockIndex.find(hashBlock); + if (mi == mapBlockIndex.end()) + return 0; + CBlockIndex* pindex = (*mi).second; + if (!pindex || !pindex->IsInMainChain()) + return 0; + + return pindexBest->nHeight - pindex->nHeight + 1; +} + + + +void CWalletTx::AddSupportingTransactions(CTxDB& txdb) +{ + vtxPrev.clear(); + + const int COPY_DEPTH = 3; + if (SetMerkleBranch() < COPY_DEPTH) + { + vector vWorkQueue; + foreach(const CTxIn& txin, vin) + vWorkQueue.push_back(txin.prevout.hash); + + // This critsect is OK because txdb is already open + CRITICAL_BLOCK(cs_mapWallet) + { + map mapWalletPrev; + set setAlreadyDone; + for (int i = 0; i < vWorkQueue.size(); i++) + { + uint256 hash = vWorkQueue[i]; + if (setAlreadyDone.count(hash)) + continue; + setAlreadyDone.insert(hash); + + CMerkleTx tx; + if (mapWallet.count(hash)) + { + tx = mapWallet[hash]; + foreach(const CMerkleTx& txWalletPrev, mapWallet[hash].vtxPrev) + mapWalletPrev[txWalletPrev.GetHash()] = &txWalletPrev; + } + else if (mapWalletPrev.count(hash)) + { + tx = *mapWalletPrev[hash]; + } + else if (!fClient && txdb.ReadDiskTx(hash, tx)) + { + ; + } + else + { + printf("ERROR: AddSupportingTransactions() : unsupported transaction\n"); + continue; + } + + int nDepth = tx.SetMerkleBranch(); + vtxPrev.push_back(tx); + + if (nDepth < COPY_DEPTH) + foreach(const CTxIn& txin, tx.vin) + vWorkQueue.push_back(txin.prevout.hash); + } + } + } + + reverse(vtxPrev.begin(), vtxPrev.end()); +} + + + + + + + + + + + +bool CTransaction::AcceptTransaction(CTxDB& txdb, bool fCheckInputs, bool* pfMissingInputs) +{ + if (pfMissingInputs) + *pfMissingInputs = false; + + // Coinbase is only valid in a block, not as a loose transaction + if (IsCoinBase()) + return error("AcceptTransaction() : coinbase as individual tx"); + + if (!CheckTransaction()) + return error("AcceptTransaction() : CheckTransaction failed"); + + // Do we already have it? + uint256 hash = GetHash(); + CRITICAL_BLOCK(cs_mapTransactions) + if (mapTransactions.count(hash)) + return false; + if (fCheckInputs) + if (txdb.ContainsTx(hash)) + return false; + + // Check for conflicts with in-memory transactions + CTransaction* ptxOld = NULL; + for (int i = 0; i < vin.size(); i++) + { + COutPoint outpoint = vin[i].prevout; + if (mapNextTx.count(outpoint)) + { + // Allow replacing with a newer version of the same transaction + if (i != 0) + return false; + ptxOld = mapNextTx[outpoint].ptx; + if (!IsNewerThan(*ptxOld)) + return false; + for (int i = 0; i < vin.size(); i++) + { + COutPoint outpoint = vin[i].prevout; + if (!mapNextTx.count(outpoint) || mapNextTx[outpoint].ptx != ptxOld) + return false; + } + break; + } + } + + // Check against previous transactions + map mapUnused; + int64 nFees = 0; + if (fCheckInputs && !ConnectInputs(txdb, mapUnused, CDiskTxPos(1,1,1), 0, nFees, false, false)) + { + if (pfMissingInputs) + *pfMissingInputs = true; + return error("AcceptTransaction() : ConnectInputs failed %s", hash.ToString().substr(0,6).c_str()); + } + + // Store transaction in memory + CRITICAL_BLOCK(cs_mapTransactions) + { + if (ptxOld) + { + printf("mapTransaction.erase(%s) replacing with new version\n", ptxOld->GetHash().ToString().c_str()); + mapTransactions.erase(ptxOld->GetHash()); + } + AddToMemoryPool(); + } + + ///// are we sure this is ok when loading transactions or restoring block txes + // If updated, erase old tx from wallet + if (ptxOld) + EraseFromWallet(ptxOld->GetHash()); + + printf("AcceptTransaction(): accepted %s\n", hash.ToString().substr(0,6).c_str()); + return true; +} + + +bool CTransaction::AddToMemoryPool() +{ + // Add to memory pool without checking anything. Don't call this directly, + // call AcceptTransaction to properly check the transaction first. + CRITICAL_BLOCK(cs_mapTransactions) + { + uint256 hash = GetHash(); + mapTransactions[hash] = *this; + for (int i = 0; i < vin.size(); i++) + mapNextTx[vin[i].prevout] = CInPoint(&mapTransactions[hash], i); + nTransactionsUpdated++; + } + return true; +} + + +bool CTransaction::RemoveFromMemoryPool() +{ + // Remove transaction from memory pool + CRITICAL_BLOCK(cs_mapTransactions) + { + foreach(const CTxIn& txin, vin) + mapNextTx.erase(txin.prevout); + mapTransactions.erase(GetHash()); + nTransactionsUpdated++; + } + return true; +} + + + + + + +int CMerkleTx::GetDepthInMainChain() const +{ + if (hashBlock == 0 || nIndex == -1) + return 0; + + // Find the block it claims to be in + map::iterator mi = mapBlockIndex.find(hashBlock); + if (mi == mapBlockIndex.end()) + return 0; + CBlockIndex* pindex = (*mi).second; + if (!pindex || !pindex->IsInMainChain()) + return 0; + + // Make sure the merkle branch connects to this block + if (!fMerkleVerified) + { + if (CBlock::CheckMerkleBranch(GetHash(), vMerkleBranch, nIndex) != pindex->hashMerkleRoot) + return 0; + fMerkleVerified = true; + } + + return pindexBest->nHeight - pindex->nHeight + 1; +} + + +int CMerkleTx::GetBlocksToMaturity() const +{ + if (!IsCoinBase()) + return 0; + return max(0, (COINBASE_MATURITY+20) - GetDepthInMainChain()); +} + + +bool CMerkleTx::AcceptTransaction(CTxDB& txdb, bool fCheckInputs) +{ + if (fClient) + { + if (!IsInMainChain() && !ClientConnectInputs()) + return false; + return CTransaction::AcceptTransaction(txdb, false); + } + else + { + return CTransaction::AcceptTransaction(txdb, fCheckInputs); + } +} + + + +bool CWalletTx::AcceptWalletTransaction(CTxDB& txdb, bool fCheckInputs) +{ + CRITICAL_BLOCK(cs_mapTransactions) + { + foreach(CMerkleTx& tx, vtxPrev) + { + if (!tx.IsCoinBase()) + { + uint256 hash = tx.GetHash(); + if (!mapTransactions.count(hash) && !txdb.ContainsTx(hash)) + tx.AcceptTransaction(txdb, fCheckInputs); + } + } + if (!IsCoinBase()) + return AcceptTransaction(txdb, fCheckInputs); + } + return true; +} + +void ReacceptWalletTransactions() +{ + // Reaccept any txes of ours that aren't already in a block + CTxDB txdb("r"); + CRITICAL_BLOCK(cs_mapWallet) + { + foreach(PAIRTYPE(const uint256, CWalletTx)& item, mapWallet) + { + CWalletTx& wtx = item.second; + if (!wtx.IsCoinBase() && !txdb.ContainsTx(wtx.GetHash())) + wtx.AcceptWalletTransaction(txdb, false); + } + } +} + + +void CWalletTx::RelayWalletTransaction(CTxDB& txdb) +{ + foreach(const CMerkleTx& tx, vtxPrev) + { + if (!tx.IsCoinBase()) + { + uint256 hash = tx.GetHash(); + if (!txdb.ContainsTx(hash)) + RelayMessage(CInv(MSG_TX, hash), (CTransaction)tx); + } + } + if (!IsCoinBase()) + { + uint256 hash = GetHash(); + if (!txdb.ContainsTx(hash)) + { + printf("Relaying wtx %s\n", hash.ToString().substr(0,6).c_str()); + RelayMessage(CInv(MSG_TX, hash), (CTransaction)*this); + } + } +} + +void RelayWalletTransactions() +{ + static int64 nLastTime; + if (GetTime() - nLastTime < 10 * 60) + return; + nLastTime = GetTime(); + + // Rebroadcast any of our txes that aren't in a block yet + printf("RelayWalletTransactions()\n"); + CTxDB txdb("r"); + CRITICAL_BLOCK(cs_mapWallet) + { + // Sort them in chronological order + multimap mapSorted; + foreach(PAIRTYPE(const uint256, CWalletTx)& item, mapWallet) + { + CWalletTx& wtx = item.second; + mapSorted.insert(make_pair(wtx.nTimeReceived, &wtx)); + } + foreach(PAIRTYPE(const unsigned int, CWalletTx*)& item, mapSorted) + { + CWalletTx& wtx = *item.second; + wtx.RelayWalletTransaction(txdb); + } + } +} + + + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CBlock and CBlockIndex +// + +bool CBlock::ReadFromDisk(const CBlockIndex* pblockindex, bool fReadTransactions) +{ + return ReadFromDisk(pblockindex->nFile, pblockindex->nBlockPos, fReadTransactions); +} + +uint256 GetOrphanRoot(const CBlock* pblock) +{ + // Work back to the first block in the orphan chain + while (mapOrphanBlocks.count(pblock->hashPrevBlock)) + pblock = mapOrphanBlocks[pblock->hashPrevBlock]; + return pblock->GetHash(); +} + +int64 CBlock::GetBlockValue(int64 nFees) const +{ + int64 nSubsidy = 50 * COIN; + + // Subsidy is cut in half every 4 years + nSubsidy >>= (nBestHeight / 210000); + + return nSubsidy + nFees; +} + +unsigned int GetNextWorkRequired(const CBlockIndex* pindexLast) +{ + const unsigned int nTargetTimespan = 14 * 24 * 60 * 60; // two weeks + const unsigned int nTargetSpacing = 10 * 60; + const unsigned int nInterval = nTargetTimespan / nTargetSpacing; + + // Genesis block + if (pindexLast == NULL) + return bnProofOfWorkLimit.GetCompact(); + + // Only change once per interval + if ((pindexLast->nHeight+1) % nInterval != 0) + return pindexLast->nBits; + + // Go back by what we want to be 14 days worth of blocks + const CBlockIndex* pindexFirst = pindexLast; + for (int i = 0; pindexFirst && i < nInterval-1; i++) + pindexFirst = pindexFirst->pprev; + assert(pindexFirst); + + // Limit adjustment step + unsigned int nActualTimespan = pindexLast->nTime - pindexFirst->nTime; + printf(" nActualTimespan = %d before bounds\n", nActualTimespan); + if (nActualTimespan < nTargetTimespan/4) + nActualTimespan = nTargetTimespan/4; + if (nActualTimespan > nTargetTimespan*4) + nActualTimespan = nTargetTimespan*4; + + // Retarget + CBigNum bnNew; + bnNew.SetCompact(pindexLast->nBits); + bnNew *= nActualTimespan; + bnNew /= nTargetTimespan; + + if (bnNew > bnProofOfWorkLimit) + bnNew = bnProofOfWorkLimit; + + /// debug print + printf("\n\n\nGetNextWorkRequired RETARGET *****\n"); + printf("nTargetTimespan = %d nActualTimespan = %d\n", nTargetTimespan, nActualTimespan); + printf("Before: %08x %s\n", pindexLast->nBits, CBigNum().SetCompact(pindexLast->nBits).getuint256().ToString().c_str()); + printf("After: %08x %s\n", bnNew.GetCompact(), bnNew.getuint256().ToString().c_str()); + + return bnNew.GetCompact(); +} + + + + + + + + + +bool CTransaction::DisconnectInputs(CTxDB& txdb) +{ + // Relinquish previous transactions' spent pointers + if (!IsCoinBase()) + { + foreach(const CTxIn& txin, vin) + { + COutPoint prevout = txin.prevout; + + // Get prev txindex from disk + CTxIndex txindex; + if (!txdb.ReadTxIndex(prevout.hash, txindex)) + return error("DisconnectInputs() : ReadTxIndex failed"); + + if (prevout.n >= txindex.vSpent.size()) + return error("DisconnectInputs() : prevout.n out of range"); + + // Mark outpoint as not spent + txindex.vSpent[prevout.n].SetNull(); + + // Write back + txdb.UpdateTxIndex(prevout.hash, txindex); + } + } + + // Remove transaction from index + if (!txdb.EraseTxIndex(*this)) + return error("DisconnectInputs() : EraseTxPos failed"); + + return true; +} + + +bool CTransaction::ConnectInputs(CTxDB& txdb, map& mapTestPool, CDiskTxPos posThisTx, int nHeight, int64& nFees, bool fBlock, bool fMiner, int64 nMinFee) +{ + // Take over previous transactions' spent pointers + if (!IsCoinBase()) + { + int64 nValueIn = 0; + for (int i = 0; i < vin.size(); i++) + { + COutPoint prevout = vin[i].prevout; + + // Read txindex + CTxIndex txindex; + bool fFound = true; + if (fMiner && mapTestPool.count(prevout.hash)) + { + // Get txindex from current proposed changes + txindex = mapTestPool[prevout.hash]; + } + else + { + // Read txindex from txdb + fFound = txdb.ReadTxIndex(prevout.hash, txindex); + } + if (!fFound && (fBlock || fMiner)) + return fMiner ? false : error("ConnectInputs() : %s prev tx %s index entry not found", GetHash().ToString().substr(0,6).c_str(), prevout.hash.ToString().substr(0,6).c_str()); + + // Read txPrev + CTransaction txPrev; + if (!fFound || txindex.pos == CDiskTxPos(1,1,1)) + { + // Get prev tx from single transactions in memory + CRITICAL_BLOCK(cs_mapTransactions) + { + if (!mapTransactions.count(prevout.hash)) + return error("ConnectInputs() : %s mapTransactions prev not found %s", GetHash().ToString().substr(0,6).c_str(), prevout.hash.ToString().substr(0,6).c_str()); + txPrev = mapTransactions[prevout.hash]; + } + if (!fFound) + txindex.vSpent.resize(txPrev.vout.size()); + } + else + { + // Get prev tx from disk + if (!txPrev.ReadFromDisk(txindex.pos)) + return error("ConnectInputs() : %s ReadFromDisk prev tx %s failed", GetHash().ToString().substr(0,6).c_str(), prevout.hash.ToString().substr(0,6).c_str()); + } + + if (prevout.n >= txPrev.vout.size() || prevout.n >= txindex.vSpent.size()) + return error("ConnectInputs() : %s prevout.n out of range %d %d %d", GetHash().ToString().substr(0,6).c_str(), prevout.n, txPrev.vout.size(), txindex.vSpent.size()); + + // If prev is coinbase, check that it's matured + if (txPrev.IsCoinBase()) + for (CBlockIndex* pindex = pindexBest; pindex && nBestHeight - pindex->nHeight < COINBASE_MATURITY-1; pindex = pindex->pprev) + if (pindex->nBlockPos == txindex.pos.nBlockPos && pindex->nFile == txindex.pos.nFile) + return error("ConnectInputs() : tried to spend coinbase at depth %d", nBestHeight - pindex->nHeight); + + // Verify signature + if (!VerifySignature(txPrev, *this, i)) + return error("ConnectInputs() : %s VerifySignature failed", GetHash().ToString().substr(0,6).c_str()); + + // Check for conflicts + if (!txindex.vSpent[prevout.n].IsNull()) + return fMiner ? false : error("ConnectInputs() : %s prev tx already used at %s", GetHash().ToString().substr(0,6).c_str(), txindex.vSpent[prevout.n].ToString().c_str()); + + // Mark outpoints as spent + txindex.vSpent[prevout.n] = posThisTx; + + // Write back + if (fBlock) + txdb.UpdateTxIndex(prevout.hash, txindex); + else if (fMiner) + mapTestPool[prevout.hash] = txindex; + + nValueIn += txPrev.vout[prevout.n].nValue; + } + + // Tally transaction fees + int64 nTxFee = nValueIn - GetValueOut(); + if (nTxFee < 0) + return error("ConnectInputs() : %s nTxFee < 0", GetHash().ToString().substr(0,6).c_str()); + if (nTxFee < nMinFee) + return false; + nFees += nTxFee; + } + + if (fBlock) + { + // Add transaction to disk index + if (!txdb.AddTxIndex(*this, posThisTx, nHeight)) + return error("ConnectInputs() : AddTxPos failed"); + } + else if (fMiner) + { + // Add transaction to test pool + mapTestPool[GetHash()] = CTxIndex(CDiskTxPos(1,1,1), vout.size()); + } + + return true; +} + + +bool CTransaction::ClientConnectInputs() +{ + if (IsCoinBase()) + return false; + + // Take over previous transactions' spent pointers + CRITICAL_BLOCK(cs_mapTransactions) + { + int64 nValueIn = 0; + for (int i = 0; i < vin.size(); i++) + { + // Get prev tx from single transactions in memory + COutPoint prevout = vin[i].prevout; + if (!mapTransactions.count(prevout.hash)) + return false; + CTransaction& txPrev = mapTransactions[prevout.hash]; + + if (prevout.n >= txPrev.vout.size()) + return false; + + // Verify signature + if (!VerifySignature(txPrev, *this, i)) + return error("ConnectInputs() : VerifySignature failed"); + + ///// this is redundant with the mapNextTx stuff, not sure which I want to get rid of + ///// this has to go away now that posNext is gone + // // Check for conflicts + // if (!txPrev.vout[prevout.n].posNext.IsNull()) + // return error("ConnectInputs() : prev tx already used"); + // + // // Flag outpoints as used + // txPrev.vout[prevout.n].posNext = posThisTx; + + nValueIn += txPrev.vout[prevout.n].nValue; + } + if (GetValueOut() > nValueIn) + return false; + } + + return true; +} + + + + +bool CBlock::DisconnectBlock(CTxDB& txdb, CBlockIndex* pindex) +{ + // Disconnect in reverse order + for (int i = vtx.size()-1; i >= 0; i--) + if (!vtx[i].DisconnectInputs(txdb)) + return false; + + // Update block index on disk without changing it in memory. + // The memory index structure will be changed after the db commits. + if (pindex->pprev) + { + CDiskBlockIndex blockindexPrev(pindex->pprev); + blockindexPrev.hashNext = 0; + txdb.WriteBlockIndex(blockindexPrev); + } + + return true; +} + +bool CBlock::ConnectBlock(CTxDB& txdb, CBlockIndex* pindex) +{ + //// issue here: it doesn't know the version + unsigned int nTxPos = pindex->nBlockPos + ::GetSerializeSize(CBlock(), SER_DISK) - 1 + GetSizeOfCompactSize(vtx.size()); + + map mapUnused; + int64 nFees = 0; + foreach(CTransaction& tx, vtx) + { + CDiskTxPos posThisTx(pindex->nFile, pindex->nBlockPos, nTxPos); + nTxPos += ::GetSerializeSize(tx, SER_DISK); + + if (!tx.ConnectInputs(txdb, mapUnused, posThisTx, pindex->nHeight, nFees, true, false)) + return false; + } + + if (vtx[0].GetValueOut() > GetBlockValue(nFees)) + return false; + + // Update block index on disk without changing it in memory. + // The memory index structure will be changed after the db commits. + if (pindex->pprev) + { + CDiskBlockIndex blockindexPrev(pindex->pprev); + blockindexPrev.hashNext = pindex->GetBlockHash(); + txdb.WriteBlockIndex(blockindexPrev); + } + + // Watch for transactions paying to me + foreach(CTransaction& tx, vtx) + AddToWalletIfMine(tx, this); + + return true; +} + + + +bool Reorganize(CTxDB& txdb, CBlockIndex* pindexNew) +{ + printf("*** REORGANIZE ***\n"); + + // Find the fork + CBlockIndex* pfork = pindexBest; + CBlockIndex* plonger = pindexNew; + while (pfork != plonger) + { + if (!(pfork = pfork->pprev)) + return error("Reorganize() : pfork->pprev is null"); + while (plonger->nHeight > pfork->nHeight) + if (!(plonger = plonger->pprev)) + return error("Reorganize() : plonger->pprev is null"); + } + + // List of what to disconnect + vector vDisconnect; + for (CBlockIndex* pindex = pindexBest; pindex != pfork; pindex = pindex->pprev) + vDisconnect.push_back(pindex); + + // List of what to connect + vector vConnect; + for (CBlockIndex* pindex = pindexNew; pindex != pfork; pindex = pindex->pprev) + vConnect.push_back(pindex); + reverse(vConnect.begin(), vConnect.end()); + + // Disconnect shorter branch + vector vResurrect; + foreach(CBlockIndex* pindex, vDisconnect) + { + CBlock block; + if (!block.ReadFromDisk(pindex->nFile, pindex->nBlockPos, true)) + return error("Reorganize() : ReadFromDisk for disconnect failed"); + if (!block.DisconnectBlock(txdb, pindex)) + return error("Reorganize() : DisconnectBlock failed"); + + // Queue memory transactions to resurrect + foreach(const CTransaction& tx, block.vtx) + if (!tx.IsCoinBase()) + vResurrect.push_back(tx); + } + + // Connect longer branch + vector vDelete; + for (int i = 0; i < vConnect.size(); i++) + { + CBlockIndex* pindex = vConnect[i]; + CBlock block; + if (!block.ReadFromDisk(pindex->nFile, pindex->nBlockPos, true)) + return error("Reorganize() : ReadFromDisk for connect failed"); + if (!block.ConnectBlock(txdb, pindex)) + { + // Invalid block, delete the rest of this branch + txdb.TxnAbort(); + for (int j = i; j < vConnect.size(); j++) + { + CBlockIndex* pindex = vConnect[j]; + pindex->EraseBlockFromDisk(); + txdb.EraseBlockIndex(pindex->GetBlockHash()); + mapBlockIndex.erase(pindex->GetBlockHash()); + delete pindex; + } + return error("Reorganize() : ConnectBlock failed"); + } + + // Queue memory transactions to delete + foreach(const CTransaction& tx, block.vtx) + vDelete.push_back(tx); + } + if (!txdb.WriteHashBestChain(pindexNew->GetBlockHash())) + return error("Reorganize() : WriteHashBestChain failed"); + + // Commit now because resurrecting could take some time + txdb.TxnCommit(); + + // Disconnect shorter branch + foreach(CBlockIndex* pindex, vDisconnect) + if (pindex->pprev) + pindex->pprev->pnext = NULL; + + // Connect longer branch + foreach(CBlockIndex* pindex, vConnect) + if (pindex->pprev) + pindex->pprev->pnext = pindex; + + // Resurrect memory transactions that were in the disconnected branch + foreach(CTransaction& tx, vResurrect) + tx.AcceptTransaction(txdb, false); + + // Delete redundant memory transactions that are in the connected branch + foreach(CTransaction& tx, vDelete) + tx.RemoveFromMemoryPool(); + + return true; +} + + +bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) +{ + // Check for duplicate + uint256 hash = GetHash(); + if (mapBlockIndex.count(hash)) + return error("AddToBlockIndex() : %s already exists", hash.ToString().substr(0,14).c_str()); + + // Construct new block index object + CBlockIndex* pindexNew = new CBlockIndex(nFile, nBlockPos, *this); + if (!pindexNew) + return error("AddToBlockIndex() : new CBlockIndex failed"); + map::iterator mi = mapBlockIndex.insert(make_pair(hash, pindexNew)).first; + pindexNew->phashBlock = &((*mi).first); + map::iterator miPrev = mapBlockIndex.find(hashPrevBlock); + if (miPrev != mapBlockIndex.end()) + { + pindexNew->pprev = (*miPrev).second; + pindexNew->nHeight = pindexNew->pprev->nHeight + 1; + } + + CTxDB txdb; + txdb.TxnBegin(); + txdb.WriteBlockIndex(CDiskBlockIndex(pindexNew)); + + // New best + if (pindexNew->nHeight > nBestHeight) + { + if (pindexGenesisBlock == NULL && hash == hashGenesisBlock) + { + pindexGenesisBlock = pindexNew; + txdb.WriteHashBestChain(hash); + } + else if (hashPrevBlock == hashBestChain) + { + // Adding to current best branch + if (!ConnectBlock(txdb, pindexNew) || !txdb.WriteHashBestChain(hash)) + { + txdb.TxnAbort(); + pindexNew->EraseBlockFromDisk(); + mapBlockIndex.erase(pindexNew->GetBlockHash()); + delete pindexNew; + return error("AddToBlockIndex() : ConnectBlock failed"); + } + txdb.TxnCommit(); + pindexNew->pprev->pnext = pindexNew; + + // Delete redundant memory transactions + foreach(CTransaction& tx, vtx) + tx.RemoveFromMemoryPool(); + } + else + { + // New best branch + if (!Reorganize(txdb, pindexNew)) + { + txdb.TxnAbort(); + return error("AddToBlockIndex() : Reorganize failed"); + } + } + + // New best link + hashBestChain = hash; + pindexBest = pindexNew; + nBestHeight = pindexBest->nHeight; + nTransactionsUpdated++; + printf("AddToBlockIndex: new best=%s height=%d\n", hashBestChain.ToString().substr(0,14).c_str(), nBestHeight); + } + + txdb.TxnCommit(); + txdb.Close(); + + // Relay wallet transactions that haven't gotten in yet + if (pindexNew == pindexBest) + RelayWalletTransactions(); + + MainFrameRepaint(); + return true; +} + + + + +bool CBlock::CheckBlock() const +{ + // These are checks that are independent of context + // that can be verified before saving an orphan block. + + // Size limits + if (vtx.empty() || vtx.size() > MAX_SIZE || ::GetSerializeSize(*this, SER_DISK) > MAX_SIZE) + return error("CheckBlock() : size limits failed"); + + // Check timestamp + if (nTime > GetAdjustedTime() + 2 * 60 * 60) + return error("CheckBlock() : block timestamp too far in the future"); + + // First transaction must be coinbase, the rest must not be + if (vtx.empty() || !vtx[0].IsCoinBase()) + return error("CheckBlock() : first tx is not coinbase"); + for (int i = 1; i < vtx.size(); i++) + if (vtx[i].IsCoinBase()) + return error("CheckBlock() : more than one coinbase"); + + // Check transactions + foreach(const CTransaction& tx, vtx) + if (!tx.CheckTransaction()) + return error("CheckBlock() : CheckTransaction failed"); + + // Check proof of work matches claimed amount + if (CBigNum().SetCompact(nBits) > bnProofOfWorkLimit) + return error("CheckBlock() : nBits below minimum work"); + if (GetHash() > CBigNum().SetCompact(nBits).getuint256()) + return error("CheckBlock() : hash doesn't match nBits"); + + // Check merkleroot + if (hashMerkleRoot != BuildMerkleTree()) + return error("CheckBlock() : hashMerkleRoot mismatch"); + + return true; +} + +bool CBlock::AcceptBlock() +{ + // Check for duplicate + uint256 hash = GetHash(); + if (mapBlockIndex.count(hash)) + return error("AcceptBlock() : block already in mapBlockIndex"); + + // Get prev block index + map::iterator mi = mapBlockIndex.find(hashPrevBlock); + if (mi == mapBlockIndex.end()) + return error("AcceptBlock() : prev block not found"); + CBlockIndex* pindexPrev = (*mi).second; + + // Check timestamp against prev + if (nTime <= pindexPrev->GetMedianTimePast()) + return error("AcceptBlock() : block's timestamp is too early"); + + // Check proof of work + if (nBits != GetNextWorkRequired(pindexPrev)) + return error("AcceptBlock() : incorrect proof of work"); + + // Write block to history file + if (!CheckDiskSpace(::GetSerializeSize(*this, SER_DISK))) + return error("AcceptBlock() : out of disk space"); + unsigned int nFile; + unsigned int nBlockPos; + if (!WriteToDisk(!fClient, nFile, nBlockPos)) + return error("AcceptBlock() : WriteToDisk failed"); + if (!AddToBlockIndex(nFile, nBlockPos)) + return error("AcceptBlock() : AddToBlockIndex failed"); + + if (hashBestChain == hash) + RelayInventory(CInv(MSG_BLOCK, hash)); + + // // Add atoms to user reviews for coins created + // vector vchPubKey; + // if (ExtractPubKey(vtx[0].vout[0].scriptPubKey, false, vchPubKey)) + // { + // unsigned short nAtom = GetRand(USHRT_MAX - 100) + 100; + // vector vAtoms(1, nAtom); + // AddAtomsAndPropagate(Hash(vchPubKey.begin(), vchPubKey.end()), vAtoms, true); + // } + + return true; +} + +bool ProcessBlock(CNode* pfrom, CBlock* pblock) +{ + // Check for duplicate + uint256 hash = pblock->GetHash(); + if (mapBlockIndex.count(hash)) + return error("ProcessBlock() : already have block %d %s", mapBlockIndex[hash]->nHeight, hash.ToString().substr(0,14).c_str()); + if (mapOrphanBlocks.count(hash)) + return error("ProcessBlock() : already have block (orphan) %s", hash.ToString().substr(0,14).c_str()); + + // Preliminary checks + if (!pblock->CheckBlock()) + { + delete pblock; + return error("ProcessBlock() : CheckBlock FAILED"); + } + + // If don't already have its previous block, shunt it off to holding area until we get it + if (!mapBlockIndex.count(pblock->hashPrevBlock)) + { + printf("ProcessBlock: ORPHAN BLOCK, prev=%s\n", pblock->hashPrevBlock.ToString().substr(0,14).c_str()); + mapOrphanBlocks.insert(make_pair(hash, pblock)); + mapOrphanBlocksByPrev.insert(make_pair(pblock->hashPrevBlock, pblock)); + + // Ask this guy to fill in what we're missing + if (pfrom) + pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), GetOrphanRoot(pblock)); + return true; + } + + // Store to disk + if (!pblock->AcceptBlock()) + { + delete pblock; + return error("ProcessBlock() : AcceptBlock FAILED"); + } + delete pblock; + + // Recursively process any orphan blocks that depended on this one + vector vWorkQueue; + vWorkQueue.push_back(hash); + for (int i = 0; i < vWorkQueue.size(); i++) + { + uint256 hashPrev = vWorkQueue[i]; + for (multimap::iterator mi = mapOrphanBlocksByPrev.lower_bound(hashPrev); + mi != mapOrphanBlocksByPrev.upper_bound(hashPrev); + ++mi) + { + CBlock* pblockOrphan = (*mi).second; + if (pblockOrphan->AcceptBlock()) + vWorkQueue.push_back(pblockOrphan->GetHash()); + mapOrphanBlocks.erase(pblockOrphan->GetHash()); + delete pblockOrphan; + } + mapOrphanBlocksByPrev.erase(hashPrev); + } + + printf("ProcessBlock: ACCEPTED\n"); + return true; +} + + + + + + + + +template +bool ScanMessageStart(Stream& s) +{ + // Scan ahead to the next pchMessageStart, which should normally be immediately + // at the file pointer. Leaves file pointer at end of pchMessageStart. + s.clear(0); + short prevmask = s.exceptions(0); + const char* p = BEGIN(pchMessageStart); + try + { + loop + { + char c; + s.read(&c, 1); + if (s.fail()) + { + s.clear(0); + s.exceptions(prevmask); + return false; + } + if (*p != c) + p = BEGIN(pchMessageStart); + if (*p == c) + { + if (++p == END(pchMessageStart)) + { + s.clear(0); + s.exceptions(prevmask); + return true; + } + } + } + } + catch (...) + { + s.clear(0); + s.exceptions(prevmask); + return false; + } +} + +string GetAppDir() +{ + string strDir; + if (!strSetDataDir.empty()) + { + strDir = strSetDataDir; + } + else if (getenv("APPDATA")) + { + strDir = strprintf("%s\\Bitcoin", getenv("APPDATA")); + } + else if (getenv("USERPROFILE")) + { + string strAppData = strprintf("%s\\Application Data", getenv("USERPROFILE")); + static bool fMkdirDone; + if (!fMkdirDone) + { + fMkdirDone = true; + _mkdir(strAppData.c_str()); + } + strDir = strprintf("%s\\Bitcoin", strAppData.c_str()); + } + else + { + return "."; + } + static bool fMkdirDone; + if (!fMkdirDone) + { + fMkdirDone = true; + _mkdir(strDir.c_str()); + } + return strDir; +} + +bool CheckDiskSpace(int64 nAdditionalBytes) +{ + uint64 nFreeBytesAvailable = 0; // bytes available to caller + uint64 nTotalNumberOfBytes = 0; // bytes on disk + uint64 nTotalNumberOfFreeBytes = 0; // free bytes on disk + + if (!GetDiskFreeSpaceEx(GetAppDir().c_str(), + (PULARGE_INTEGER)&nFreeBytesAvailable, + (PULARGE_INTEGER)&nTotalNumberOfBytes, + (PULARGE_INTEGER)&nTotalNumberOfFreeBytes)) + { + printf("ERROR: GetDiskFreeSpaceEx() failed\n"); + return true; + } + + // Check for 15MB because database could create another 10MB log file at any time + if ((int64)nFreeBytesAvailable < 15000000 + nAdditionalBytes) + { + fShutdown = true; + wxMessageBox("Warning: Your disk space is low ", "Bitcoin", wxICON_EXCLAMATION); + _beginthread(Shutdown, 0, NULL); + return false; + } + return true; +} + +FILE* OpenBlockFile(unsigned int nFile, unsigned int nBlockPos, const char* pszMode) +{ + if (nFile == -1) + return NULL; + FILE* file = fopen(strprintf("%s\\blk%04d.dat", GetAppDir().c_str(), nFile).c_str(), pszMode); + if (!file) + return NULL; + if (nBlockPos != 0 && !strchr(pszMode, 'a') && !strchr(pszMode, 'w')) + { + if (fseek(file, nBlockPos, SEEK_SET) != 0) + { + fclose(file); + return NULL; + } + } + return file; +} + +static unsigned int nCurrentBlockFile = 1; + +FILE* AppendBlockFile(unsigned int& nFileRet) +{ + nFileRet = 0; + loop + { + FILE* file = OpenBlockFile(nCurrentBlockFile, 0, "ab"); + if (!file) + return NULL; + if (fseek(file, 0, SEEK_END) != 0) + return NULL; + // FAT32 filesize max 4GB, fseek and ftell max 2GB, so we must stay under 2GB + if (ftell(file) < 0x7F000000 - MAX_SIZE) + { + nFileRet = nCurrentBlockFile; + return file; + } + fclose(file); + nCurrentBlockFile++; + } +} + +bool LoadBlockIndex(bool fAllowNew) +{ + // + // Load block index + // + CTxDB txdb("cr"); + if (!txdb.LoadBlockIndex()) + return false; + txdb.Close(); + + // + // Init with genesis block + // + if (mapBlockIndex.empty()) + { + if (!fAllowNew) + return false; + + + // Genesis Block: + // GetHash() = 0x000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f + // hashMerkleRoot = 0x4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b + // txNew.vin[0].scriptSig = 486604799 4 0x736B6E616220726F662074756F6C69616220646E6F63657320666F206B6E697262206E6F20726F6C6C65636E61684320393030322F6E614A2F33302073656D695420656854 + // txNew.vout[0].nValue = 5000000000 + // txNew.vout[0].scriptPubKey = 0x5F1DF16B2B704C8A578D0BBAF74D385CDE12C11EE50455F3C438EF4C3FBCF649B6DE611FEAE06279A60939E028A8D65C10B73071A6F16719274855FEB0FD8A6704 OP_CHECKSIG + // block.nVersion = 1 + // block.nTime = 1231006505 + // block.nBits = 0x1d00ffff + // block.nNonce = 2083236893 + // CBlock(hash=000000000019d6, ver=1, hashPrevBlock=00000000000000, hashMerkleRoot=4a5e1e, nTime=1231006505, nBits=1d00ffff, nNonce=2083236893, vtx=1) + // CTransaction(hash=4a5e1e, ver=1, vin.size=1, vout.size=1, nLockTime=0) + // CTxIn(COutPoint(000000, -1), coinbase 04ffff001d0104455468652054696d65732030332f4a616e2f32303039204368616e63656c6c6f72206f6e206272696e6b206f66207365636f6e64206261696c6f757420666f722062616e6b73) + // CTxOut(nValue=50.00000000, scriptPubKey=0x5F1DF16B2B704C8A578D0B) + // vMerkleTree: 4a5e1e + + // Genesis block + char* pszTimestamp = "The Times 03/Jan/2009 Chancellor on brink of second bailout for banks"; + CTransaction txNew; + txNew.vin.resize(1); + txNew.vout.resize(1); + txNew.vin[0].scriptSig = CScript() << 486604799 << CBigNum(4) << vector((unsigned char*)pszTimestamp, (unsigned char*)pszTimestamp + strlen(pszTimestamp)); + txNew.vout[0].nValue = 50 * COIN; + txNew.vout[0].scriptPubKey = CScript() << CBigNum("0x5F1DF16B2B704C8A578D0BBAF74D385CDE12C11EE50455F3C438EF4C3FBCF649B6DE611FEAE06279A60939E028A8D65C10B73071A6F16719274855FEB0FD8A6704") << OP_CHECKSIG; + CBlock block; + block.vtx.push_back(txNew); + block.hashPrevBlock = 0; + block.hashMerkleRoot = block.BuildMerkleTree(); + block.nVersion = 1; + block.nTime = 1231006505; + block.nBits = 0x1d00ffff; + block.nNonce = 2083236893; + + //// debug print, delete this later + printf("%s\n", block.GetHash().ToString().c_str()); + printf("%s\n", block.hashMerkleRoot.ToString().c_str()); + printf("%s\n", hashGenesisBlock.ToString().c_str()); + txNew.vout[0].scriptPubKey.print(); + block.print(); + assert(block.hashMerkleRoot == uint256("0x4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b")); + + assert(block.GetHash() == hashGenesisBlock); + + // Start new block file + unsigned int nFile; + unsigned int nBlockPos; + if (!block.WriteToDisk(!fClient, nFile, nBlockPos)) + return error("LoadBlockIndex() : writing genesis block to disk failed"); + if (!block.AddToBlockIndex(nFile, nBlockPos)) + return error("LoadBlockIndex() : genesis block not accepted"); + } + + return true; +} + + + +void PrintBlockTree() +{ + // precompute tree structure + map > mapNext; + for (map::iterator mi = mapBlockIndex.begin(); mi != mapBlockIndex.end(); ++mi) + { + CBlockIndex* pindex = (*mi).second; + mapNext[pindex->pprev].push_back(pindex); + // test + //while (rand() % 3 == 0) + // mapNext[pindex->pprev].push_back(pindex); + } + + vector > vStack; + vStack.push_back(make_pair(0, pindexGenesisBlock)); + + int nPrevCol = 0; + while (!vStack.empty()) + { + int nCol = vStack.back().first; + CBlockIndex* pindex = vStack.back().second; + vStack.pop_back(); + + // print split or gap + if (nCol > nPrevCol) + { + for (int i = 0; i < nCol-1; i++) + printf("| "); + printf("|\\\n"); + } + else if (nCol < nPrevCol) + { + for (int i = 0; i < nCol; i++) + printf("| "); + printf("|\n"); + } + nPrevCol = nCol; + + // print columns + for (int i = 0; i < nCol; i++) + printf("| "); + + // print item + CBlock block; + block.ReadFromDisk(pindex, true); + printf("%d (%u,%u) %s %s tx %d", + pindex->nHeight, + pindex->nFile, + pindex->nBlockPos, + block.GetHash().ToString().substr(0,14).c_str(), + DateTimeStr(block.nTime).c_str(), + block.vtx.size()); + + CRITICAL_BLOCK(cs_mapWallet) + { + if (mapWallet.count(block.vtx[0].GetHash())) + { + CWalletTx& wtx = mapWallet[block.vtx[0].GetHash()]; + printf(" mine: %d %d %d", wtx.GetDepthInMainChain(), wtx.GetBlocksToMaturity(), wtx.GetCredit()); + } + } + printf("\n"); + + + // put the main timechain first + vector& vNext = mapNext[pindex]; + for (int i = 0; i < vNext.size(); i++) + { + if (vNext[i]->pnext) + { + swap(vNext[0], vNext[i]); + break; + } + } + + // iterate children + for (int i = 0; i < vNext.size(); i++) + vStack.push_back(make_pair(nCol+i, vNext[i])); + } +} + + + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// Messages +// + + +bool AlreadyHave(CTxDB& txdb, const CInv& inv) +{ + switch (inv.type) + { + case MSG_TX: return mapTransactions.count(inv.hash) || txdb.ContainsTx(inv.hash); + case MSG_BLOCK: return mapBlockIndex.count(inv.hash) || mapOrphanBlocks.count(inv.hash); + case MSG_REVIEW: return true; + case MSG_PRODUCT: return mapProducts.count(inv.hash); + } + // Don't know what it is, just say we already got one + return true; +} + + + + + + + +bool ProcessMessages(CNode* pfrom) +{ + CDataStream& vRecv = pfrom->vRecv; + if (vRecv.empty()) + return true; + printf("ProcessMessages(%d bytes)\n", vRecv.size()); + + // + // Message format + // (4) message start + // (12) command + // (4) size + // (x) data + // + + loop + { + // Scan for message start + CDataStream::iterator pstart = search(vRecv.begin(), vRecv.end(), BEGIN(pchMessageStart), END(pchMessageStart)); + if (vRecv.end() - pstart < sizeof(CMessageHeader)) + { + if (vRecv.size() > sizeof(CMessageHeader)) + { + printf("\n\nPROCESSMESSAGE MESSAGESTART NOT FOUND\n\n"); + vRecv.erase(vRecv.begin(), vRecv.end() - sizeof(CMessageHeader)); + } + break; + } + if (pstart - vRecv.begin() > 0) + printf("\n\nPROCESSMESSAGE SKIPPED %d BYTES\n\n", pstart - vRecv.begin()); + vRecv.erase(vRecv.begin(), pstart); + + // Read header + CMessageHeader hdr; + vRecv >> hdr; + if (!hdr.IsValid()) + { + printf("\n\nPROCESSMESSAGE: ERRORS IN HEADER %s\n\n\n", hdr.GetCommand().c_str()); + continue; + } + string strCommand = hdr.GetCommand(); + + // Message size + unsigned int nMessageSize = hdr.nMessageSize; + if (nMessageSize > vRecv.size()) + { + // Rewind and wait for rest of message + ///// need a mechanism to give up waiting for overlong message size error + printf("MESSAGE-BREAK\n"); + vRecv.insert(vRecv.begin(), BEGIN(hdr), END(hdr)); + Sleep(100); + break; + } + + // Copy message to its own buffer + CDataStream vMsg(vRecv.begin(), vRecv.begin() + nMessageSize, vRecv.nType, vRecv.nVersion); + vRecv.ignore(nMessageSize); + + // Process message + bool fRet = false; + try + { + CheckForShutdown(2); + CRITICAL_BLOCK(cs_main) + fRet = ProcessMessage(pfrom, strCommand, vMsg); + CheckForShutdown(2); + } + CATCH_PRINT_EXCEPTION("ProcessMessage()") + if (!fRet) + printf("ProcessMessage(%s, %d bytes) from %s to %s FAILED\n", strCommand.c_str(), nMessageSize, pfrom->addr.ToString().c_str(), addrLocalHost.ToString().c_str()); + } + + vRecv.Compact(); + return true; +} + + + + +bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) +{ + static map > mapReuseKey; + printf("received: %-12s (%d bytes) ", strCommand.c_str(), vRecv.size()); + for (int i = 0; i < min(vRecv.size(), (unsigned int)20); i++) + printf("%02x ", vRecv[i] & 0xff); + printf("\n"); + if (nDropMessagesTest > 0 && GetRand(nDropMessagesTest) == 0) + { + printf("dropmessages DROPPING RECV MESSAGE\n"); + return true; + } + + + + if (strCommand == "version") + { + // Can only do this once + if (pfrom->nVersion != 0) + return false; + + int64 nTime; + CAddress addrMe; + vRecv >> pfrom->nVersion >> pfrom->nServices >> nTime >> addrMe; + if (pfrom->nVersion == 0) + return false; + + pfrom->vSend.SetVersion(min(pfrom->nVersion, VERSION)); + pfrom->vRecv.SetVersion(min(pfrom->nVersion, VERSION)); + + pfrom->fClient = !(pfrom->nServices & NODE_NETWORK); + if (pfrom->fClient) + { + pfrom->vSend.nType |= SER_BLOCKHEADERONLY; + pfrom->vRecv.nType |= SER_BLOCKHEADERONLY; + } + + AddTimeData(pfrom->addr.ip, nTime); + + // Ask the first connected node for block updates + static bool fAskedForBlocks; + if (!fAskedForBlocks && !pfrom->fClient) + { + fAskedForBlocks = true; + pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), uint256(0)); + } + + printf("version message: %s has version %d, addrMe=%s\n", pfrom->addr.ToString().c_str(), pfrom->nVersion, addrMe.ToString().c_str()); + } + + + else if (pfrom->nVersion == 0) + { + // Must have a version message before anything else + return false; + } + + + else if (strCommand == "addr") + { + vector vAddr; + vRecv >> vAddr; + + // Store the new addresses + CAddrDB addrdb; + foreach(const CAddress& addr, vAddr) + { + if (fShutdown) + return true; + if (AddAddress(addrdb, addr)) + { + // Put on lists to send to other nodes + pfrom->setAddrKnown.insert(addr); + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + if (!pnode->setAddrKnown.count(addr)) + pnode->vAddrToSend.push_back(addr); + } + } + } + + + else if (strCommand == "inv") + { + vector vInv; + vRecv >> vInv; + + CTxDB txdb("r"); + foreach(const CInv& inv, vInv) + { + if (fShutdown) + return true; + pfrom->AddInventoryKnown(inv); + + bool fAlreadyHave = AlreadyHave(txdb, inv); + printf(" got inventory: %s %s\n", inv.ToString().c_str(), fAlreadyHave ? "have" : "new"); + + if (!fAlreadyHave) + pfrom->AskFor(inv); + else if (inv.type == MSG_BLOCK && mapOrphanBlocks.count(inv.hash)) + pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), GetOrphanRoot(mapOrphanBlocks[inv.hash])); + } + } + + + else if (strCommand == "getdata") + { + vector vInv; + vRecv >> vInv; + + foreach(const CInv& inv, vInv) + { + if (fShutdown) + return true; + printf("received getdata for: %s\n", inv.ToString().c_str()); + + if (inv.type == MSG_BLOCK) + { + // Send block from disk + map::iterator mi = mapBlockIndex.find(inv.hash); + if (mi != mapBlockIndex.end()) + { + //// could optimize this to send header straight from blockindex for client + CBlock block; + block.ReadFromDisk((*mi).second, !pfrom->fClient); + pfrom->PushMessage("block", block); + } + } + else if (inv.IsKnownType()) + { + // Send stream from relay memory + CRITICAL_BLOCK(cs_mapRelay) + { + map::iterator mi = mapRelay.find(inv); + if (mi != mapRelay.end()) + pfrom->PushMessage(inv.GetCommand(), (*mi).second); + } + } + } + } + + + else if (strCommand == "getblocks") + { + CBlockLocator locator; + uint256 hashStop; + vRecv >> locator >> hashStop; + + // Find the first block the caller has in the main chain + CBlockIndex* pindex = locator.GetBlockIndex(); + + // Send the rest of the chain + if (pindex) + pindex = pindex->pnext; + printf("getblocks %d to %s\n", (pindex ? pindex->nHeight : -1), hashStop.ToString().substr(0,14).c_str()); + for (; pindex; pindex = pindex->pnext) + { + if (pindex->GetBlockHash() == hashStop) + { + printf(" getblocks stopping at %d %s\n", pindex->nHeight, pindex->GetBlockHash().ToString().substr(0,14).c_str()); + break; + } + + // Bypass setInventoryKnown in case an inventory message got lost + CRITICAL_BLOCK(pfrom->cs_inventory) + { + CInv inv(MSG_BLOCK, pindex->GetBlockHash()); + // returns true if wasn't already contained in the set + if (pfrom->setInventoryKnown2.insert(inv).second) + { + pfrom->setInventoryKnown.erase(inv); + pfrom->vInventoryToSend.push_back(inv); + } + } + } + } + + + else if (strCommand == "tx") + { + vector vWorkQueue; + CDataStream vMsg(vRecv); + CTransaction tx; + vRecv >> tx; + + CInv inv(MSG_TX, tx.GetHash()); + pfrom->AddInventoryKnown(inv); + + bool fMissingInputs = false; + if (tx.AcceptTransaction(true, &fMissingInputs)) + { + AddToWalletIfMine(tx, NULL); + RelayMessage(inv, vMsg); + mapAlreadyAskedFor.erase(inv); + vWorkQueue.push_back(inv.hash); + + // Recursively process any orphan transactions that depended on this one + for (int i = 0; i < vWorkQueue.size(); i++) + { + uint256 hashPrev = vWorkQueue[i]; + for (multimap::iterator mi = mapOrphanTransactionsByPrev.lower_bound(hashPrev); + mi != mapOrphanTransactionsByPrev.upper_bound(hashPrev); + ++mi) + { + const CDataStream& vMsg = *((*mi).second); + CTransaction tx; + CDataStream(vMsg) >> tx; + CInv inv(MSG_TX, tx.GetHash()); + + if (tx.AcceptTransaction(true)) + { + printf(" accepted orphan tx %s\n", inv.hash.ToString().substr(0,6).c_str()); + AddToWalletIfMine(tx, NULL); + RelayMessage(inv, vMsg); + mapAlreadyAskedFor.erase(inv); + vWorkQueue.push_back(inv.hash); + } + } + } + + foreach(uint256 hash, vWorkQueue) + EraseOrphanTx(hash); + } + else if (fMissingInputs) + { + printf("storing orphan tx %s\n", inv.hash.ToString().substr(0,6).c_str()); + AddOrphanTx(vMsg); + } + } + + + else if (strCommand == "review") + { + CDataStream vMsg(vRecv); + CReview review; + vRecv >> review; + + CInv inv(MSG_REVIEW, review.GetHash()); + pfrom->AddInventoryKnown(inv); + + if (review.AcceptReview()) + { + // Relay the original message as-is in case it's a higher version than we know how to parse + RelayMessage(inv, vMsg); + mapAlreadyAskedFor.erase(inv); + } + } + + + else if (strCommand == "block") + { + auto_ptr pblock(new CBlock); + vRecv >> *pblock; + + //// debug print + printf("received block:\n"); pblock->print(); + + CInv inv(MSG_BLOCK, pblock->GetHash()); + pfrom->AddInventoryKnown(inv); + + if (ProcessBlock(pfrom, pblock.release())) + mapAlreadyAskedFor.erase(inv); + } + + + else if (strCommand == "getaddr") + { + pfrom->vAddrToSend.clear(); + int64 nSince = GetAdjustedTime() - 5 * 24 * 60 * 60; // in the last 5 days + CRITICAL_BLOCK(cs_mapAddresses) + { + unsigned int nSize = mapAddresses.size(); + foreach(const PAIRTYPE(vector, CAddress)& item, mapAddresses) + { + if (fShutdown) + return true; + const CAddress& addr = item.second; + //// will need this if we lose IRC + //if (addr.nTime > nSince || (rand() % nSize) < 500) + if (addr.nTime > nSince) + pfrom->vAddrToSend.push_back(addr); + } + } + } + + + else if (strCommand == "checkorder") + { + uint256 hashReply; + CWalletTx order; + vRecv >> hashReply >> order; + + /// we have a chance to check the order here + + // Keep giving the same key to the same ip until they use it + if (!mapReuseKey.count(pfrom->addr.ip)) + mapReuseKey[pfrom->addr.ip] = GenerateNewKey(); + + // Send back approval of order and pubkey to use + CScript scriptPubKey; + scriptPubKey << mapReuseKey[pfrom->addr.ip] << OP_CHECKSIG; + pfrom->PushMessage("reply", hashReply, (int)0, scriptPubKey); + } + + + else if (strCommand == "submitorder") + { + uint256 hashReply; + CWalletTx wtxNew; + vRecv >> hashReply >> wtxNew; + + // Broadcast + if (!wtxNew.AcceptWalletTransaction()) + { + pfrom->PushMessage("reply", hashReply, (int)1); + return error("submitorder AcceptWalletTransaction() failed, returning error 1"); + } + wtxNew.fTimeReceivedIsTxTime = true; + AddToWallet(wtxNew); + wtxNew.RelayWalletTransaction(); + mapReuseKey.erase(pfrom->addr.ip); + + // Send back confirmation + pfrom->PushMessage("reply", hashReply, (int)0); + } + + + else if (strCommand == "reply") + { + uint256 hashReply; + vRecv >> hashReply; + + CRequestTracker tracker; + CRITICAL_BLOCK(pfrom->cs_mapRequests) + { + map::iterator mi = pfrom->mapRequests.find(hashReply); + if (mi != pfrom->mapRequests.end()) + { + tracker = (*mi).second; + pfrom->mapRequests.erase(mi); + } + } + if (!tracker.IsNull()) + tracker.fn(tracker.param1, vRecv); + } + + + else + { + // Ignore unknown commands for extensibility + printf("ProcessMessage(%s) : Ignored unknown message\n", strCommand.c_str()); + } + + + if (!vRecv.empty()) + printf("ProcessMessage(%s) : %d extra bytes\n", strCommand.c_str(), vRecv.size()); + + return true; +} + + + + + + + + + +bool SendMessages(CNode* pto) +{ + CheckForShutdown(2); + CRITICAL_BLOCK(cs_main) + { + // Don't send anything until we get their version message + if (pto->nVersion == 0) + return true; + + + // + // Message: addr + // + vector vAddrToSend; + vAddrToSend.reserve(pto->vAddrToSend.size()); + foreach(const CAddress& addr, pto->vAddrToSend) + if (!pto->setAddrKnown.count(addr)) + vAddrToSend.push_back(addr); + pto->vAddrToSend.clear(); + if (!vAddrToSend.empty()) + pto->PushMessage("addr", vAddrToSend); + + + // + // Message: inventory + // + vector vInventoryToSend; + CRITICAL_BLOCK(pto->cs_inventory) + { + vInventoryToSend.reserve(pto->vInventoryToSend.size()); + foreach(const CInv& inv, pto->vInventoryToSend) + { + // returns true if wasn't already contained in the set + if (pto->setInventoryKnown.insert(inv).second) + vInventoryToSend.push_back(inv); + } + pto->vInventoryToSend.clear(); + pto->setInventoryKnown2.clear(); + } + if (!vInventoryToSend.empty()) + pto->PushMessage("inv", vInventoryToSend); + + + // + // Message: getdata + // + vector vAskFor; + int64 nNow = GetTime() * 1000000; + CTxDB txdb("r"); + while (!pto->mapAskFor.empty() && (*pto->mapAskFor.begin()).first <= nNow) + { + const CInv& inv = (*pto->mapAskFor.begin()).second; + printf("sending getdata: %s\n", inv.ToString().c_str()); + if (!AlreadyHave(txdb, inv)) + vAskFor.push_back(inv); + pto->mapAskFor.erase(pto->mapAskFor.begin()); + } + if (!vAskFor.empty()) + pto->PushMessage("getdata", vAskFor); + + } + return true; +} + + + + + + + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// BitcoinMiner +// + +int FormatHashBlocks(void* pbuffer, unsigned int len) +{ + unsigned char* pdata = (unsigned char*)pbuffer; + unsigned int blocks = 1 + ((len + 8) / 64); + unsigned char* pend = pdata + 64 * blocks; + memset(pdata + len, 0, 64 * blocks - len); + pdata[len] = 0x80; + unsigned int bits = len * 8; + pend[-1] = (bits >> 0) & 0xff; + pend[-2] = (bits >> 8) & 0xff; + pend[-3] = (bits >> 16) & 0xff; + pend[-4] = (bits >> 24) & 0xff; + return blocks; +} + +using CryptoPP::ByteReverse; +static int detectlittleendian = 1; + +void BlockSHA256(const void* pin, unsigned int nBlocks, void* pout) +{ + unsigned int* pinput = (unsigned int*)pin; + unsigned int* pstate = (unsigned int*)pout; + + CryptoPP::SHA256::InitState(pstate); + + if (*(char*)&detectlittleendian != 0) + { + for (int n = 0; n < nBlocks; n++) + { + unsigned int pbuf[16]; + for (int i = 0; i < 16; i++) + pbuf[i] = ByteReverse(pinput[n * 16 + i]); + CryptoPP::SHA256::Transform(pstate, pbuf); + } + for (int i = 0; i < 8; i++) + pstate[i] = ByteReverse(pstate[i]); + } + else + { + for (int n = 0; n < nBlocks; n++) + CryptoPP::SHA256::Transform(pstate, pinput + n * 16); + } +} + + +bool BitcoinMiner() +{ + printf("BitcoinMiner started\n"); + SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); + + CKey key; + key.MakeNewKey(); + CBigNum bnExtraNonce = 0; + while (fGenerateBitcoins) + { + Sleep(50); + CheckForShutdown(3); + while (vNodes.empty()) + { + Sleep(1000); + CheckForShutdown(3); + } + + unsigned int nTransactionsUpdatedLast = nTransactionsUpdated; + CBlockIndex* pindexPrev = pindexBest; + unsigned int nBits = GetNextWorkRequired(pindexPrev); + + + // + // Create coinbase tx + // + CTransaction txNew; + txNew.vin.resize(1); + txNew.vin[0].prevout.SetNull(); + txNew.vin[0].scriptSig << nBits << ++bnExtraNonce; + txNew.vout.resize(1); + txNew.vout[0].scriptPubKey << key.GetPubKey() << OP_CHECKSIG; + + + // + // Create new block + // + auto_ptr pblock(new CBlock()); + if (!pblock.get()) + return false; + + // Add our coinbase tx as first transaction + pblock->vtx.push_back(txNew); + + // Collect the latest transactions into the block + int64 nFees = 0; + CRITICAL_BLOCK(cs_main) + CRITICAL_BLOCK(cs_mapTransactions) + { + CTxDB txdb("r"); + map mapTestPool; + vector vfAlreadyAdded(mapTransactions.size()); + bool fFoundSomething = true; + unsigned int nBlockSize = 0; + while (fFoundSomething && nBlockSize < MAX_SIZE/2) + { + fFoundSomething = false; + unsigned int n = 0; + for (map::iterator mi = mapTransactions.begin(); mi != mapTransactions.end(); ++mi, ++n) + { + if (vfAlreadyAdded[n]) + continue; + CTransaction& tx = (*mi).second; + if (tx.IsCoinBase() || !tx.IsFinal()) + continue; + + // Transaction fee requirements, mainly only needed for flood control + // Under 10K (about 80 inputs) is free for first 100 transactions + // Base rate is 0.01 per KB + int64 nMinFee = tx.GetMinFee(pblock->vtx.size() < 100); + + map mapTestPoolTmp(mapTestPool); + if (!tx.ConnectInputs(txdb, mapTestPoolTmp, CDiskTxPos(1,1,1), 0, nFees, false, true, nMinFee)) + continue; + swap(mapTestPool, mapTestPoolTmp); + + pblock->vtx.push_back(tx); + nBlockSize += ::GetSerializeSize(tx, SER_NETWORK); + vfAlreadyAdded[n] = true; + fFoundSomething = true; + } + } + } + pblock->nBits = nBits; + pblock->vtx[0].vout[0].nValue = pblock->GetBlockValue(nFees); + printf("\n\nRunning BitcoinMiner with %d transactions in block\n", pblock->vtx.size()); + + + // + // Prebuild hash buffer + // + struct unnamed1 + { + struct unnamed2 + { + int nVersion; + uint256 hashPrevBlock; + uint256 hashMerkleRoot; + unsigned int nTime; + unsigned int nBits; + unsigned int nNonce; + } + block; + unsigned char pchPadding0[64]; + uint256 hash1; + unsigned char pchPadding1[64]; + } + tmp; + + tmp.block.nVersion = pblock->nVersion; + tmp.block.hashPrevBlock = pblock->hashPrevBlock = (pindexPrev ? pindexPrev->GetBlockHash() : 0); + tmp.block.hashMerkleRoot = pblock->hashMerkleRoot = pblock->BuildMerkleTree(); + tmp.block.nTime = pblock->nTime = max((pindexPrev ? pindexPrev->GetMedianTimePast()+1 : 0), GetAdjustedTime()); + tmp.block.nBits = pblock->nBits = nBits; + tmp.block.nNonce = pblock->nNonce = 1; + + unsigned int nBlocks0 = FormatHashBlocks(&tmp.block, sizeof(tmp.block)); + unsigned int nBlocks1 = FormatHashBlocks(&tmp.hash1, sizeof(tmp.hash1)); + + + // + // Search + // + unsigned int nStart = GetTime(); + uint256 hashTarget = CBigNum().SetCompact(pblock->nBits).getuint256(); + uint256 hash; + loop + { + BlockSHA256(&tmp.block, nBlocks0, &tmp.hash1); + BlockSHA256(&tmp.hash1, nBlocks1, &hash); + + + if (hash <= hashTarget) + { + pblock->nNonce = tmp.block.nNonce; + assert(hash == pblock->GetHash()); + + //// debug print + printf("BitcoinMiner:\n"); + printf("proof-of-work found \n hash: %s \ntarget: %s\n", hash.GetHex().c_str(), hashTarget.GetHex().c_str()); + pblock->print(); + + SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL); + CRITICAL_BLOCK(cs_main) + { + // Save key + if (!AddKey(key)) + return false; + key.MakeNewKey(); + + // Process this block the same as if we had received it from another node + if (!ProcessBlock(NULL, pblock.release())) + printf("ERROR in BitcoinMiner, ProcessBlock, block not accepted\n"); + } + SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); + + Sleep(500); + break; + } + + // Update nTime every few seconds + if ((++tmp.block.nNonce & 0x3ffff) == 0) + { + CheckForShutdown(3); + if (tmp.block.nNonce == 0) + break; + if (pindexPrev != pindexBest) + break; + if (nTransactionsUpdated != nTransactionsUpdatedLast && GetTime() - nStart > 60) + break; + if (!fGenerateBitcoins) + break; + tmp.block.nTime = pblock->nTime = max(pindexPrev->GetMedianTimePast()+1, GetAdjustedTime()); + } + } + } + + return true; +} + + + + + + + + + + + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// Actions +// + + +int64 GetBalance() +{ + int64 nStart, nEnd; + QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + + int64 nTotal = 0; + CRITICAL_BLOCK(cs_mapWallet) + { + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + CWalletTx* pcoin = &(*it).second; + if (!pcoin->IsFinal() || pcoin->fSpent) + continue; + nTotal += pcoin->GetCredit(); + } + } + + QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); + ///printf(" GetBalance() time = %16I64d\n", nEnd - nStart); + return nTotal; +} + + + +bool SelectCoins(int64 nTargetValue, set& setCoinsRet) +{ + setCoinsRet.clear(); + + // List of values less than target + int64 nLowestLarger = _I64_MAX; + CWalletTx* pcoinLowestLarger = NULL; + vector > vValue; + int64 nTotalLower = 0; + + CRITICAL_BLOCK(cs_mapWallet) + { + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + CWalletTx* pcoin = &(*it).second; + if (!pcoin->IsFinal() || pcoin->fSpent) + continue; + int64 n = pcoin->GetCredit(); + if (n <= 0) + continue; + if (n < nTargetValue) + { + vValue.push_back(make_pair(n, pcoin)); + nTotalLower += n; + } + else if (n == nTargetValue) + { + setCoinsRet.insert(pcoin); + return true; + } + else if (n < nLowestLarger) + { + nLowestLarger = n; + pcoinLowestLarger = pcoin; + } + } + } + + if (nTotalLower < nTargetValue) + { + if (pcoinLowestLarger == NULL) + return false; + setCoinsRet.insert(pcoinLowestLarger); + return true; + } + + // Solve subset sum by stochastic approximation + sort(vValue.rbegin(), vValue.rend()); + vector vfIncluded; + vector vfBest(vValue.size(), true); + int64 nBest = nTotalLower; + + for (int nRep = 0; nRep < 1000 && nBest != nTargetValue; nRep++) + { + vfIncluded.assign(vValue.size(), false); + int64 nTotal = 0; + bool fReachedTarget = false; + for (int nPass = 0; nPass < 2 && !fReachedTarget; nPass++) + { + for (int i = 0; i < vValue.size(); i++) + { + if (nPass == 0 ? rand() % 2 : !vfIncluded[i]) + { + nTotal += vValue[i].first; + vfIncluded[i] = true; + if (nTotal >= nTargetValue) + { + fReachedTarget = true; + if (nTotal < nBest) + { + nBest = nTotal; + vfBest = vfIncluded; + } + nTotal -= vValue[i].first; + vfIncluded[i] = false; + } + } + } + } + } + + // If the next larger is still closer, return it + if (pcoinLowestLarger && nLowestLarger - nTargetValue <= nBest - nTargetValue) + setCoinsRet.insert(pcoinLowestLarger); + else + { + for (int i = 0; i < vValue.size(); i++) + if (vfBest[i]) + setCoinsRet.insert(vValue[i].second); + + //// debug print + printf("SelectCoins() best subset: "); + for (int i = 0; i < vValue.size(); i++) + if (vfBest[i]) + printf("%s ", FormatMoney(vValue[i].first).c_str()); + printf("total %s\n", FormatMoney(nBest).c_str()); + } + + return true; +} + + + + +bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, int64& nFeeRequiredRet) +{ + nFeeRequiredRet = 0; + CRITICAL_BLOCK(cs_main) + { + // txdb must be opened before the mapWallet lock + CTxDB txdb("r"); + CRITICAL_BLOCK(cs_mapWallet) + { + int64 nFee = nTransactionFee; + loop + { + wtxNew.vin.clear(); + wtxNew.vout.clear(); + if (nValue < 0) + return false; + int64 nValueOut = nValue; + nValue += nFee; + + // Choose coins to use + set setCoins; + if (!SelectCoins(nValue, setCoins)) + return false; + int64 nValueIn = 0; + foreach(CWalletTx* pcoin, setCoins) + nValueIn += pcoin->GetCredit(); + + // Fill vout[0] to the payee + wtxNew.vout.push_back(CTxOut(nValueOut, scriptPubKey)); + + // Fill vout[1] back to self with any change + if (nValueIn > nValue) + { + /// todo: for privacy, should randomize the order of outputs, + // would also have to use a new key for the change. + // Use the same key as one of the coins + vector vchPubKey; + CTransaction& txFirst = *(*setCoins.begin()); + foreach(const CTxOut& txout, txFirst.vout) + if (txout.IsMine()) + if (ExtractPubKey(txout.scriptPubKey, true, vchPubKey)) + break; + if (vchPubKey.empty()) + return false; + + // Fill vout[1] to ourself + CScript scriptPubKey; + scriptPubKey << vchPubKey << OP_CHECKSIG; + wtxNew.vout.push_back(CTxOut(nValueIn - nValue, scriptPubKey)); + } + + // Fill vin + foreach(CWalletTx* pcoin, setCoins) + for (int nOut = 0; nOut < pcoin->vout.size(); nOut++) + if (pcoin->vout[nOut].IsMine()) + wtxNew.vin.push_back(CTxIn(pcoin->GetHash(), nOut)); + + // Sign + int nIn = 0; + foreach(CWalletTx* pcoin, setCoins) + for (int nOut = 0; nOut < pcoin->vout.size(); nOut++) + if (pcoin->vout[nOut].IsMine()) + SignSignature(*pcoin, wtxNew, nIn++); + + // Check that enough fee is included + if (nFee < wtxNew.GetMinFee(true)) + { + nFee = nFeeRequiredRet = wtxNew.GetMinFee(true); + continue; + } + + // Fill vtxPrev by copying from previous transactions vtxPrev + wtxNew.AddSupportingTransactions(txdb); + wtxNew.fTimeReceivedIsTxTime = true; + + break; + } + } + } + return true; +} + +// Call after CreateTransaction unless you want to abort +bool CommitTransactionSpent(const CWalletTx& wtxNew) +{ + CRITICAL_BLOCK(cs_main) + CRITICAL_BLOCK(cs_mapWallet) + { + //// todo: make this transactional, never want to add a transaction + //// without marking spent transactions + + // Add tx to wallet, because if it has change it's also ours, + // otherwise just for transaction history. + AddToWallet(wtxNew); + + // Mark old coins as spent + set setCoins; + foreach(const CTxIn& txin, wtxNew.vin) + setCoins.insert(&mapWallet[txin.prevout.hash]); + foreach(CWalletTx* pcoin, setCoins) + { + pcoin->fSpent = true; + pcoin->WriteToDisk(); + vWalletUpdated.push_back(make_pair(pcoin->GetHash(), false)); + } + } + MainFrameRepaint(); + return true; +} + + + + +bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) +{ + CRITICAL_BLOCK(cs_main) + { + int64 nFeeRequired; + if (!CreateTransaction(scriptPubKey, nValue, wtxNew, nFeeRequired)) + { + string strError; + if (nValue + nFeeRequired > GetBalance()) + strError = strprintf("Error: This is an oversized transaction that requires a transaction fee of %s ", FormatMoney(nFeeRequired).c_str()); + else + strError = "Error: Transaction creation failed "; + wxMessageBox(strError, "Sending..."); + return error("SendMoney() : %s\n", strError.c_str()); + } + if (!CommitTransactionSpent(wtxNew)) + { + wxMessageBox("Error finalizing transaction ", "Sending..."); + return error("SendMoney() : Error finalizing transaction"); + } + + printf("SendMoney: %s\n", wtxNew.GetHash().ToString().substr(0,6).c_str()); + + // Broadcast + if (!wtxNew.AcceptTransaction()) + { + // This must not fail. The transaction has already been signed and recorded. + throw runtime_error("SendMoney() : wtxNew.AcceptTransaction() failed\n"); + wxMessageBox("Error: Transaction not valid ", "Sending..."); + return error("SendMoney() : Error: Transaction not valid"); + } + wtxNew.RelayWalletTransaction(); + } + MainFrameRepaint(); + return true; +} diff --git a/main.h b/main.h new file mode 100644 index 00000000..3432b316 --- /dev/null +++ b/main.h @@ -0,0 +1,1329 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +class COutPoint; +class CInPoint; +class CDiskTxPos; +class CCoinBase; +class CTxIn; +class CTxOut; +class CTransaction; +class CBlock; +class CBlockIndex; +class CWalletTx; +class CKeyItem; + +static const unsigned int MAX_SIZE = 0x02000000; +static const int64 COIN = 100000000; +static const int64 CENT = 1000000; +static const int COINBASE_MATURITY = 100; + +static const CBigNum bnProofOfWorkLimit(~uint256(0) >> 32); + + + + + + +extern CCriticalSection cs_main; +extern map mapBlockIndex; +extern const uint256 hashGenesisBlock; +extern CBlockIndex* pindexGenesisBlock; +extern int nBestHeight; +extern uint256 hashBestChain; +extern CBlockIndex* pindexBest; +extern unsigned int nTransactionsUpdated; +extern string strSetDataDir; +extern int nDropMessagesTest; + +// Settings +extern int fGenerateBitcoins; +extern int64 nTransactionFee; +extern CAddress addrIncoming; + + + + + + + +string GetAppDir(); +bool CheckDiskSpace(int64 nAdditionalBytes=0); +FILE* OpenBlockFile(unsigned int nFile, unsigned int nBlockPos, const char* pszMode="rb"); +FILE* AppendBlockFile(unsigned int& nFileRet); +bool AddKey(const CKey& key); +vector GenerateNewKey(); +bool AddToWallet(const CWalletTx& wtxIn); +void ReacceptWalletTransactions(); +void RelayWalletTransactions(); +bool LoadBlockIndex(bool fAllowNew=true); +void PrintBlockTree(); +bool BitcoinMiner(); +bool ProcessMessages(CNode* pfrom); +bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv); +bool SendMessages(CNode* pto); +int64 GetBalance(); +bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& txNew, int64& nFeeRequiredRet); +bool CommitTransactionSpent(const CWalletTx& wtxNew); +bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew); + + + + + + + + + + + +class CDiskTxPos +{ +public: + unsigned int nFile; + unsigned int nBlockPos; + unsigned int nTxPos; + + CDiskTxPos() + { + SetNull(); + } + + CDiskTxPos(unsigned int nFileIn, unsigned int nBlockPosIn, unsigned int nTxPosIn) + { + nFile = nFileIn; + nBlockPos = nBlockPosIn; + nTxPos = nTxPosIn; + } + + IMPLEMENT_SERIALIZE( READWRITE(FLATDATA(*this)); ) + void SetNull() { nFile = -1; nBlockPos = 0; nTxPos = 0; } + bool IsNull() const { return (nFile == -1); } + + friend bool operator==(const CDiskTxPos& a, const CDiskTxPos& b) + { + return (a.nFile == b.nFile && + a.nBlockPos == b.nBlockPos && + a.nTxPos == b.nTxPos); + } + + friend bool operator!=(const CDiskTxPos& a, const CDiskTxPos& b) + { + return !(a == b); + } + + string ToString() const + { + if (IsNull()) + return strprintf("null"); + else + return strprintf("(nFile=%d, nBlockPos=%d, nTxPos=%d)", nFile, nBlockPos, nTxPos); + } + + void print() const + { + printf("%s", ToString().c_str()); + } +}; + + + + +class CInPoint +{ +public: + CTransaction* ptx; + unsigned int n; + + CInPoint() { SetNull(); } + CInPoint(CTransaction* ptxIn, unsigned int nIn) { ptx = ptxIn; n = nIn; } + void SetNull() { ptx = NULL; n = -1; } + bool IsNull() const { return (ptx == NULL && n == -1); } +}; + + + + +class COutPoint +{ +public: + uint256 hash; + unsigned int n; + + COutPoint() { SetNull(); } + COutPoint(uint256 hashIn, unsigned int nIn) { hash = hashIn; n = nIn; } + IMPLEMENT_SERIALIZE( READWRITE(FLATDATA(*this)); ) + void SetNull() { hash = 0; n = -1; } + bool IsNull() const { return (hash == 0 && n == -1); } + + friend bool operator<(const COutPoint& a, const COutPoint& b) + { + return (a.hash < b.hash || (a.hash == b.hash && a.n < b.n)); + } + + friend bool operator==(const COutPoint& a, const COutPoint& b) + { + return (a.hash == b.hash && a.n == b.n); + } + + friend bool operator!=(const COutPoint& a, const COutPoint& b) + { + return !(a == b); + } + + string ToString() const + { + return strprintf("COutPoint(%s, %d)", hash.ToString().substr(0,6).c_str(), n); + } + + void print() const + { + printf("%s\n", ToString().c_str()); + } +}; + + + + +// +// An input of a transaction. It contains the location of the previous +// transaction's output that it claims and a signature that matches the +// output's public key. +// +class CTxIn +{ +public: + COutPoint prevout; + CScript scriptSig; + unsigned int nSequence; + + CTxIn() + { + nSequence = UINT_MAX; + } + + explicit CTxIn(COutPoint prevoutIn, CScript scriptSigIn=CScript(), unsigned int nSequenceIn=UINT_MAX) + { + prevout = prevoutIn; + scriptSig = scriptSigIn; + nSequence = nSequenceIn; + } + + CTxIn(uint256 hashPrevTx, unsigned int nOut, CScript scriptSigIn=CScript(), unsigned int nSequenceIn=UINT_MAX) + { + prevout = COutPoint(hashPrevTx, nOut); + scriptSig = scriptSigIn; + nSequence = nSequenceIn; + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(prevout); + READWRITE(scriptSig); + READWRITE(nSequence); + ) + + bool IsFinal() const + { + return (nSequence == UINT_MAX); + } + + friend bool operator==(const CTxIn& a, const CTxIn& b) + { + return (a.prevout == b.prevout && + a.scriptSig == b.scriptSig && + a.nSequence == b.nSequence); + } + + friend bool operator!=(const CTxIn& a, const CTxIn& b) + { + return !(a == b); + } + + string ToString() const + { + string str; + str += strprintf("CTxIn("); + str += prevout.ToString(); + if (prevout.IsNull()) + str += strprintf(", coinbase %s", HexStr(scriptSig.begin(), scriptSig.end(), false).c_str()); + else + str += strprintf(", scriptSig=%s", scriptSig.ToString().substr(0,24).c_str()); + if (nSequence != UINT_MAX) + str += strprintf(", nSequence=%u", nSequence); + str += ")"; + return str; + } + + void print() const + { + printf("%s\n", ToString().c_str()); + } + + bool IsMine() const; + int64 GetDebit() const; +}; + + + + +// +// An output of a transaction. It contains the public key that the next input +// must be able to sign with to claim it. +// +class CTxOut +{ +public: + int64 nValue; + CScript scriptPubKey; + +public: + CTxOut() + { + SetNull(); + } + + CTxOut(int64 nValueIn, CScript scriptPubKeyIn) + { + nValue = nValueIn; + scriptPubKey = scriptPubKeyIn; + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(nValue); + READWRITE(scriptPubKey); + ) + + void SetNull() + { + nValue = -1; + scriptPubKey.clear(); + } + + bool IsNull() + { + return (nValue == -1); + } + + uint256 GetHash() const + { + return SerializeHash(*this); + } + + bool IsMine() const + { + return ::IsMine(scriptPubKey); + } + + int64 GetCredit() const + { + if (IsMine()) + return nValue; + return 0; + } + + friend bool operator==(const CTxOut& a, const CTxOut& b) + { + return (a.nValue == b.nValue && + a.scriptPubKey == b.scriptPubKey); + } + + friend bool operator!=(const CTxOut& a, const CTxOut& b) + { + return !(a == b); + } + + string ToString() const + { + if (scriptPubKey.size() < 6) + return "CTxOut(error)"; + return strprintf("CTxOut(nValue=%I64d.%08I64d, scriptPubKey=%s)", nValue / COIN, nValue % COIN, scriptPubKey.ToString().substr(0,24).c_str()); + } + + void print() const + { + printf("%s\n", ToString().c_str()); + } +}; + + + + +// +// The basic transaction that is broadcasted on the network and contained in +// blocks. A transaction can contain multiple inputs and outputs. +// +class CTransaction +{ +public: + int nVersion; + vector vin; + vector vout; + int nLockTime; + + + CTransaction() + { + SetNull(); + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(this->nVersion); + nVersion = this->nVersion; + READWRITE(vin); + READWRITE(vout); + READWRITE(nLockTime); + ) + + void SetNull() + { + nVersion = 1; + vin.clear(); + vout.clear(); + nLockTime = 0; + } + + bool IsNull() const + { + return (vin.empty() && vout.empty()); + } + + uint256 GetHash() const + { + return SerializeHash(*this); + } + + bool IsFinal() const + { + if (nLockTime == 0 || nLockTime < nBestHeight) + return true; + foreach(const CTxIn& txin, vin) + if (!txin.IsFinal()) + return false; + return true; + } + + bool IsNewerThan(const CTransaction& old) const + { + if (vin.size() != old.vin.size()) + return false; + for (int i = 0; i < vin.size(); i++) + if (vin[i].prevout != old.vin[i].prevout) + return false; + + bool fNewer = false; + unsigned int nLowest = UINT_MAX; + for (int i = 0; i < vin.size(); i++) + { + if (vin[i].nSequence != old.vin[i].nSequence) + { + if (vin[i].nSequence <= nLowest) + { + fNewer = false; + nLowest = vin[i].nSequence; + } + if (old.vin[i].nSequence < nLowest) + { + fNewer = true; + nLowest = old.vin[i].nSequence; + } + } + } + return fNewer; + } + + bool IsCoinBase() const + { + return (vin.size() == 1 && vin[0].prevout.IsNull()); + } + + bool CheckTransaction() const + { + // Basic checks that don't depend on any context + if (vin.empty() || vout.empty()) + return error("CTransaction::CheckTransaction() : vin or vout empty"); + + // Check for negative values + foreach(const CTxOut& txout, vout) + if (txout.nValue < 0) + return error("CTransaction::CheckTransaction() : txout.nValue negative"); + + if (IsCoinBase()) + { + if (vin[0].scriptSig.size() < 2 || vin[0].scriptSig.size() > 100) + return error("CTransaction::CheckTransaction() : coinbase script size"); + } + else + { + foreach(const CTxIn& txin, vin) + if (txin.prevout.IsNull()) + return error("CTransaction::CheckTransaction() : prevout is null"); + } + + return true; + } + + bool IsMine() const + { + foreach(const CTxOut& txout, vout) + if (txout.IsMine()) + return true; + return false; + } + + int64 GetDebit() const + { + int64 nDebit = 0; + foreach(const CTxIn& txin, vin) + nDebit += txin.GetDebit(); + return nDebit; + } + + int64 GetCredit() const + { + int64 nCredit = 0; + foreach(const CTxOut& txout, vout) + nCredit += txout.GetCredit(); + return nCredit; + } + + int64 GetValueOut() const + { + int64 nValueOut = 0; + foreach(const CTxOut& txout, vout) + { + if (txout.nValue < 0) + throw runtime_error("CTransaction::GetValueOut() : negative value"); + nValueOut += txout.nValue; + } + return nValueOut; + } + + int64 GetMinFee(bool fDiscount=false) const + { + // Base fee is 1 cent per kilobyte + unsigned int nBytes = ::GetSerializeSize(*this, SER_NETWORK); + int64 nMinFee = (1 + (int64)nBytes / 1000) * CENT; + + // First 100 transactions in a block are free + if (fDiscount && nBytes < 10000) + nMinFee = 0; + + // To limit dust spam, require a 0.01 fee if any output is less than 0.01 + if (nMinFee < CENT) + foreach(const CTxOut& txout, vout) + if (txout.nValue < CENT) + nMinFee = CENT; + + return nMinFee; + } + + + + bool ReadFromDisk(CDiskTxPos pos, FILE** pfileRet=NULL) + { + CAutoFile filein = OpenBlockFile(pos.nFile, 0, pfileRet ? "rb+" : "rb"); + if (!filein) + return error("CTransaction::ReadFromDisk() : OpenBlockFile failed"); + + // Read transaction + if (fseek(filein, pos.nTxPos, SEEK_SET) != 0) + return error("CTransaction::ReadFromDisk() : fseek failed"); + filein >> *this; + + // Return file pointer + if (pfileRet) + { + if (fseek(filein, pos.nTxPos, SEEK_SET) != 0) + return error("CTransaction::ReadFromDisk() : second fseek failed"); + *pfileRet = filein.release(); + } + return true; + } + + + friend bool operator==(const CTransaction& a, const CTransaction& b) + { + return (a.nVersion == b.nVersion && + a.vin == b.vin && + a.vout == b.vout && + a.nLockTime == b.nLockTime); + } + + friend bool operator!=(const CTransaction& a, const CTransaction& b) + { + return !(a == b); + } + + + string ToString() const + { + string str; + str += strprintf("CTransaction(hash=%s, ver=%d, vin.size=%d, vout.size=%d, nLockTime=%d)\n", + GetHash().ToString().substr(0,6).c_str(), + nVersion, + vin.size(), + vout.size(), + nLockTime); + for (int i = 0; i < vin.size(); i++) + str += " " + vin[i].ToString() + "\n"; + for (int i = 0; i < vout.size(); i++) + str += " " + vout[i].ToString() + "\n"; + return str; + } + + void print() const + { + printf("%s", ToString().c_str()); + } + + + + bool DisconnectInputs(CTxDB& txdb); + bool ConnectInputs(CTxDB& txdb, map& mapTestPool, CDiskTxPos posThisTx, int nHeight, int64& nFees, bool fBlock, bool fMiner, int64 nMinFee=0); + bool ClientConnectInputs(); + + bool AcceptTransaction(CTxDB& txdb, bool fCheckInputs=true, bool* pfMissingInputs=NULL); + + bool AcceptTransaction(bool fCheckInputs=true, bool* pfMissingInputs=NULL) + { + CTxDB txdb("r"); + return AcceptTransaction(txdb, fCheckInputs, pfMissingInputs); + } + +protected: + bool AddToMemoryPool(); +public: + bool RemoveFromMemoryPool(); +}; + + + + + +// +// A transaction with a merkle branch linking it to the block chain +// +class CMerkleTx : public CTransaction +{ +public: + uint256 hashBlock; + vector vMerkleBranch; + int nIndex; + + // memory only + mutable bool fMerkleVerified; + + + CMerkleTx() + { + Init(); + } + + CMerkleTx(const CTransaction& txIn) : CTransaction(txIn) + { + Init(); + } + + void Init() + { + hashBlock = 0; + nIndex = -1; + fMerkleVerified = false; + } + + int64 GetCredit() const + { + // Must wait until coinbase is safely deep enough in the chain before valuing it + if (IsCoinBase() && GetBlocksToMaturity() > 0) + return 0; + return CTransaction::GetCredit(); + } + + IMPLEMENT_SERIALIZE + ( + nSerSize += SerReadWrite(s, *(CTransaction*)this, nType, nVersion, ser_action); + nVersion = this->nVersion; + READWRITE(hashBlock); + READWRITE(vMerkleBranch); + READWRITE(nIndex); + ) + + + int SetMerkleBranch(const CBlock* pblock=NULL); + int GetDepthInMainChain() const; + bool IsInMainChain() const { return GetDepthInMainChain() > 0; } + int GetBlocksToMaturity() const; + bool AcceptTransaction(CTxDB& txdb, bool fCheckInputs=true); + bool AcceptTransaction() { CTxDB txdb("r"); return AcceptTransaction(txdb); } +}; + + + + +// +// A transaction with a bunch of additional info that only the owner cares +// about. It includes any unrecorded transactions needed to link it back +// to the block chain. +// +class CWalletTx : public CMerkleTx +{ +public: + vector vtxPrev; + map mapValue; + vector > vOrderForm; + unsigned int fTimeReceivedIsTxTime; + unsigned int nTimeReceived; // time received by this node + char fFromMe; + char fSpent; + //// probably need to sign the order info so know it came from payer + + // memory only + mutable unsigned int nTimeDisplayed; + + + CWalletTx() + { + Init(); + } + + CWalletTx(const CMerkleTx& txIn) : CMerkleTx(txIn) + { + Init(); + } + + CWalletTx(const CTransaction& txIn) : CMerkleTx(txIn) + { + Init(); + } + + void Init() + { + fTimeReceivedIsTxTime = false; + nTimeReceived = 0; + fFromMe = false; + fSpent = false; + nTimeDisplayed = 0; + } + + IMPLEMENT_SERIALIZE + ( + nSerSize += SerReadWrite(s, *(CMerkleTx*)this, nType, nVersion, ser_action); + nVersion = this->nVersion; + READWRITE(vtxPrev); + READWRITE(mapValue); + READWRITE(vOrderForm); + READWRITE(fTimeReceivedIsTxTime); + READWRITE(nTimeReceived); + READWRITE(fFromMe); + READWRITE(fSpent); + ) + + bool WriteToDisk() + { + return CWalletDB().WriteTx(GetHash(), *this); + } + + + int64 GetTxTime() const; + + void AddSupportingTransactions(CTxDB& txdb); + + bool AcceptWalletTransaction(CTxDB& txdb, bool fCheckInputs=true); + bool AcceptWalletTransaction() { CTxDB txdb("r"); return AcceptWalletTransaction(txdb); } + + void RelayWalletTransaction(CTxDB& txdb); + void RelayWalletTransaction() { CTxDB txdb("r"); RelayWalletTransaction(txdb); } +}; + + + + +// +// A txdb record that contains the disk location of a transaction and the +// locations of transactions that spend its outputs. vSpent is really only +// used as a flag, but having the location is very helpful for debugging. +// +class CTxIndex +{ +public: + CDiskTxPos pos; + vector vSpent; + + CTxIndex() + { + SetNull(); + } + + CTxIndex(const CDiskTxPos& posIn, unsigned int nOutputs) + { + pos = posIn; + vSpent.resize(nOutputs); + } + + IMPLEMENT_SERIALIZE + ( + if (!(nType & SER_GETHASH)) + READWRITE(nVersion); + READWRITE(pos); + READWRITE(vSpent); + ) + + void SetNull() + { + pos.SetNull(); + vSpent.clear(); + } + + bool IsNull() + { + return pos.IsNull(); + } + + friend bool operator==(const CTxIndex& a, const CTxIndex& b) + { + if (a.pos != b.pos || a.vSpent.size() != b.vSpent.size()) + return false; + for (int i = 0; i < a.vSpent.size(); i++) + if (a.vSpent[i] != b.vSpent[i]) + return false; + return true; + } + + friend bool operator!=(const CTxIndex& a, const CTxIndex& b) + { + return !(a == b); + } +}; + + + + + +// +// Nodes collect new transactions into a block, hash them into a hash tree, +// and scan through nonce values to make the block's hash satisfy proof-of-work +// requirements. When they solve the proof-of-work, they broadcast the block +// to everyone and the block is added to the block chain. The first transaction +// in the block is a special one that creates a new coin owned by the creator +// of the block. +// +// Blocks are appended to blk0001.dat files on disk. Their location on disk +// is indexed by CBlockIndex objects in memory. +// +class CBlock +{ +public: + // header + int nVersion; + uint256 hashPrevBlock; + uint256 hashMerkleRoot; + unsigned int nTime; + unsigned int nBits; + unsigned int nNonce; + + // network and disk + vector vtx; + + // memory only + mutable vector vMerkleTree; + + + CBlock() + { + SetNull(); + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(this->nVersion); + nVersion = this->nVersion; + READWRITE(hashPrevBlock); + READWRITE(hashMerkleRoot); + READWRITE(nTime); + READWRITE(nBits); + READWRITE(nNonce); + + // ConnectBlock depends on vtx being last so it can calculate offset + if (!(nType & (SER_GETHASH|SER_BLOCKHEADERONLY))) + READWRITE(vtx); + else if (fRead) + const_cast(this)->vtx.clear(); + ) + + void SetNull() + { + nVersion = 1; + hashPrevBlock = 0; + hashMerkleRoot = 0; + nTime = 0; + nBits = 0; + nNonce = 0; + vtx.clear(); + vMerkleTree.clear(); + } + + bool IsNull() const + { + return (nBits == 0); + } + + uint256 GetHash() const + { + return Hash(BEGIN(nVersion), END(nNonce)); + } + + + uint256 BuildMerkleTree() const + { + vMerkleTree.clear(); + foreach(const CTransaction& tx, vtx) + vMerkleTree.push_back(tx.GetHash()); + int j = 0; + for (int nSize = vtx.size(); nSize > 1; nSize = (nSize + 1) / 2) + { + for (int i = 0; i < nSize; i += 2) + { + int i2 = min(i+1, nSize-1); + vMerkleTree.push_back(Hash(BEGIN(vMerkleTree[j+i]), END(vMerkleTree[j+i]), + BEGIN(vMerkleTree[j+i2]), END(vMerkleTree[j+i2]))); + } + j += nSize; + } + return (vMerkleTree.empty() ? 0 : vMerkleTree.back()); + } + + vector GetMerkleBranch(int nIndex) const + { + if (vMerkleTree.empty()) + BuildMerkleTree(); + vector vMerkleBranch; + int j = 0; + for (int nSize = vtx.size(); nSize > 1; nSize = (nSize + 1) / 2) + { + int i = min(nIndex^1, nSize-1); + vMerkleBranch.push_back(vMerkleTree[j+i]); + nIndex >>= 1; + j += nSize; + } + return vMerkleBranch; + } + + static uint256 CheckMerkleBranch(uint256 hash, const vector& vMerkleBranch, int nIndex) + { + if (nIndex == -1) + return 0; + foreach(const uint256& otherside, vMerkleBranch) + { + if (nIndex & 1) + hash = Hash(BEGIN(otherside), END(otherside), BEGIN(hash), END(hash)); + else + hash = Hash(BEGIN(hash), END(hash), BEGIN(otherside), END(otherside)); + nIndex >>= 1; + } + return hash; + } + + + bool WriteToDisk(bool fWriteTransactions, unsigned int& nFileRet, unsigned int& nBlockPosRet) + { + // Open history file to append + CAutoFile fileout = AppendBlockFile(nFileRet); + if (!fileout) + return error("CBlock::WriteToDisk() : AppendBlockFile failed"); + if (!fWriteTransactions) + fileout.nType |= SER_BLOCKHEADERONLY; + + // Write index header + unsigned int nSize = fileout.GetSerializeSize(*this); + fileout << FLATDATA(pchMessageStart) << nSize; + + // Write block + nBlockPosRet = ftell(fileout); + if (nBlockPosRet == -1) + return error("CBlock::WriteToDisk() : ftell failed"); + fileout << *this; + + return true; + } + + bool ReadFromDisk(unsigned int nFile, unsigned int nBlockPos, bool fReadTransactions) + { + SetNull(); + + // Open history file to read + CAutoFile filein = OpenBlockFile(nFile, nBlockPos, "rb"); + if (!filein) + return error("CBlock::ReadFromDisk() : OpenBlockFile failed"); + if (!fReadTransactions) + filein.nType |= SER_BLOCKHEADERONLY; + + // Read block + filein >> *this; + + // Check the header + if (CBigNum().SetCompact(nBits) > bnProofOfWorkLimit) + return error("CBlock::ReadFromDisk() : nBits errors in block header"); + if (GetHash() > CBigNum().SetCompact(nBits).getuint256()) + return error("CBlock::ReadFromDisk() : GetHash() errors in block header"); + + return true; + } + + + + void print() const + { + printf("CBlock(hash=%s, ver=%d, hashPrevBlock=%s, hashMerkleRoot=%s, nTime=%u, nBits=%08x, nNonce=%u, vtx=%d)\n", + GetHash().ToString().substr(0,14).c_str(), + nVersion, + hashPrevBlock.ToString().substr(0,14).c_str(), + hashMerkleRoot.ToString().substr(0,6).c_str(), + nTime, nBits, nNonce, + vtx.size()); + for (int i = 0; i < vtx.size(); i++) + { + printf(" "); + vtx[i].print(); + } + printf(" vMerkleTree: "); + for (int i = 0; i < vMerkleTree.size(); i++) + printf("%s ", vMerkleTree[i].ToString().substr(0,6).c_str()); + printf("\n"); + } + + + int64 GetBlockValue(int64 nFees) const; + bool DisconnectBlock(CTxDB& txdb, CBlockIndex* pindex); + bool ConnectBlock(CTxDB& txdb, CBlockIndex* pindex); + bool ReadFromDisk(const CBlockIndex* blockindex, bool fReadTransactions); + bool AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos); + bool CheckBlock() const; + bool AcceptBlock(); +}; + + + + + + +// +// The block chain is a tree shaped structure starting with the +// genesis block at the root, with each block potentially having multiple +// candidates to be the next block. pprev and pnext link a path through the +// main/longest chain. A blockindex may have multiple pprev pointing back +// to it, but pnext will only point forward to the longest branch, or will +// be null if the block is not part of the longest chain. +// +class CBlockIndex +{ +public: + const uint256* phashBlock; + CBlockIndex* pprev; + CBlockIndex* pnext; + unsigned int nFile; + unsigned int nBlockPos; + int nHeight; + + // block header + int nVersion; + uint256 hashMerkleRoot; + unsigned int nTime; + unsigned int nBits; + unsigned int nNonce; + + + CBlockIndex() + { + phashBlock = NULL; + pprev = NULL; + pnext = NULL; + nFile = 0; + nBlockPos = 0; + nHeight = 0; + + nVersion = 0; + hashMerkleRoot = 0; + nTime = 0; + nBits = 0; + nNonce = 0; + } + + CBlockIndex(unsigned int nFileIn, unsigned int nBlockPosIn, CBlock& block) + { + phashBlock = NULL; + pprev = NULL; + pnext = NULL; + nFile = nFileIn; + nBlockPos = nBlockPosIn; + nHeight = 0; + + nVersion = block.nVersion; + hashMerkleRoot = block.hashMerkleRoot; + nTime = block.nTime; + nBits = block.nBits; + nNonce = block.nNonce; + } + + uint256 GetBlockHash() const + { + return *phashBlock; + } + + bool IsInMainChain() const + { + return (pnext || this == pindexBest); + } + + bool EraseBlockFromDisk() + { + // Open history file + CAutoFile fileout = OpenBlockFile(nFile, nBlockPos, "rb+"); + if (!fileout) + return false; + + // Overwrite with empty null block + CBlock block; + block.SetNull(); + fileout << block; + + return true; + } + + enum { nMedianTimeSpan=11 }; + + int64 GetMedianTimePast() const + { + unsigned int pmedian[nMedianTimeSpan]; + unsigned int* pbegin = &pmedian[nMedianTimeSpan]; + unsigned int* pend = &pmedian[nMedianTimeSpan]; + + const CBlockIndex* pindex = this; + for (int i = 0; i < nMedianTimeSpan && pindex; i++, pindex = pindex->pprev) + *(--pbegin) = pindex->nTime; + + sort(pbegin, pend); + return pbegin[(pend - pbegin)/2]; + } + + int64 GetMedianTime() const + { + const CBlockIndex* pindex = this; + for (int i = 0; i < nMedianTimeSpan/2; i++) + { + if (!pindex->pnext) + return nTime; + pindex = pindex->pnext; + } + return pindex->GetMedianTimePast(); + } + + + + string ToString() const + { + return strprintf("CBlockIndex(nprev=%08x, pnext=%08x, nFile=%d, nBlockPos=%-6d nHeight=%d, merkle=%s, hashBlock=%s)", + pprev, pnext, nFile, nBlockPos, nHeight, + hashMerkleRoot.ToString().substr(0,6).c_str(), + GetBlockHash().ToString().substr(0,14).c_str()); + } + + void print() const + { + printf("%s\n", ToString().c_str()); + } +}; + + + +// +// Used to marshal pointers into hashes for db storage. +// +class CDiskBlockIndex : public CBlockIndex +{ +public: + uint256 hashPrev; + uint256 hashNext; + + CDiskBlockIndex() + { + hashPrev = 0; + hashNext = 0; + } + + explicit CDiskBlockIndex(CBlockIndex* pindex) : CBlockIndex(*pindex) + { + hashPrev = (pprev ? pprev->GetBlockHash() : 0); + hashNext = (pnext ? pnext->GetBlockHash() : 0); + } + + IMPLEMENT_SERIALIZE + ( + if (!(nType & SER_GETHASH)) + READWRITE(nVersion); + + READWRITE(hashNext); + READWRITE(nFile); + READWRITE(nBlockPos); + READWRITE(nHeight); + + // block header + READWRITE(this->nVersion); + READWRITE(hashPrev); + READWRITE(hashMerkleRoot); + READWRITE(nTime); + READWRITE(nBits); + READWRITE(nNonce); + ) + + uint256 GetBlockHash() const + { + CBlock block; + block.nVersion = nVersion; + block.hashPrevBlock = hashPrev; + block.hashMerkleRoot = hashMerkleRoot; + block.nTime = nTime; + block.nBits = nBits; + block.nNonce = nNonce; + return block.GetHash(); + } + + + string ToString() const + { + string str = "CDiskBlockIndex("; + str += CBlockIndex::ToString(); + str += strprintf("\n hashBlock=%s, hashPrev=%s, hashNext=%s)", + GetBlockHash().ToString().c_str(), + hashPrev.ToString().substr(0,14).c_str(), + hashNext.ToString().substr(0,14).c_str()); + return str; + } + + void print() const + { + printf("%s\n", ToString().c_str()); + } +}; + + + + + + + + +// +// Describes a place in the block chain to another node such that if the +// other node doesn't have the same branch, it can find a recent common trunk. +// The further back it is, the further before the fork it may be. +// +class CBlockLocator +{ +protected: + vector vHave; +public: + + CBlockLocator() + { + } + + explicit CBlockLocator(const CBlockIndex* pindex) + { + Set(pindex); + } + + explicit CBlockLocator(uint256 hashBlock) + { + map::iterator mi = mapBlockIndex.find(hashBlock); + if (mi != mapBlockIndex.end()) + Set((*mi).second); + } + + IMPLEMENT_SERIALIZE + ( + if (!(nType & SER_GETHASH)) + READWRITE(nVersion); + READWRITE(vHave); + ) + + void Set(const CBlockIndex* pindex) + { + vHave.clear(); + int nStep = 1; + while (pindex) + { + vHave.push_back(pindex->GetBlockHash()); + + // Exponentially larger steps back + for (int i = 0; pindex && i < nStep; i++) + pindex = pindex->pprev; + if (vHave.size() > 10) + nStep *= 2; + } + vHave.push_back(hashGenesisBlock); + } + + CBlockIndex* GetBlockIndex() + { + // Find the first block the caller has in the main chain + foreach(const uint256& hash, vHave) + { + map::iterator mi = mapBlockIndex.find(hash); + if (mi != mapBlockIndex.end()) + { + CBlockIndex* pindex = (*mi).second; + if (pindex->IsInMainChain()) + return pindex; + } + } + return pindexGenesisBlock; + } + + uint256 GetBlockHash() + { + // Find the first block the caller has in the main chain + foreach(const uint256& hash, vHave) + { + map::iterator mi = mapBlockIndex.find(hash); + if (mi != mapBlockIndex.end()) + { + CBlockIndex* pindex = (*mi).second; + if (pindex->IsInMainChain()) + return hash; + } + } + return hashGenesisBlock; + } + + int GetHeight() + { + CBlockIndex* pindex = GetBlockIndex(); + if (!pindex) + return 0; + return pindex->nHeight; + } +}; + + + + + + + + + + + + +extern map mapTransactions; +extern map mapWallet; +extern vector > vWalletUpdated; +extern CCriticalSection cs_mapWallet; +extern map, CPrivKey> mapKeys; +extern map > mapPubKeys; +extern CCriticalSection cs_mapKeys; +extern CKey keyUser; diff --git a/makefile b/makefile new file mode 100644 index 00000000..534eb521 --- /dev/null +++ b/makefile @@ -0,0 +1,83 @@ +# Copyright (c) 2009 Satoshi Nakamoto +# Distributed under the MIT/X11 software license, see the accompanying +# file license.txt or http://www.opensource.org/licenses/mit-license.php. + + +ifneq "$(BUILD)" "debug" +ifneq "$(BUILD)" "release" +BUILD=debug +endif +endif +ifeq "$(BUILD)" "debug" +D=d +# note: gcc 3.x profile doesn't work +#DEBUGFLAGS=-O0 -g -pg -D__WXDEBUG__ +DEBUGFLAGS=-g -D__WXDEBUG__ +endif + + + +INCLUDEPATHS=-I"/boost" -I"/DB/build_unix" -I"/OpenSSL/include" -I"/wxWidgets/lib/vc_lib/mswd" -I"/wxWidgets/include" +LIBPATHS=-L"/DB/build_unix" -L"/OpenSSL/out" -L"/wxWidgets/lib/gcc_lib" +LIBS= \ + -l db_cxx \ + -l eay32 \ + -l wxmsw28$(D)_richtext -l wxmsw28$(D)_html -l wxmsw28$(D)_core -l wxbase28$(D) -l wxtiff$(D) -l wxjpeg$(D) -l wxpng$(D) -l wxzlib$(D) -l wxregex$(D) -l wxexpat$(D) \ + -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 +WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH +CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h + + + +all: bitcoin.exe + + +headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h + g++ -c $(CFLAGS) -o $@ $< + +obj/util.o: util.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/script.o: script.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/db.o: db.cpp $(HEADERS) market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/net.o: net.cpp $(HEADERS) net.h + g++ -c $(CFLAGS) -o $@ $< + +obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h + g++ -c $(CFLAGS) -o $@ $< + +obj/market.o: market.cpp $(HEADERS) market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/uibase.o: uibase.cpp uibase.h + g++ -c $(CFLAGS) -o $@ $< + +obj/sha.o: sha.cpp sha.h + g++ -c $(CFLAGS) -O3 -o $@ $< + +obj/irc.o: irc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp + windres $(WXDEFS) $(INCLUDEPATHS) -o $@ -i $< + + + +OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o + +bitcoin.exe: headers.h.gch $(OBJS) + -kill /f bitcoin.exe + g++ $(CFLAGS) -mwindows -Wl,--subsystem,windows -o $@ $(LIBPATHS) $(OBJS) $(LIBS) + +clean: + -del /Q obj\* + -del /Q headers.h.gch diff --git a/makefile.vc b/makefile.vc new file mode 100644 index 00000000..bf7e9bc5 --- /dev/null +++ b/makefile.vc @@ -0,0 +1,77 @@ +# Copyright (c) 2009 Satoshi Nakamoto +# Distributed under the MIT/X11 software license, see the accompanying +# file license.txt or http://www.opensource.org/licenses/mit-license.php. + + +!IF "$(BUILD)" != "debug" && "$(BUILD)" != "release" +BUILD=debug +!ENDIF +!IF "$(BUILD)" == "debug" +D=d +DEBUGFLAGS=/Zi /Od /D__WXDEBUG__ +!ENDIF + + + +INCLUDEPATHS=/I"/boost" /I"/DB/build_windows" /I"/OpenSSL/include" /I"/wxWidgets/lib/vc_lib/mswd" /I"/wxWidgets/include" +LIBPATHS=/LIBPATH:"/DB/build_windows/$(BUILD)" /LIBPATH:"/OpenSSL/out" /LIBPATH:"/wxWidgets/lib/vc_lib" +LIBS= \ + libdb47s$(D).lib \ + libeay32.lib \ + wxmsw28$(D)_richtext.lib wxmsw28$(D)_html.lib wxmsw28$(D)_core.lib wxbase28$(D).lib wxtiff$(D).lib wxjpeg$(D).lib wxpng$(D).lib wxzlib$(D).lib wxregex$(D).lib wxexpat$(D).lib \ + kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib +WXDEFS=/DWIN32 /D__WXMSW__ /D_WINDOWS /DNOPCH +CFLAGS=/c /nologo /Ob0 /MD$(D) /EHsc /GR /Zm300 /YX /Fpobj/headers.pch $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h + + + +all: bitcoin.exe + + +obj\util.obj: util.cpp $(HEADERS) + cl $(CFLAGS) /Fo$@ %s + +obj\script.obj: script.cpp $(HEADERS) + cl $(CFLAGS) /Fo$@ %s + +obj\db.obj: db.cpp $(HEADERS) market.h + cl $(CFLAGS) /Fo$@ %s + +obj\net.obj: net.cpp $(HEADERS) net.h + cl $(CFLAGS) /Fo$@ %s + +obj\main.obj: main.cpp $(HEADERS) net.h market.h + cl $(CFLAGS) /Fo$@ %s + +obj\market.obj: market.cpp $(HEADERS) market.h + cl $(CFLAGS) /Fo$@ %s + +obj\ui.obj: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h + cl $(CFLAGS) /Fo$@ %s + +obj\uibase.obj: uibase.cpp uibase.h + cl $(CFLAGS) /Fo$@ %s + +obj\sha.obj: sha.cpp sha.h + cl $(CFLAGS) /O2 /Fo$@ %s + +obj\irc.obj: irc.cpp $(HEADERS) + cl $(CFLAGS) /Fo$@ %s + +obj\ui.res: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp + rc $(INCLUDEPATHS) $(WXDEFS) /Fo$@ %s + + + +OBJS=obj\util.obj obj\script.obj obj\db.obj obj\net.obj obj\main.obj obj\market.obj \ + obj\ui.obj obj\uibase.obj obj\sha.obj obj\irc.obj obj\ui.res + +bitcoin.exe: $(OBJS) + -kill /f bitcoin.exe & sleep 1 + link /nologo /DEBUG /SUBSYSTEM:WINDOWS /OUT:$@ $(LIBPATHS) $** $(LIBS) + +clean: + -del /Q obj\* + -del *.ilk + -del *.pdb diff --git a/market.cpp b/market.cpp new file mode 100644 index 00000000..22b5365f --- /dev/null +++ b/market.cpp @@ -0,0 +1,264 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" + + + + + + + + + + +// +// Global state variables +// + +//// later figure out how these are persisted +map mapMyProducts; + + + + +map mapProducts; +CCriticalSection cs_mapProducts; + +bool AdvertInsert(const CProduct& product) +{ + uint256 hash = product.GetHash(); + bool fNew = false; + bool fUpdated = false; + + CRITICAL_BLOCK(cs_mapProducts) + { + // Insert or find existing product + pair::iterator, bool> item = mapProducts.insert(make_pair(hash, product)); + CProduct* pproduct = &(*(item.first)).second; + fNew = item.second; + + // Update if newer + if (product.nSequence > pproduct->nSequence) + { + *pproduct = product; + fUpdated = true; + } + } + + //if (fNew) + // NotifyProductAdded(hash); + //else if (fUpdated) + // NotifyProductUpdated(hash); + + return (fNew || fUpdated); +} + +void AdvertErase(const CProduct& product) +{ + uint256 hash = product.GetHash(); + CRITICAL_BLOCK(cs_mapProducts) + mapProducts.erase(hash); + //NotifyProductDeleted(hash); +} + + + + + + + + + + + + + + + + + + + +template +unsigned int Union(T& v1, T& v2) +{ + // v1 = v1 union v2 + // v1 and v2 must be sorted + // returns the number of elements added to v1 + + ///// need to check that this is equivalent, then delete this comment + //vector vUnion(v1.size() + v2.size()); + //vUnion.erase(set_union(v1.begin(), v1.end(), + // v2.begin(), v2.end(), + // vUnion.begin()), + // vUnion.end()); + + T vUnion; + vUnion.reserve(v1.size() + v2.size()); + set_union(v1.begin(), v1.end(), + v2.begin(), v2.end(), + back_inserter(vUnion)); + unsigned int nAdded = vUnion.size() - v1.size(); + if (nAdded > 0) + v1 = vUnion; + return nAdded; +} + +void CUser::AddAtom(unsigned short nAtom, bool fOrigin) +{ + // Ignore duplicates + if (binary_search(vAtomsIn.begin(), vAtomsIn.end(), nAtom) || + find(vAtomsNew.begin(), vAtomsNew.end(), nAtom) != vAtomsNew.end()) + return; + + //// instead of zero atom, should change to free atom that propagates, + //// limited to lower than a certain value like 5 so conflicts quickly + // The zero atom never propagates, + // new atoms always propagate through the user that created them + if (nAtom == 0 || fOrigin) + { + vector vTmp(1, nAtom); + Union(vAtomsIn, vTmp); + if (fOrigin) + vAtomsOut.push_back(nAtom); + return; + } + + vAtomsNew.push_back(nAtom); + + if (vAtomsNew.size() >= nFlowthroughRate || vAtomsOut.empty()) + { + // Select atom to flow through to vAtomsOut + vAtomsOut.push_back(vAtomsNew[GetRand(vAtomsNew.size())]); + + // Merge vAtomsNew into vAtomsIn + sort(vAtomsNew.begin(), vAtomsNew.end()); + Union(vAtomsIn, vAtomsNew); + vAtomsNew.clear(); + } +} + +bool AddAtomsAndPropagate(uint256 hashUserStart, const vector& vAtoms, bool fOrigin) +{ + CReviewDB reviewdb; + map > pmapPropagate[2]; + pmapPropagate[0][hashUserStart] = vAtoms; + + for (int side = 0; !pmapPropagate[side].empty(); side = 1 - side) + { + map >& mapFrom = pmapPropagate[side]; + map >& mapTo = pmapPropagate[1 - side]; + + for (map >::iterator mi = mapFrom.begin(); mi != mapFrom.end(); ++mi) + { + const uint256& hashUser = (*mi).first; + const vector& vReceived = (*mi).second; + + ///// this would be a lot easier on the database if it put the new atom at the beginning of the list, + ///// so the change would be right next to the vector size. + + // Read user + CUser user; + reviewdb.ReadUser(hashUser, user); + unsigned int nIn = user.vAtomsIn.size(); + unsigned int nNew = user.vAtomsNew.size(); + unsigned int nOut = user.vAtomsOut.size(); + + // Add atoms received + foreach(unsigned short nAtom, vReceived) + user.AddAtom(nAtom, fOrigin); + fOrigin = false; + + // Don't bother writing to disk if no changes + if (user.vAtomsIn.size() == nIn && user.vAtomsNew.size() == nNew) + continue; + + // Propagate + if (user.vAtomsOut.size() > nOut) + foreach(const uint256& hash, user.vLinksOut) + mapTo[hash].insert(mapTo[hash].end(), user.vAtomsOut.begin() + nOut, user.vAtomsOut.end()); + + // Write back + if (!reviewdb.WriteUser(hashUser, user)) + return false; + } + mapFrom.clear(); + } + return true; +} + + + + + + +bool CReview::AcceptReview() +{ + // Timestamp + nTime = GetTime(); + + // Check signature + if (!CKey::Verify(vchPubKeyFrom, GetSigHash(), vchSig)) + return false; + + CReviewDB reviewdb; + + // Add review text to recipient + vector vReviews; + reviewdb.ReadReviews(hashTo, vReviews); + vReviews.push_back(*this); + if (!reviewdb.WriteReviews(hashTo, vReviews)) + return false; + + // Add link from sender + CUser user; + uint256 hashFrom = Hash(vchPubKeyFrom.begin(), vchPubKeyFrom.end()); + reviewdb.ReadUser(hashFrom, user); + user.vLinksOut.push_back(hashTo); + if (!reviewdb.WriteUser(hashFrom, user)) + return false; + + reviewdb.Close(); + + // Propagate atoms to recipient + vector vZeroAtom(1, 0); + if (!AddAtomsAndPropagate(hashTo, user.vAtomsOut.size() ? user.vAtomsOut : vZeroAtom, false)) + return false; + + return true; +} + + + + + +bool CProduct::CheckSignature() +{ + return (CKey::Verify(vchPubKeyFrom, GetSigHash(), vchSig)); +} + +bool CProduct::CheckProduct() +{ + if (!CheckSignature()) + return false; + + // Make sure it's a summary product + if (!mapDetails.empty() || !vOrderForm.empty()) + return false; + + // Look up seller's atom count + CReviewDB reviewdb("r"); + CUser user; + reviewdb.ReadUser(GetUserHash(), user); + nAtoms = user.GetAtomCount(); + reviewdb.Close(); + + ////// delme, this is now done by AdvertInsert + //// Store to memory + //CRITICAL_BLOCK(cs_mapProducts) + // mapProducts[GetHash()] = *this; + + return true; +} diff --git a/market.h b/market.h new file mode 100644 index 00000000..27147873 --- /dev/null +++ b/market.h @@ -0,0 +1,182 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +class CUser; +class CReview; +class CProduct; + +static const unsigned int nFlowthroughRate = 2; + + + + +bool AdvertInsert(const CProduct& product); +void AdvertErase(const CProduct& product); +bool AddAtomsAndPropagate(uint256 hashUserStart, const vector& vAtoms, bool fOrigin); + + + + + + + + +class CUser +{ +public: + vector vAtomsIn; + vector vAtomsNew; + vector vAtomsOut; + vector vLinksOut; + + CUser() + { + } + + IMPLEMENT_SERIALIZE + ( + if (!(nType & SER_GETHASH)) + READWRITE(nVersion); + READWRITE(vAtomsIn); + READWRITE(vAtomsNew); + READWRITE(vAtomsOut); + READWRITE(vLinksOut); + ) + + void SetNull() + { + vAtomsIn.clear(); + vAtomsNew.clear(); + vAtomsOut.clear(); + vLinksOut.clear(); + } + + uint256 GetHash() const { return SerializeHash(*this); } + + + int GetAtomCount() const + { + return (vAtomsIn.size() + vAtomsNew.size()); + } + + void AddAtom(unsigned short nAtom, bool fOrigin); +}; + + + + + + + +class CReview +{ +public: + int nVersion; + uint256 hashTo; + map mapValue; + vector vchPubKeyFrom; + vector vchSig; + + // memory only + unsigned int nTime; + int nAtoms; + + + CReview() + { + nVersion = 1; + hashTo = 0; + nTime = 0; + nAtoms = 0; + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(this->nVersion); + nVersion = this->nVersion; + if (!(nType & SER_DISK)) + READWRITE(hashTo); + READWRITE(mapValue); + READWRITE(vchPubKeyFrom); + if (!(nType & SER_GETHASH)) + READWRITE(vchSig); + ) + + uint256 GetHash() const { return SerializeHash(*this); } + uint256 GetSigHash() const { return SerializeHash(*this, SER_GETHASH|SER_SKIPSIG); } + uint256 GetUserHash() const { return Hash(vchPubKeyFrom.begin(), vchPubKeyFrom.end()); } + + + bool AcceptReview(); +}; + + + + + + + +class CProduct +{ +public: + int nVersion; + CAddress addr; + map mapValue; + map mapDetails; + vector > vOrderForm; + unsigned int nSequence; + vector vchPubKeyFrom; + vector vchSig; + + // disk only + int nAtoms; + + // memory only + set setSources; + + CProduct() + { + nVersion = 1; + nAtoms = 0; + nSequence = 0; + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(this->nVersion); + nVersion = this->nVersion; + READWRITE(addr); + READWRITE(mapValue); + if (!(nType & SER_GETHASH)) + { + READWRITE(mapDetails); + READWRITE(vOrderForm); + READWRITE(nSequence); + } + READWRITE(vchPubKeyFrom); + if (!(nType & SER_GETHASH)) + READWRITE(vchSig); + if (nType & SER_DISK) + READWRITE(nAtoms); + ) + + uint256 GetHash() const { return SerializeHash(*this); } + uint256 GetSigHash() const { return SerializeHash(*this, SER_GETHASH|SER_SKIPSIG); } + uint256 GetUserHash() const { return Hash(vchPubKeyFrom.begin(), vchPubKeyFrom.end()); } + + + bool CheckSignature(); + bool CheckProduct(); +}; + + + + + + + + +extern map mapProducts; +extern CCriticalSection cs_mapProducts; +extern map mapMyProducts; diff --git a/mingwm10.dll b/mingwm10.dll new file mode 100644 index 0000000000000000000000000000000000000000..bf8544ce13ad82018a4aa13abc1594297e4d891b GIT binary patch literal 11673 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 x4(UFf=#(JFDkOW?G&iQ|nyfl8)l*8X1Z) + +void ThreadMessageHandler2(void* parg); +void ThreadSocketHandler2(void* parg); +void ThreadOpenConnections2(void* parg); + + + + + + +// +// Global state variables +// +bool fClient = false; +uint64 nLocalServices = (fClient ? 0 : NODE_NETWORK); +CAddress addrLocalHost(0, DEFAULT_PORT, nLocalServices); +CNode nodeLocalHost(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); +CNode* pnodeLocalHost = &nodeLocalHost; +bool fShutdown = false; +array vfThreadRunning; +vector vNodes; +CCriticalSection cs_vNodes; +map, CAddress> mapAddresses; +CCriticalSection cs_mapAddresses; +map mapRelay; +deque > vRelayExpiration; +CCriticalSection cs_mapRelay; +map mapAlreadyAskedFor; + + + +CAddress addrProxy; + +bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) +{ + hSocketRet = INVALID_SOCKET; + + SOCKET hSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); + if (hSocket == INVALID_SOCKET) + return false; + + bool fRoutable = !(addrConnect.GetByte(3) == 10 || (addrConnect.GetByte(3) == 192 && addrConnect.GetByte(2) == 168)); + bool fProxy = (addrProxy.ip && fRoutable); + struct sockaddr_in sockaddr = (fProxy ? addrProxy.GetSockAddr() : addrConnect.GetSockAddr()); + + if (connect(hSocket, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) == SOCKET_ERROR) + { + closesocket(hSocket); + return false; + } + + if (fProxy) + { + printf("Proxy connecting to %s\n", addrConnect.ToString().c_str()); + char pszSocks4IP[] = "\4\1\0\0\0\0\0\0user"; + memcpy(pszSocks4IP + 2, &addrConnect.port, 2); + memcpy(pszSocks4IP + 4, &addrConnect.ip, 4); + char* pszSocks4 = pszSocks4IP; + int nSize = sizeof(pszSocks4IP); + + int ret = send(hSocket, pszSocks4, nSize, 0); + if (ret != nSize) + { + closesocket(hSocket); + return error("Error sending to proxy\n"); + } + char pchRet[8]; + if (recv(hSocket, pchRet, 8, 0) != 8) + { + closesocket(hSocket); + return error("Error reading proxy response\n"); + } + if (pchRet[1] != 0x5a) + { + closesocket(hSocket); + return error("Proxy returned error %d\n", pchRet[1]); + } + printf("Proxy connection established %s\n", addrConnect.ToString().c_str()); + } + + hSocketRet = hSocket; + return true; +} + + + +bool GetMyExternalIP2(const CAddress& addrConnect, const char* pszGet, const char* pszKeyword, unsigned int& ipRet) +{ + SOCKET hSocket; + if (!ConnectSocket(addrConnect, hSocket)) + return error("GetMyExternalIP() : connection to %s failed\n", addrConnect.ToString().c_str()); + + send(hSocket, pszGet, strlen(pszGet), 0); + + string strLine; + while (RecvLine(hSocket, strLine)) + { + if (strLine.empty()) + { + loop + { + if (!RecvLine(hSocket, strLine)) + { + closesocket(hSocket); + return false; + } + if (strLine.find(pszKeyword) != -1) + { + strLine = strLine.substr(strLine.find(pszKeyword) + strlen(pszKeyword)); + break; + } + } + closesocket(hSocket); + if (strLine.find("<")) + strLine = strLine.substr(0, strLine.find("<")); + strLine = strLine.substr(strspn(strLine.c_str(), " \t\n\r")); + strLine = wxString(strLine).Trim(); + CAddress addr(strLine.c_str()); + printf("GetMyExternalIP() received [%s] %s\n", strLine.c_str(), addr.ToString().c_str()); + if (addr.ip == 0 || !addr.IsRoutable()) + return false; + ipRet = addr.ip; + return true; + } + } + closesocket(hSocket); + return error("GetMyExternalIP() : connection closed\n"); +} + + +bool GetMyExternalIP(unsigned int& ipRet) +{ + CAddress addrConnect; + char* pszGet; + char* pszKeyword; + + for (int nLookup = 0; nLookup <= 1; nLookup++) + for (int nHost = 1; nHost <= 2; nHost++) + { + if (nHost == 1) + { + addrConnect = CAddress("70.86.96.218:80"); // www.ipaddressworld.com + + if (nLookup == 1) + { + struct hostent* phostent = gethostbyname("www.ipaddressworld.com"); + if (phostent && phostent->h_addr_list && phostent->h_addr_list[0]) + addrConnect = CAddress(*(u_long*)phostent->h_addr_list[0], htons(80)); + } + + pszGet = "GET /ip.php HTTP/1.1\r\n" + "Host: www.ipaddressworld.com\r\n" + "User-Agent: Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.1)\r\n" + "Connection: close\r\n" + "\r\n"; + + pszKeyword = "IP:"; + } + else if (nHost == 2) + { + addrConnect = CAddress("208.78.68.70:80"); // checkip.dyndns.org + + if (nLookup == 1) + { + struct hostent* phostent = gethostbyname("checkip.dyndns.org"); + if (phostent && phostent->h_addr_list && phostent->h_addr_list[0]) + addrConnect = CAddress(*(u_long*)phostent->h_addr_list[0], htons(80)); + } + + pszGet = "GET / HTTP/1.1\r\n" + "Host: checkip.dyndns.org\r\n" + "User-Agent: Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.1)\r\n" + "Connection: close\r\n" + "\r\n"; + + pszKeyword = "Address:"; + } + + if (GetMyExternalIP2(addrConnect, pszGet, pszKeyword, ipRet)) + return true; + } + + return false; +} + + + + + +bool AddAddress(CAddrDB& addrdb, const CAddress& addr) +{ + if (!addr.IsRoutable()) + return false; + if (addr.ip == addrLocalHost.ip) + return false; + CRITICAL_BLOCK(cs_mapAddresses) + { + map, CAddress>::iterator it = mapAddresses.find(addr.GetKey()); + if (it == mapAddresses.end()) + { + // New address + mapAddresses.insert(make_pair(addr.GetKey(), addr)); + addrdb.WriteAddress(addr); + return true; + } + else + { + CAddress& addrFound = (*it).second; + if ((addrFound.nServices | addr.nServices) != addrFound.nServices) + { + // Services have been added + addrFound.nServices |= addr.nServices; + addrdb.WriteAddress(addrFound); + return true; + } + } + } + return false; +} + + + + + +void AbandonRequests(void (*fn)(void*, CDataStream&), void* param1) +{ + // If the dialog might get closed before the reply comes back, + // call this in the destructor so it doesn't get called after it's deleted. + CRITICAL_BLOCK(cs_vNodes) + { + foreach(CNode* pnode, vNodes) + { + CRITICAL_BLOCK(pnode->cs_mapRequests) + { + for (map::iterator mi = pnode->mapRequests.begin(); mi != pnode->mapRequests.end();) + { + CRequestTracker& tracker = (*mi).second; + if (tracker.fn == fn && tracker.param1 == param1) + pnode->mapRequests.erase(mi++); + else + mi++; + } + } + } + } +} + + + + + + + +// +// Subscription methods for the broadcast and subscription system. +// Channel numbers are message numbers, i.e. MSG_TABLE and MSG_PRODUCT. +// +// The subscription system uses a meet-in-the-middle strategy. +// With 100,000 nodes, if senders broadcast to 1000 random nodes and receivers +// subscribe to 1000 random nodes, 99.995% (1 - 0.99^1000) of messages will get through. +// + +bool AnySubscribed(unsigned int nChannel) +{ + if (pnodeLocalHost->IsSubscribed(nChannel)) + return true; + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + if (pnode->IsSubscribed(nChannel)) + return true; + return false; +} + +bool CNode::IsSubscribed(unsigned int nChannel) +{ + if (nChannel >= vfSubscribe.size()) + return false; + return vfSubscribe[nChannel]; +} + +void CNode::Subscribe(unsigned int nChannel, unsigned int nHops) +{ + if (nChannel >= vfSubscribe.size()) + return; + + if (!AnySubscribed(nChannel)) + { + // Relay subscribe + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + if (pnode != this) + pnode->PushMessage("subscribe", nChannel, nHops); + } + + vfSubscribe[nChannel] = true; +} + +void CNode::CancelSubscribe(unsigned int nChannel) +{ + if (nChannel >= vfSubscribe.size()) + return; + + // Prevent from relaying cancel if wasn't subscribed + if (!vfSubscribe[nChannel]) + return; + vfSubscribe[nChannel] = false; + + if (!AnySubscribed(nChannel)) + { + // Relay subscription cancel + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + if (pnode != this) + pnode->PushMessage("sub-cancel", nChannel); + + // Clear memory, no longer subscribed + if (nChannel == MSG_PRODUCT) + CRITICAL_BLOCK(cs_mapProducts) + mapProducts.clear(); + } +} + + + + + + + + + +CNode* FindNode(unsigned int ip) +{ + CRITICAL_BLOCK(cs_vNodes) + { + foreach(CNode* pnode, vNodes) + if (pnode->addr.ip == ip) + return (pnode); + } + return NULL; +} + +CNode* FindNode(CAddress addr) +{ + CRITICAL_BLOCK(cs_vNodes) + { + foreach(CNode* pnode, vNodes) + if (pnode->addr == addr) + return (pnode); + } + return NULL; +} + +CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) +{ + if (addrConnect.ip == addrLocalHost.ip) + return NULL; + + // Look for an existing connection + CNode* pnode = FindNode(addrConnect.ip); + if (pnode) + { + if (nTimeout != 0) + pnode->AddRef(nTimeout); + else + pnode->AddRef(); + return pnode; + } + + /// debug print + printf("trying %s\n", addrConnect.ToString().c_str()); + + // Connect + SOCKET hSocket; + if (ConnectSocket(addrConnect, hSocket)) + { + /// debug print + printf("connected %s\n", addrConnect.ToString().c_str()); + + // Set to nonblocking + u_long nOne = 1; + if (ioctlsocket(hSocket, FIONBIO, &nOne) == SOCKET_ERROR) + printf("ConnectSocket() : ioctlsocket nonblocking setting failed, error %d\n", WSAGetLastError()); + + // Add node + CNode* pnode = new CNode(hSocket, addrConnect, false); + if (nTimeout != 0) + pnode->AddRef(nTimeout); + else + pnode->AddRef(); + CRITICAL_BLOCK(cs_vNodes) + vNodes.push_back(pnode); + + CRITICAL_BLOCK(cs_mapAddresses) + mapAddresses[addrConnect.GetKey()].nLastFailed = 0; + return pnode; + } + else + { + CRITICAL_BLOCK(cs_mapAddresses) + mapAddresses[addrConnect.GetKey()].nLastFailed = GetTime(); + return NULL; + } +} + +void CNode::Disconnect() +{ + printf("disconnecting node %s\n", addr.ToString().c_str()); + + closesocket(hSocket); + + // If outbound and never got version message, mark address as failed + if (!fInbound && nVersion == 0) + CRITICAL_BLOCK(cs_mapAddresses) + mapAddresses[addr.GetKey()].nLastFailed = GetTime(); + + // All of a nodes broadcasts and subscriptions are automatically torn down + // when it goes down, so a node has to stay up to keep its broadcast going. + + CRITICAL_BLOCK(cs_mapProducts) + for (map::iterator mi = mapProducts.begin(); mi != mapProducts.end();) + AdvertRemoveSource(this, MSG_PRODUCT, 0, (*(mi++)).second); + + // Cancel subscriptions + for (unsigned int nChannel = 0; nChannel < vfSubscribe.size(); nChannel++) + if (vfSubscribe[nChannel]) + CancelSubscribe(nChannel); +} + + + + + + + + + + + + + +void ThreadSocketHandler(void* parg) +{ + IMPLEMENT_RANDOMIZE_STACK(ThreadSocketHandler(parg)); + + loop + { + vfThreadRunning[0] = true; + CheckForShutdown(0); + try + { + ThreadSocketHandler2(parg); + } + CATCH_PRINT_EXCEPTION("ThreadSocketHandler()") + vfThreadRunning[0] = false; + Sleep(5000); + } +} + +void ThreadSocketHandler2(void* parg) +{ + printf("ThreadSocketHandler started\n"); + SOCKET hListenSocket = *(SOCKET*)parg; + list vNodesDisconnected; + int nPrevNodeCount = 0; + + loop + { + // + // Disconnect nodes + // + CRITICAL_BLOCK(cs_vNodes) + { + // Disconnect unused nodes + vector vNodesCopy = vNodes; + foreach(CNode* pnode, vNodesCopy) + { + if (pnode->ReadyToDisconnect() && pnode->vRecv.empty() && pnode->vSend.empty()) + { + // remove from vNodes + vNodes.erase(remove(vNodes.begin(), vNodes.end(), pnode), vNodes.end()); + pnode->Disconnect(); + + // hold in disconnected pool until all refs are released + pnode->nReleaseTime = max(pnode->nReleaseTime, GetTime() + 5 * 60); + if (pnode->fNetworkNode) + pnode->Release(); + vNodesDisconnected.push_back(pnode); + } + } + + // Delete disconnected nodes + list vNodesDisconnectedCopy = vNodesDisconnected; + foreach(CNode* pnode, vNodesDisconnectedCopy) + { + // wait until threads are done using it + if (pnode->GetRefCount() <= 0) + { + bool fDelete = false; + TRY_CRITICAL_BLOCK(pnode->cs_vSend) + TRY_CRITICAL_BLOCK(pnode->cs_vRecv) + TRY_CRITICAL_BLOCK(pnode->cs_mapRequests) + TRY_CRITICAL_BLOCK(pnode->cs_inventory) + fDelete = true; + if (fDelete) + { + vNodesDisconnected.remove(pnode); + delete pnode; + } + } + } + } + if (vNodes.size() != nPrevNodeCount) + { + nPrevNodeCount = vNodes.size(); + MainFrameRepaint(); + } + + + // + // Find which sockets have data to receive + // + struct timeval timeout; + timeout.tv_sec = 0; + timeout.tv_usec = 50000; // frequency to poll pnode->vSend + + struct fd_set fdsetRecv; + struct fd_set fdsetSend; + FD_ZERO(&fdsetRecv); + FD_ZERO(&fdsetSend); + SOCKET hSocketMax = 0; + FD_SET(hListenSocket, &fdsetRecv); + hSocketMax = max(hSocketMax, hListenSocket); + CRITICAL_BLOCK(cs_vNodes) + { + foreach(CNode* pnode, vNodes) + { + FD_SET(pnode->hSocket, &fdsetRecv); + hSocketMax = max(hSocketMax, pnode->hSocket); + TRY_CRITICAL_BLOCK(pnode->cs_vSend) + if (!pnode->vSend.empty()) + FD_SET(pnode->hSocket, &fdsetSend); + } + } + + vfThreadRunning[0] = false; + int nSelect = select(hSocketMax + 1, &fdsetRecv, &fdsetSend, NULL, &timeout); + vfThreadRunning[0] = true; + CheckForShutdown(0); + if (nSelect == SOCKET_ERROR) + { + int nErr = WSAGetLastError(); + printf("select failed: %d\n", nErr); + for (int i = 0; i <= hSocketMax; i++) + { + FD_SET(i, &fdsetRecv); + FD_SET(i, &fdsetSend); + } + Sleep(timeout.tv_usec/1000); + } + RandAddSeed(); + + //// debug print + //foreach(CNode* pnode, vNodes) + //{ + // printf("vRecv = %-5d ", pnode->vRecv.size()); + // printf("vSend = %-5d ", pnode->vSend.size()); + //} + //printf("\n"); + + + // + // Accept new connections + // + if (FD_ISSET(hListenSocket, &fdsetRecv)) + { + struct sockaddr_in sockaddr; + int len = sizeof(sockaddr); + SOCKET hSocket = accept(hListenSocket, (struct sockaddr*)&sockaddr, &len); + CAddress addr(sockaddr); + if (hSocket == INVALID_SOCKET) + { + if (WSAGetLastError() != WSAEWOULDBLOCK) + printf("ERROR ThreadSocketHandler accept failed: %d\n", WSAGetLastError()); + } + else + { + printf("accepted connection from %s\n", addr.ToString().c_str()); + CNode* pnode = new CNode(hSocket, addr, true); + pnode->AddRef(); + CRITICAL_BLOCK(cs_vNodes) + vNodes.push_back(pnode); + } + } + + + // + // Service each socket + // + vector vNodesCopy; + CRITICAL_BLOCK(cs_vNodes) + vNodesCopy = vNodes; + foreach(CNode* pnode, vNodesCopy) + { + CheckForShutdown(0); + SOCKET hSocket = pnode->hSocket; + + // + // Receive + // + if (FD_ISSET(hSocket, &fdsetRecv)) + { + TRY_CRITICAL_BLOCK(pnode->cs_vRecv) + { + CDataStream& vRecv = pnode->vRecv; + unsigned int nPos = vRecv.size(); + + // typical socket buffer is 8K-64K + const unsigned int nBufSize = 0x10000; + vRecv.resize(nPos + nBufSize); + int nBytes = recv(hSocket, &vRecv[nPos], nBufSize, 0); + vRecv.resize(nPos + max(nBytes, 0)); + if (nBytes == 0) + { + // socket closed gracefully + if (!pnode->fDisconnect) + printf("recv: socket closed\n"); + pnode->fDisconnect = true; + } + else if (nBytes < 0) + { + // socket error + int nErr = WSAGetLastError(); + if (nErr != WSAEWOULDBLOCK && nErr != WSAEMSGSIZE && nErr != WSAEINTR && nErr != WSAEINPROGRESS) + { + if (!pnode->fDisconnect) + printf("recv failed: %d\n", nErr); + pnode->fDisconnect = true; + } + } + } + } + + // + // Send + // + if (FD_ISSET(hSocket, &fdsetSend)) + { + TRY_CRITICAL_BLOCK(pnode->cs_vSend) + { + CDataStream& vSend = pnode->vSend; + if (!vSend.empty()) + { + int nBytes = send(hSocket, &vSend[0], vSend.size(), 0); + if (nBytes > 0) + { + vSend.erase(vSend.begin(), vSend.begin() + nBytes); + } + else if (nBytes == 0) + { + if (pnode->ReadyToDisconnect()) + pnode->vSend.clear(); + } + else + { + printf("send error %d\n", nBytes); + if (pnode->ReadyToDisconnect()) + pnode->vSend.clear(); + } + } + } + } + } + + + Sleep(10); + } +} + + + + + + + + + + +void ThreadOpenConnections(void* parg) +{ + IMPLEMENT_RANDOMIZE_STACK(ThreadOpenConnections(parg)); + + loop + { + vfThreadRunning[1] = true; + CheckForShutdown(1); + try + { + ThreadOpenConnections2(parg); + } + CATCH_PRINT_EXCEPTION("ThreadOpenConnections()") + vfThreadRunning[1] = false; + Sleep(5000); + } +} + +void ThreadOpenConnections2(void* parg) +{ + printf("ThreadOpenConnections started\n"); + + // Initiate network connections + int nTry = 0; + bool fIRCOnly = false; + const int nMaxConnections = 15; + loop + { + // Wait + vfThreadRunning[1] = false; + Sleep(500); + while (vNodes.size() >= nMaxConnections || vNodes.size() >= mapAddresses.size()) + { + CheckForShutdown(1); + Sleep(2000); + } + vfThreadRunning[1] = true; + CheckForShutdown(1); + + + // + // The IP selection process is designed to limit vulnerability to address flooding. + // Any class C (a.b.c.?) has an equal chance of being chosen, then an IP is + // chosen within the class C. An attacker may be able to allocate many IPs, but + // they would normally be concentrated in blocks of class C's. They can hog the + // attention within their class C, but not the whole IP address space overall. + // A lone node in a class C will get as much attention as someone holding all 255 + // IPs in another class C. + // + + // Every other try is with IRC addresses only + fIRCOnly = !fIRCOnly; + if (mapIRCAddresses.empty()) + fIRCOnly = false; + else if (nTry++ < 30 && vNodes.size() < nMaxConnections/2) + fIRCOnly = true; + + // Make a list of unique class C's + unsigned char pchIPCMask[4] = { 0xff, 0xff, 0xff, 0x00 }; + unsigned int nIPCMask = *(unsigned int*)pchIPCMask; + vector vIPC; + CRITICAL_BLOCK(cs_mapIRCAddresses) + CRITICAL_BLOCK(cs_mapAddresses) + { + vIPC.reserve(mapAddresses.size()); + unsigned int nPrev = 0; + foreach(const PAIRTYPE(vector, CAddress)& item, mapAddresses) + { + const CAddress& addr = item.second; + if (!addr.IsIPv4()) + continue; + if (fIRCOnly && !mapIRCAddresses.count(item.first)) + continue; + + // Taking advantage of mapAddresses being in sorted order, + // with IPs of the same class C grouped together. + unsigned int ipC = addr.ip & nIPCMask; + if (ipC != nPrev) + vIPC.push_back(nPrev = ipC); + } + } + if (vIPC.empty()) + continue; + + // Choose a random class C + unsigned int ipC = vIPC[GetRand(vIPC.size())]; + + // Organize all addresses in the class C by IP + map > mapIP; + CRITICAL_BLOCK(cs_mapIRCAddresses) + CRITICAL_BLOCK(cs_mapAddresses) + { + int64 nDelay = ((30 * 60) << vNodes.size()); + if (!fIRCOnly) + { + nDelay *= 2; + if (vNodes.size() >= 3) + nDelay *= 4; + if (!mapIRCAddresses.empty()) + nDelay *= 100; + } + + for (map, CAddress>::iterator mi = mapAddresses.lower_bound(CAddress(ipC, 0).GetKey()); + mi != mapAddresses.upper_bound(CAddress(ipC | ~nIPCMask, 0xffff).GetKey()); + ++mi) + { + const CAddress& addr = (*mi).second; + if (fIRCOnly && !mapIRCAddresses.count((*mi).first)) + continue; + + int64 nRandomizer = (addr.nLastFailed * addr.ip * 7777U) % 20000; + if (GetTime() - addr.nLastFailed > nDelay * nRandomizer / 10000) + mapIP[addr.ip].push_back(addr); + } + } + if (mapIP.empty()) + continue; + + // Choose a random IP in the class C + map >::iterator mi = mapIP.begin(); + advance(mi, GetRand(mapIP.size())); + + // Once we've chosen an IP, we'll try every given port before moving on + foreach(const CAddress& addrConnect, (*mi).second) + { + // + // Initiate outbound network connection + // + CheckForShutdown(1); + if (addrConnect.ip == addrLocalHost.ip || !addrConnect.IsIPv4() || FindNode(addrConnect.ip)) + continue; + + vfThreadRunning[1] = false; + CNode* pnode = ConnectNode(addrConnect); + vfThreadRunning[1] = true; + CheckForShutdown(1); + if (!pnode) + continue; + pnode->fNetworkNode = true; + + if (addrLocalHost.IsRoutable()) + { + // Advertise our address + vector vAddrToSend; + vAddrToSend.push_back(addrLocalHost); + pnode->PushMessage("addr", vAddrToSend); + } + + // Get as many addresses as we can + pnode->PushMessage("getaddr"); + + ////// should the one on the receiving end do this too? + // Subscribe our local subscription list + const unsigned int nHops = 0; + for (unsigned int nChannel = 0; nChannel < pnodeLocalHost->vfSubscribe.size(); nChannel++) + if (pnodeLocalHost->vfSubscribe[nChannel]) + pnode->PushMessage("subscribe", nChannel, nHops); + + break; + } + } +} + + + + + + + + +void ThreadMessageHandler(void* parg) +{ + IMPLEMENT_RANDOMIZE_STACK(ThreadMessageHandler(parg)); + + loop + { + vfThreadRunning[2] = true; + CheckForShutdown(2); + try + { + ThreadMessageHandler2(parg); + } + CATCH_PRINT_EXCEPTION("ThreadMessageHandler()") + vfThreadRunning[2] = false; + Sleep(5000); + } +} + +void ThreadMessageHandler2(void* parg) +{ + printf("ThreadMessageHandler started\n"); + SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_BELOW_NORMAL); + loop + { + // Poll the connected nodes for messages + vector vNodesCopy; + CRITICAL_BLOCK(cs_vNodes) + vNodesCopy = vNodes; + foreach(CNode* pnode, vNodesCopy) + { + pnode->AddRef(); + + // Receive messages + TRY_CRITICAL_BLOCK(pnode->cs_vRecv) + ProcessMessages(pnode); + + // Send messages + TRY_CRITICAL_BLOCK(pnode->cs_vSend) + SendMessages(pnode); + + pnode->Release(); + } + + // Wait and allow messages to bunch up + vfThreadRunning[2] = false; + Sleep(100); + vfThreadRunning[2] = true; + CheckForShutdown(2); + } +} + + + + + + + + + +//// todo: start one thread per processor, use getenv("NUMBER_OF_PROCESSORS") +void ThreadBitcoinMiner(void* parg) +{ + vfThreadRunning[3] = true; + CheckForShutdown(3); + try + { + bool fRet = BitcoinMiner(); + printf("BitcoinMiner returned %s\n\n\n", fRet ? "true" : "false"); + } + CATCH_PRINT_EXCEPTION("BitcoinMiner()") + vfThreadRunning[3] = false; +} + + + + + + + + + + + +bool StartNode(string& strError) +{ + strError = ""; + + // Sockets startup + WSADATA wsadata; + int ret = WSAStartup(MAKEWORD(2,2), &wsadata); + if (ret != NO_ERROR) + { + strError = strprintf("Error: TCP/IP socket library failed to start (WSAStartup returned error %d)", ret); + printf("%s\n", strError.c_str()); + return false; + } + + // Get local host ip + char pszHostName[255]; + if (gethostname(pszHostName, 255) == SOCKET_ERROR) + { + strError = strprintf("Error: Unable to get IP address of this computer (gethostname returned error %d)", WSAGetLastError()); + printf("%s\n", strError.c_str()); + return false; + } + struct hostent* phostent = gethostbyname(pszHostName); + if (!phostent) + { + strError = strprintf("Error: Unable to get IP address of this computer (gethostbyname returned error %d)", WSAGetLastError()); + printf("%s\n", strError.c_str()); + return false; + } + addrLocalHost = CAddress(*(long*)(phostent->h_addr_list[0]), + DEFAULT_PORT, + nLocalServices); + printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); + + // Create socket for listening for incoming connections + SOCKET hListenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); + if (hListenSocket == INVALID_SOCKET) + { + strError = strprintf("Error: Couldn't open socket for incoming connections (socket returned error %d)", WSAGetLastError()); + printf("%s\n", strError.c_str()); + return false; + } + + // Set to nonblocking, incoming connections will also inherit this + u_long nOne = 1; + if (ioctlsocket(hListenSocket, FIONBIO, &nOne) == SOCKET_ERROR) + { + strError = strprintf("Error: Couldn't set properties on socket for incoming connections (ioctlsocket returned error %d)", WSAGetLastError()); + printf("%s\n", strError.c_str()); + return false; + } + + // The sockaddr_in structure specifies the address family, + // IP address, and port for the socket that is being bound + int nRetryLimit = 15; + struct sockaddr_in sockaddr = addrLocalHost.GetSockAddr(); + if (bind(hListenSocket, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) == SOCKET_ERROR) + { + int nErr = WSAGetLastError(); + if (nErr == WSAEADDRINUSE) + strError = strprintf("Error: Unable to bind to port %s on this computer. The program is probably already running.", addrLocalHost.ToString().c_str()); + else + strError = strprintf("Error: Unable to bind to port %s on this computer (bind returned error %d)", addrLocalHost.ToString().c_str(), nErr); + printf("%s\n", strError.c_str()); + return false; + } + printf("bound to addrLocalHost = %s\n\n", addrLocalHost.ToString().c_str()); + + // Listen for incoming connections + if (listen(hListenSocket, SOMAXCONN) == SOCKET_ERROR) + { + strError = strprintf("Error: Listening for incoming connections failed (listen returned error %d)", WSAGetLastError()); + printf("%s\n", strError.c_str()); + return false; + } + + // Get our external IP address for incoming connections + if (addrIncoming.ip) + addrLocalHost.ip = addrIncoming.ip; + + if (GetMyExternalIP(addrLocalHost.ip)) + { + addrIncoming = addrLocalHost; + CWalletDB().WriteSetting("addrIncoming", addrIncoming); + } + + // Get addresses from IRC and advertise ours + if (_beginthread(ThreadIRCSeed, 0, NULL) == -1) + printf("Error: _beginthread(ThreadIRCSeed) failed\n"); + + // + // Start threads + // + if (_beginthread(ThreadSocketHandler, 0, new SOCKET(hListenSocket)) == -1) + { + strError = "Error: _beginthread(ThreadSocketHandler) failed"; + printf("%s\n", strError.c_str()); + return false; + } + + if (_beginthread(ThreadOpenConnections, 0, NULL) == -1) + { + strError = "Error: _beginthread(ThreadOpenConnections) failed"; + printf("%s\n", strError.c_str()); + return false; + } + + if (_beginthread(ThreadMessageHandler, 0, NULL) == -1) + { + strError = "Error: _beginthread(ThreadMessageHandler) failed"; + printf("%s\n", strError.c_str()); + return false; + } + + return true; +} + +bool StopNode() +{ + printf("StopNode()\n"); + fShutdown = true; + nTransactionsUpdated++; + int64 nStart = GetTime(); + while (vfThreadRunning[0] || vfThreadRunning[2] || vfThreadRunning[3]) + { + if (GetTime() - nStart > 15) + break; + Sleep(20); + } + if (vfThreadRunning[0]) printf("ThreadSocketHandler still running\n"); + if (vfThreadRunning[1]) printf("ThreadOpenConnections still running\n"); + if (vfThreadRunning[2]) printf("ThreadMessageHandler still running\n"); + if (vfThreadRunning[3]) printf("ThreadBitcoinMiner still running\n"); + while (vfThreadRunning[2]) + Sleep(20); + Sleep(50); + + // Sockets shutdown + WSACleanup(); + return true; +} + +void CheckForShutdown(int n) +{ + if (fShutdown) + { + if (n != -1) + vfThreadRunning[n] = false; + if (n == 0) + foreach(CNode* pnode, vNodes) + closesocket(pnode->hSocket); + _endthread(); + } +} diff --git a/net.h b/net.h new file mode 100644 index 00000000..cd311fc5 --- /dev/null +++ b/net.h @@ -0,0 +1,856 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +class CMessageHeader; +class CAddress; +class CInv; +class CRequestTracker; +class CNode; + + + +static const unsigned short DEFAULT_PORT = htons(8333); +static const unsigned int PUBLISH_HOPS = 5; +enum +{ + NODE_NETWORK = (1 << 0), +}; + + + + + + +bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet); +bool GetMyExternalIP(unsigned int& ipRet); +bool AddAddress(CAddrDB& addrdb, const CAddress& addr); +CNode* FindNode(unsigned int ip); +CNode* ConnectNode(CAddress addrConnect, int64 nTimeout=0); +void AbandonRequests(void (*fn)(void*, CDataStream&), void* param1); +bool AnySubscribed(unsigned int nChannel); +void ThreadBitcoinMiner(void* parg); +bool StartNode(string& strError=REF(string())); +bool StopNode(); +void CheckForShutdown(int n); + + + + + + + + + +// +// Message header +// (4) message start +// (12) command +// (4) size + +// The message start string is designed to be unlikely to occur in normal data. +// The characters are rarely used upper ascii, not valid as UTF-8, and produce +// a large 4-byte int at any alignment. +static const char pchMessageStart[4] = { 0xf9, 0xbe, 0xb4, 0xd9 }; + +class CMessageHeader +{ +public: + enum { COMMAND_SIZE=12 }; + char pchMessageStart[sizeof(::pchMessageStart)]; + char pchCommand[COMMAND_SIZE]; + unsigned int nMessageSize; + + CMessageHeader() + { + memcpy(pchMessageStart, ::pchMessageStart, sizeof(pchMessageStart)); + memset(pchCommand, 0, sizeof(pchCommand)); + pchCommand[1] = 1; + nMessageSize = -1; + } + + CMessageHeader(const char* pszCommand, unsigned int nMessageSizeIn) + { + memcpy(pchMessageStart, ::pchMessageStart, sizeof(pchMessageStart)); + strncpy(pchCommand, pszCommand, COMMAND_SIZE); + nMessageSize = nMessageSizeIn; + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(FLATDATA(pchMessageStart)); + READWRITE(FLATDATA(pchCommand)); + READWRITE(nMessageSize); + ) + + string GetCommand() + { + if (pchCommand[COMMAND_SIZE-1] == 0) + return string(pchCommand, pchCommand + strlen(pchCommand)); + else + return string(pchCommand, pchCommand + COMMAND_SIZE); + } + + bool IsValid() + { + // Check start string + if (memcmp(pchMessageStart, ::pchMessageStart, sizeof(pchMessageStart)) != 0) + return false; + + // Check the command string for errors + for (char* p1 = pchCommand; p1 < pchCommand + COMMAND_SIZE; p1++) + { + if (*p1 == 0) + { + // Must be all zeros after the first zero + for (; p1 < pchCommand + COMMAND_SIZE; p1++) + if (*p1 != 0) + return false; + } + else if (*p1 < ' ' || *p1 > 0x7E) + return false; + } + + // Message size + if (nMessageSize > 0x10000000) + { + printf("CMessageHeader::IsValid() : nMessageSize too large %u\n", nMessageSize); + return false; + } + + return true; + } +}; + + + + + + +static const unsigned char pchIPv4[12] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff }; + +class CAddress +{ +public: + uint64 nServices; + unsigned char pchReserved[12]; + unsigned int ip; + unsigned short port; + + // disk only + unsigned int nTime; + + // memory only + unsigned int nLastFailed; + + CAddress() + { + nServices = 0; + memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); + ip = 0; + port = DEFAULT_PORT; + nTime = GetAdjustedTime(); + nLastFailed = 0; + } + + CAddress(unsigned int ipIn, unsigned short portIn=DEFAULT_PORT, uint64 nServicesIn=0) + { + nServices = nServicesIn; + memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); + ip = ipIn; + port = portIn; + nTime = GetAdjustedTime(); + nLastFailed = 0; + } + + explicit CAddress(const struct sockaddr_in& sockaddr, uint64 nServicesIn=0) + { + nServices = nServicesIn; + memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); + ip = sockaddr.sin_addr.s_addr; + port = sockaddr.sin_port; + nTime = GetAdjustedTime(); + nLastFailed = 0; + } + + explicit CAddress(const char* pszIn, uint64 nServicesIn=0) + { + nServices = nServicesIn; + memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); + ip = 0; + port = DEFAULT_PORT; + nTime = GetAdjustedTime(); + nLastFailed = 0; + + char psz[100]; + if (strlen(pszIn) > ARRAYLEN(psz)-1) + return; + strcpy(psz, pszIn); + unsigned int a, b, c, d, e; + if (sscanf(psz, "%u.%u.%u.%u:%u", &a, &b, &c, &d, &e) < 4) + return; + char* pszPort = strchr(psz, ':'); + if (pszPort) + { + *pszPort++ = '\0'; + port = htons(atoi(pszPort)); + } + ip = inet_addr(psz); + } + + IMPLEMENT_SERIALIZE + ( + if (nType & SER_DISK) + { + READWRITE(nVersion); + READWRITE(nTime); + } + READWRITE(nServices); + READWRITE(FLATDATA(pchReserved)); + READWRITE(ip); + READWRITE(port); + ) + + friend inline bool operator==(const CAddress& a, const CAddress& b) + { + return (memcmp(a.pchReserved, b.pchReserved, sizeof(a.pchReserved)) == 0 && + a.ip == b.ip && + a.port == b.port); + } + + friend inline bool operator<(const CAddress& a, const CAddress& b) + { + int ret = memcmp(a.pchReserved, b.pchReserved, sizeof(a.pchReserved)); + if (ret < 0) + return true; + else if (ret == 0) + { + if (ntohl(a.ip) < ntohl(b.ip)) + return true; + else if (a.ip == b.ip) + return ntohs(a.port) < ntohs(b.port); + } + return false; + } + + vector GetKey() const + { + CDataStream ss; + ss.reserve(18); + ss << FLATDATA(pchReserved) << ip << port; + + #if defined(_MSC_VER) && _MSC_VER < 1300 + return vector((unsigned char*)&ss.begin()[0], (unsigned char*)&ss.end()[0]); + #else + return vector(ss.begin(), ss.end()); + #endif + } + + struct sockaddr_in GetSockAddr() const + { + struct sockaddr_in sockaddr; + sockaddr.sin_family = AF_INET; + sockaddr.sin_addr.s_addr = ip; + sockaddr.sin_port = port; + return sockaddr; + } + + bool IsIPv4() const + { + return (memcmp(pchReserved, pchIPv4, sizeof(pchIPv4)) == 0); + } + + bool IsRoutable() const + { + return !(GetByte(3) == 10 || (GetByte(3) == 192 && GetByte(2) == 168) || GetByte(3) == 127 || GetByte(3) == 0); + } + + unsigned char GetByte(int n) const + { + return ((unsigned char*)&ip)[3-n]; + } + + string ToStringIPPort() const + { + return strprintf("%u.%u.%u.%u:%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0), ntohs(port)); + } + + string ToStringIP() const + { + return strprintf("%u.%u.%u.%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0)); + } + + string ToString() const + { + return strprintf("%u.%u.%u.%u:%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0), ntohs(port)); + //return strprintf("%u.%u.%u.%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0)); + } + + void print() const + { + printf("CAddress(%s)\n", ToString().c_str()); + } +}; + + + + + + + +enum +{ + MSG_TX = 1, + MSG_BLOCK, + MSG_REVIEW, + MSG_PRODUCT, + MSG_TABLE, +}; + +static const char* ppszTypeName[] = +{ + "ERROR", + "tx", + "block", + "review", + "product", + "table", +}; + +class CInv +{ +public: + int type; + uint256 hash; + + CInv() + { + type = 0; + hash = 0; + } + + CInv(int typeIn, const uint256& hashIn) + { + type = typeIn; + hash = hashIn; + } + + CInv(const string& strType, const uint256& hashIn) + { + int i; + for (i = 1; i < ARRAYLEN(ppszTypeName); i++) + { + if (strType == ppszTypeName[i]) + { + type = i; + break; + } + } + if (i == ARRAYLEN(ppszTypeName)) + throw std::out_of_range(strprintf("CInv::CInv(string, uint256) : unknown type '%s'", strType.c_str())); + hash = hashIn; + } + + IMPLEMENT_SERIALIZE + ( + READWRITE(type); + READWRITE(hash); + ) + + friend inline bool operator<(const CInv& a, const CInv& b) + { + return (a.type < b.type || (a.type == b.type && a.hash < b.hash)); + } + + bool IsKnownType() const + { + return (type >= 1 && type < ARRAYLEN(ppszTypeName)); + } + + const char* GetCommand() const + { + if (!IsKnownType()) + throw std::out_of_range(strprintf("CInv::GetCommand() : type=% unknown type", type)); + return ppszTypeName[type]; + } + + string ToString() const + { + return strprintf("%s %s", GetCommand(), hash.ToString().substr(0,14).c_str()); + } + + void print() const + { + printf("CInv(%s)\n", ToString().c_str()); + } +}; + + + + + +class CRequestTracker +{ +public: + void (*fn)(void*, CDataStream&); + void* param1; + + explicit CRequestTracker(void (*fnIn)(void*, CDataStream&)=NULL, void* param1In=NULL) + { + fn = fnIn; + param1 = param1In; + } + + bool IsNull() + { + return fn == NULL; + } +}; + + + + + +extern bool fClient; +extern uint64 nLocalServices; +extern CAddress addrLocalHost; +extern CNode* pnodeLocalHost; +extern bool fShutdown; +extern array vfThreadRunning; +extern vector vNodes; +extern CCriticalSection cs_vNodes; +extern map, CAddress> mapAddresses; +extern CCriticalSection cs_mapAddresses; +extern map mapRelay; +extern deque > vRelayExpiration; +extern CCriticalSection cs_mapRelay; +extern map mapAlreadyAskedFor; +extern CAddress addrProxy; + + + + + +class CNode +{ +public: + // socket + uint64 nServices; + SOCKET hSocket; + CDataStream vSend; + CDataStream vRecv; + CCriticalSection cs_vSend; + CCriticalSection cs_vRecv; + unsigned int nPushPos; + CAddress addr; + int nVersion; + bool fClient; + bool fInbound; + bool fNetworkNode; + bool fDisconnect; +protected: + int nRefCount; +public: + int64 nReleaseTime; + map mapRequests; + CCriticalSection cs_mapRequests; + + // flood + vector vAddrToSend; + set setAddrKnown; + + // inventory based relay + set setInventoryKnown; + set setInventoryKnown2; + vector vInventoryToSend; + CCriticalSection cs_inventory; + multimap mapAskFor; + + // publish and subscription + vector vfSubscribe; + + + CNode(SOCKET hSocketIn, CAddress addrIn, bool fInboundIn=false) + { + nServices = 0; + hSocket = hSocketIn; + vSend.SetType(SER_NETWORK); + vRecv.SetType(SER_NETWORK); + nPushPos = -1; + addr = addrIn; + nVersion = 0; + fClient = false; // set by version message + fInbound = fInboundIn; + fNetworkNode = false; + fDisconnect = false; + nRefCount = 0; + nReleaseTime = 0; + vfSubscribe.assign(256, false); + + // Push a version message + /// when NTP implemented, change to just nTime = GetAdjustedTime() + int64 nTime = (fInbound ? GetAdjustedTime() : GetTime()); + PushMessage("version", VERSION, nLocalServices, nTime, addr); + } + + ~CNode() + { + if (hSocket != INVALID_SOCKET) + closesocket(hSocket); + } + +private: + CNode(const CNode&); + void operator=(const CNode&); +public: + + + bool ReadyToDisconnect() + { + return fDisconnect || GetRefCount() <= 0; + } + + int GetRefCount() + { + return max(nRefCount, 0) + (GetTime() < nReleaseTime ? 1 : 0); + } + + void AddRef(int64 nTimeout=0) + { + if (nTimeout != 0) + nReleaseTime = max(nReleaseTime, GetTime() + nTimeout); + else + nRefCount++; + } + + void Release() + { + nRefCount--; + } + + + + void AddInventoryKnown(const CInv& inv) + { + CRITICAL_BLOCK(cs_inventory) + setInventoryKnown.insert(inv); + } + + void PushInventory(const CInv& inv) + { + CRITICAL_BLOCK(cs_inventory) + if (!setInventoryKnown.count(inv)) + vInventoryToSend.push_back(inv); + } + + void AskFor(const CInv& inv) + { + // We're using mapAskFor as a priority queue, + // the key is the earliest time the request can be sent + int64& nRequestTime = mapAlreadyAskedFor[inv]; + printf("askfor %s %I64d\n", inv.ToString().c_str(), nRequestTime); + + // Make sure not to reuse time indexes to keep things in the same order + int64 nNow = (GetTime() - 1) * 1000000; + static int64 nLastTime; + nLastTime = nNow = max(nNow, ++nLastTime); + + // Each retry is 2 minutes after the last + nRequestTime = max(nRequestTime + 2 * 60 * 1000000, nNow); + mapAskFor.insert(make_pair(nRequestTime, inv)); + } + + + + void BeginMessage(const char* pszCommand) + { + EnterCriticalSection(&cs_vSend); + if (nPushPos != -1) + AbortMessage(); + nPushPos = vSend.size(); + vSend << CMessageHeader(pszCommand, 0); + printf("sending: %-12s ", pszCommand); + } + + void AbortMessage() + { + if (nPushPos == -1) + return; + vSend.resize(nPushPos); + nPushPos = -1; + LeaveCriticalSection(&cs_vSend); + printf("(aborted)\n"); + } + + void EndMessage() + { + extern int nDropMessagesTest; + if (nDropMessagesTest > 0 && GetRand(nDropMessagesTest) == 0) + { + printf("dropmessages DROPPING SEND MESSAGE\n"); + AbortMessage(); + return; + } + + if (nPushPos == -1) + return; + + // Patch in the size + unsigned int nSize = vSend.size() - nPushPos - sizeof(CMessageHeader); + memcpy((char*)&vSend[nPushPos] + offsetof(CMessageHeader, nMessageSize), &nSize, sizeof(nSize)); + + printf("(%d bytes) ", nSize); + //for (int i = nPushPos+sizeof(CMessageHeader); i < min(vSend.size(), nPushPos+sizeof(CMessageHeader)+20U); i++) + // printf("%02x ", vSend[i] & 0xff); + printf("\n"); + + nPushPos = -1; + LeaveCriticalSection(&cs_vSend); + } + + void EndMessageAbortIfEmpty() + { + if (nPushPos == -1) + return; + int nSize = vSend.size() - nPushPos - sizeof(CMessageHeader); + if (nSize > 0) + EndMessage(); + else + AbortMessage(); + } + + const char* GetMessageCommand() const + { + if (nPushPos == -1) + return ""; + return &vSend[nPushPos] + offsetof(CMessageHeader, pchCommand); + } + + + + + void PushMessage(const char* pszCommand) + { + try + { + BeginMessage(pszCommand); + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1) + { + try + { + BeginMessage(pszCommand); + vSend << a1; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2, const T3& a3) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2 << a3; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2, const T3& a3, const T4& a4) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2 << a3 << a4; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + + void PushRequest(const char* pszCommand, + void (*fn)(void*, CDataStream&), void* param1) + { + uint256 hashReply; + RAND_bytes((unsigned char*)&hashReply, sizeof(hashReply)); + + CRITICAL_BLOCK(cs_mapRequests) + mapRequests[hashReply] = CRequestTracker(fn, param1); + + PushMessage(pszCommand, hashReply); + } + + template + void PushRequest(const char* pszCommand, const T1& a1, + void (*fn)(void*, CDataStream&), void* param1) + { + uint256 hashReply; + RAND_bytes((unsigned char*)&hashReply, sizeof(hashReply)); + + CRITICAL_BLOCK(cs_mapRequests) + mapRequests[hashReply] = CRequestTracker(fn, param1); + + PushMessage(pszCommand, hashReply, a1); + } + + template + void PushRequest(const char* pszCommand, const T1& a1, const T2& a2, + void (*fn)(void*, CDataStream&), void* param1) + { + uint256 hashReply; + RAND_bytes((unsigned char*)&hashReply, sizeof(hashReply)); + + CRITICAL_BLOCK(cs_mapRequests) + mapRequests[hashReply] = CRequestTracker(fn, param1); + + PushMessage(pszCommand, hashReply, a1, a2); + } + + + + bool IsSubscribed(unsigned int nChannel); + void Subscribe(unsigned int nChannel, unsigned int nHops=0); + void CancelSubscribe(unsigned int nChannel); + void Disconnect(); +}; + + + + + + + + + + +inline void RelayInventory(const CInv& inv) +{ + // Put on lists to offer to the other nodes + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + pnode->PushInventory(inv); +} + +template +void RelayMessage(const CInv& inv, const T& a) +{ + CDataStream ss(SER_NETWORK); + ss.reserve(10000); + ss << a; + RelayMessage(inv, ss); +} + +template<> +inline void RelayMessage<>(const CInv& inv, const CDataStream& ss) +{ + CRITICAL_BLOCK(cs_mapRelay) + { + // Expire old relay messages + while (!vRelayExpiration.empty() && vRelayExpiration.front().first < GetTime()) + { + mapRelay.erase(vRelayExpiration.front().second); + vRelayExpiration.pop_front(); + } + + // Save original serialized message so newer versions are preserved + mapRelay[inv] = ss; + vRelayExpiration.push_back(make_pair(GetTime() + 15 * 60, inv)); + } + + RelayInventory(inv); +} + + + + + + + + +// +// Templates for the publish and subscription system. +// The object being published as T& obj needs to have: +// a set setSources member +// specializations of AdvertInsert and AdvertErase +// Currently implemented for CTable and CProduct. +// + +template +void AdvertStartPublish(CNode* pfrom, unsigned int nChannel, unsigned int nHops, T& obj) +{ + // Add to sources + obj.setSources.insert(pfrom->addr.ip); + + if (!AdvertInsert(obj)) + return; + + // Relay + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + if (pnode != pfrom && (nHops < PUBLISH_HOPS || pnode->IsSubscribed(nChannel))) + pnode->PushMessage("publish", nChannel, nHops, obj); +} + +template +void AdvertStopPublish(CNode* pfrom, unsigned int nChannel, unsigned int nHops, T& obj) +{ + uint256 hash = obj.GetHash(); + + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + if (pnode != pfrom && (nHops < PUBLISH_HOPS || pnode->IsSubscribed(nChannel))) + pnode->PushMessage("pub-cancel", nChannel, nHops, hash); + + AdvertErase(obj); +} + +template +void AdvertRemoveSource(CNode* pfrom, unsigned int nChannel, unsigned int nHops, T& obj) +{ + // Remove a source + obj.setSources.erase(pfrom->addr.ip); + + // If no longer supported by any sources, cancel it + if (obj.setSources.empty()) + AdvertStopPublish(pfrom, nChannel, nHops, obj); +} diff --git a/rc/addressbook16.bmp b/rc/addressbook16.bmp new file mode 100644 index 0000000000000000000000000000000000000000..c5576910b1b6b95c23ed7c53adec399a944e610a GIT binary patch literal 1334 zcmZ9MKWrOS9LK*+iDcA*T|jP9K#EfVb%3_kxoL?Sz~+CTKo0?umkcM}a-)X~w`8y~ zWwata$wP;F^b*OVM^AaW@iN6rqzqX?CP!j;d|n(O=-qwq_kQ2^d%y4Zy_2r_Y;|6j z*XnE1Jw~M~TvK5-acvg<`*zOBxvAXL<4n`D6HQM%GM!B(AlC_)PJEuJ;Aq!F>#NkLovPNIJ5$BCNMjXNzRgQ_+*rp!g; zA@kIOH0r69ChE4y+o3w`mbys@n%a#*r9QS=N;g8a!={p`qHa`Dn3U9Rt|@8O6xBjv zRneM91uViXss*BIg-Ct1pujo>E&wqwzyhb34*>)GZGZ*v0ek=-zz6UFd;rgan8}zy z9K*-(2F}14I0I)qN3Dgo@D|R(;UVVyV|>JTi1A?2a=ck4gU;+G$Cp8$Dx%MwXb<4< zWDqq77(5xY4U>i@25o~j0eEeA^bPt3eWn2Qp2vg1gW((~M&aj7mEptAkqMyg_a@XPLPt?d zkC;5hhr8_UYhqQ@*XHJ?wzjskv$LbUy*(Wq9B6-kUx$Z>IyySi@$s=vPfvAzey)p) z3;p%C)5{kxw7tE}>9ScKn#}@lhQO)#%Jtps`s3Lw7VG>=O7B&F`?+#Qdc9t(-w6Nw z{nu~qFaPlTdHBKC%zr1p5Vd?$#kAAwEFRk42>mO|ReC~Q-`SIO_!hG=g_r>}+ss?m65uRi?;YjfP( literal 0 HcmV?d00001 diff --git a/rc/addressbook16mask.bmp b/rc/addressbook16mask.bmp new file mode 100644 index 0000000000000000000000000000000000000000..d3a478d1ad1d4800008fda5be5d583f3d0423480 GIT binary patch literal 126 tcmZ?rtz&=yJ0PV2!~#&v$iN7eZ~&8-#Q*>Q!Giz)F))yd?S4Sa0{}e{AL0N2 literal 0 HcmV?d00001 diff --git a/rc/addressbook20.bmp b/rc/addressbook20.bmp new file mode 100644 index 0000000000000000000000000000000000000000..2b33b228aaec9ef51fb3dd46b3eab98f3d61d7df GIT binary patch literal 1478 zcma)+J!~6g9LAp(Xj4#-1LY%>FBd2^1%kPdf|c^&f`Ee-Fy_la3iQe$IKDk(@RGsG z9usXbe`4esg7wT_5VJWhTCq&7#TB`8C#7sMrJhB7@1L`@i5T{jfVq~s9}w4E! zw_*2z22GFXsne(_Y6L`^Q&Ln9)UH?6LQUklMn%-9Rn@6^>eWhWRy~zaBdDkql+^G% zxmCM%<*{$qXH@q}s-X&AKx|alBhM=mUG_yagLz^>b>!A&$jizrda9N1+@k8GJjy9> z3#t{1s-n7^Q?J+hqv|Z8#9m33I+p%fx*CFV4wui69xl=fgy&0 zHi&!}3=9Sa0|TuQJ`4s11A_s;Fh8c5q0o@z)5B-)!(r0iHpf?1ZLO}ZYHe*z8yg$i z+}zam_O`aRwzRXequt$I?d|Ppe}7*G2M0PlJk-Dc8U1tPhSt~DIbB|sAI&C#J4GgAUy7(iLxq5=0b7wD>|14iTJ13pEig)jYa`|ijuj|Xd{$44U-+JXM>s@~B ze0F~B8Rz8vXU<$GJsqUpEY+uS&Pl@Tw=exP@s8g$^ZbVs@0%yToOqAF{OrTj+PqKh z`{=0l_^GM)#Y3On@@AKo&)K~1UcWZ2U08f)$$4~Pap8&8SARLm(^2hh-qP}ew)Tg& zy<4?!7M$E7wco$@8{Qk2Q|5D5QkK+v>=D;`4}9?ACI0Kqe*VH!_CVjizOwS^rPG<` PK3-Y5_U;`Xahmo&>vaK4 literal 0 HcmV?d00001 diff --git a/rc/addressbook20mask.bmp b/rc/addressbook20mask.bmp new file mode 100644 index 0000000000000000000000000000000000000000..56ce6125dbb83abfac437bbbcdb611f45ee57cff GIT binary patch literal 142 zcmZ?r?PGudJ0PV2#3E44$iN7e2mq2o+z`wJ7J(4||Nn>c{{KI~KrXi54>yJZ01nMB AR{#J2 literal 0 HcmV?d00001 diff --git a/rc/bitcoin.ico b/rc/bitcoin.ico new file mode 100644 index 0000000000000000000000000000000000000000..88cc240e2d4f77fa66730593a890a17aab86cbdc GIT binary patch literal 22486 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 IOZolyKiCZ4$N&HU literal 0 HcmV?d00001 diff --git a/rc/check.ico b/rc/check.ico new file mode 100644 index 0000000000000000000000000000000000000000..0c4e6e81473d47761346fd71e171a807bda5ec58 GIT binary patch literal 766 zcmdUtAritc5Ji7?>Oo=IK{2@m+>AIRcZlpMaugJUL?RfLzb&Fm5tZVcTcTu78nlc+X79T)o9yH~Bu18~wK99d`Ux|y2#0TZ_)l%(v!!U*NL^S< zm%;~9QW>8Y{WOqh07L_>g8?ux2p5ho`>mS)zsJ~LYQ7gH)G3saMH zyB~}P>RC^tqazJ22kKf^qtTH%)*+8HB(`iz!{JD6Ym}Tk7?DT0LLYf-R`9Lfu|hGN{Hh5!TD1R4whhJ=^V zS4W7?sJSvv_HRM}BgTY`h7z_r3!Cba8Q^tE($rUtjC@AJfm>U2Se|a=Q6{ASdcovs^b%O|_~O^A9J3 zo8Gy{cW=fyyV5$bWwCUxW^1*ko3u|Q!khV8bNYU6kCq$$b?M$c zXwEFX|G1o6qw%(MADT1u`ciJFjYp+xPCtD)5ti!rOE=q`nY3s1`n}Q>|Lf1qO_lCf Z2B?~wyHmPbfR5?STF&s_p4QyW^e;amw9xQ85j>RFfcv<;txRlgMp#`4+F#he?a^n U34{0`aiBa{9Y{S$UjqXJ0D?~%)Bpeg literal 0 HcmV?d00001 diff --git a/rc/send16masknoshadow.bmp b/rc/send16masknoshadow.bmp new file mode 100644 index 0000000000000000000000000000000000000000..faf24e0d8aa81faf80072c7807e52d749cad63c0 GIT binary patch literal 126 zcmZ?rtz&=yJ0PV2!~#&v$iN7eZ~&8-#Q*>Q!Geqp3=E71fcOCre_&wv{{x8s0P#N{ U1}Xv5AU;qWBo9^xQV-Mz01_z}*8l(j literal 0 HcmV?d00001 diff --git a/rc/send20.bmp b/rc/send20.bmp new file mode 100644 index 0000000000000000000000000000000000000000..2b90422b38467ae30a2b6a866adbf0ad3c3f97f3 GIT binary patch literal 1478 zcmZXSKWrOi7{;GaLQNamP(c#|QaKSso$~KuQiKB~Sfms~DLR@G4onc1ZpoM-LzWCy zrW+Vekxuf^0gqWCdGvy0mVEN)Oz9FSLza-qkr-Zn=R}0i&wlRv?)Sa-KF@b{`Tn27 z8jRb|N;g@(bgCic2K=|7!F2A@wXWLMR;S-l%UasBO*O5lhBegf^^{nmZnvk{Vs*M* z)vd1NFwv^5s?+Ie*_N3*YNI7vQmfNe&1zceFR9&btA#=fRqNMS(;o#EsMUh^LuMEX z`hm(;R&W?-bCV3Ha#)dXz6w@Qv)Pnqp2A+J@_t##W}?cIic0$>h22o2HmXLWA%EXj z&T>i`iR#U|WKyYDQqT#cz7!|1>IwW%T5c>Wx1Up`g-^$q_F>7g^h631&JZt2J*nI* zs}|R!uH?1I5QkDn3gSSeMoE>pBDEzy@#Qr<`LVB3y(BO3REP`m>+sl9exSU(I()UD ze4X{Gr_uVT@~e48a>x`Fg^DObDS{W_5uDA4MHIm!cnV8lDJ+Gh8Kc(09o)ej%)xjU zk--dR4macD#A8bFIr(tUKP1JUD(39oM0*4yQ;KPdK}x0+@087yCMn)2-b^U-hU1^& zpW@FJqCVqfNXd{|mH@8e;Pen?uDkB#>=(#Rj30j=G8h?*3`Pbc1CJnjlEKJeWN>7l z4Wgb5Mg}8;k%86-P6i``k--RH_&b^z4;n=gJtFQSa@e4ERujonQ;Um>T3K1q`ue&y zHa4`qy{)aSE$!^=Xm@v4dwY92I5^PJ(UFdikM-Bzrr#ew*4o+{Zvdf^`qjovy#Co%$I+@J$@dB6B8f(^6~4>pU}%==y%-wtc z`osH^S7B4r)6+9EckZw;_raBil`&XhDx=xP?CkZot6ui($jfisni}w_EB+5zMp!mp zc<$!Ztz7BRSQc>pbPVUO41f0E-0{K-mo9vH_i9%4?^yU_=-W@Ty#E;s&wTgc|Io$i T{JW>;KP#Mn?J0D&nXUc-U3=Ry literal 0 HcmV?d00001 diff --git a/rc/send20mask.bmp b/rc/send20mask.bmp new file mode 100644 index 0000000000000000000000000000000000000000..f124d0da084e68cac62312e9fa8a0fbf9c7c5075 GIT binary patch literal 142 zcmZ?r?PGudJ0PV2#3E44$iN7e2mq2o+z`wJ7J(4||Nm!TJix%f_yCAM0PznX{=>jv m|BrzIC + #ifndef OPENSSL_NO_RSA + #include + #endif + +Add this to crypto\err\err_all.c before the ERR_load_crypto_strings line: + void ERR_load_RSA_strings(void) { } + +Edit ms\mingw32.bat and replace the Configure line's parameters with this +no-everything list. You have to put this in the batch file because batch +files can't handle more than 9 parameters. + perl Configure mingw threads no-rc2 no-rc4 no-rc5 no-idea no-des no-bf no-cast no-aes no-camellia no-seed no-rsa no-dh + +Also REM out the following line in ms\mingw32.bat. The build fails after it's +already finished building libeay32, which is all we care about, but the +failure aborts the script before it runs dllwrap to generate libeay32.dll. + REM if errorlevel 1 goto end + +Build + ms\mingw32.bat + +If you want to use it with MSVC, generate the .lib file + lib /machine:i386 /def:ms\libeay32.def /out:out\libeay32.lib + + +Berkeley DB +----------- +MinGW with MSYS: +cd \DB\build_unix +sh ../dist/configure --enable-mingw --enable-cxx +make + + +Boost +----- +You may need Boost version 1.35 to build with MSVC 6.0. I couldn't get +version 1.37 to compile with MSVC 6.0. diff --git a/script.cpp b/script.cpp new file mode 100644 index 00000000..0e95af50 --- /dev/null +++ b/script.cpp @@ -0,0 +1,1127 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" + +bool CheckSig(vector vchSig, vector vchPubKey, CScript scriptCode, const CTransaction& txTo, unsigned int nIn, int nHashType); + + + +typedef vector valtype; +static const valtype vchFalse(0); +static const valtype vchZero(0); +static const valtype vchTrue(1, 1); +static const CBigNum bnZero(0); +static const CBigNum bnOne(1); +static const CBigNum bnFalse(0); +static const CBigNum bnTrue(1); + + +bool CastToBool(const valtype& vch) +{ + return (CBigNum(vch) != bnZero); +} + +void MakeSameSize(valtype& vch1, valtype& vch2) +{ + // Lengthen the shorter one + if (vch1.size() < vch2.size()) + vch1.resize(vch2.size(), 0); + if (vch2.size() < vch1.size()) + vch2.resize(vch1.size(), 0); +} + + + +// +// Script is a stack machine (like Forth) that evaluates a predicate +// returning a bool indicating valid or not. There are no loops. +// +#define stacktop(i) (stack.at(stack.size()+(i))) +#define altstacktop(i) (altstack.at(altstack.size()+(i))) + +bool EvalScript(const CScript& script, const CTransaction& txTo, unsigned int nIn, int nHashType, + vector >* pvStackRet) +{ + CAutoBN_CTX pctx; + CScript::const_iterator pc = script.begin(); + CScript::const_iterator pend = script.end(); + CScript::const_iterator pbegincodehash = script.begin(); + vector vfExec; + vector stack; + vector altstack; + if (pvStackRet) + pvStackRet->clear(); + + + while (pc < pend) + { + bool fExec = !count(vfExec.begin(), vfExec.end(), false); + + // + // Read instruction + // + opcodetype opcode; + valtype vchPushValue; + if (!script.GetOp(pc, opcode, vchPushValue)) + return false; + + if (fExec && opcode <= OP_PUSHDATA4) + stack.push_back(vchPushValue); + else if (fExec || (OP_IF <= opcode && opcode <= OP_ENDIF)) + switch (opcode) + { + // + // Push value + // + case OP_1NEGATE: + case OP_1: + case OP_2: + case OP_3: + case OP_4: + case OP_5: + case OP_6: + case OP_7: + case OP_8: + case OP_9: + case OP_10: + case OP_11: + case OP_12: + case OP_13: + case OP_14: + case OP_15: + case OP_16: + { + // ( -- value) + CBigNum bn((int)opcode - (int)(OP_1 - 1)); + stack.push_back(bn.getvch()); + } + break; + + + // + // Control + // + case OP_NOP: + break; + + case OP_VER: + { + CBigNum bn(VERSION); + stack.push_back(bn.getvch()); + } + break; + + case OP_IF: + case OP_NOTIF: + case OP_VERIF: + case OP_VERNOTIF: + { + // if [statements] [else [statements]] endif + bool fValue = false; + if (fExec) + { + if (stack.size() < 1) + return false; + valtype& vch = stacktop(-1); + if (opcode == OP_VERIF || opcode == OP_VERNOTIF) + fValue = (CBigNum(VERSION) >= CBigNum(vch)); + else + fValue = CastToBool(vch); + if (opcode == OP_NOTIF || opcode == OP_VERNOTIF) + fValue = !fValue; + stack.pop_back(); + } + vfExec.push_back(fValue); + } + break; + + case OP_ELSE: + { + if (vfExec.empty()) + return false; + vfExec.back() = !vfExec.back(); + } + break; + + case OP_ENDIF: + { + if (vfExec.empty()) + return false; + vfExec.pop_back(); + } + break; + + case OP_VERIFY: + { + // (true -- ) or + // (false -- false) and return + if (stack.size() < 1) + return false; + bool fValue = CastToBool(stacktop(-1)); + if (fValue) + stack.pop_back(); + else + pc = pend; + } + break; + + case OP_RETURN: + { + pc = pend; + } + break; + + + // + // Stack ops + // + case OP_TOALTSTACK: + { + if (stack.size() < 1) + return false; + altstack.push_back(stacktop(-1)); + stack.pop_back(); + } + break; + + case OP_FROMALTSTACK: + { + if (altstack.size() < 1) + return false; + stack.push_back(altstacktop(-1)); + altstack.pop_back(); + } + break; + + case OP_2DROP: + { + // (x1 x2 -- ) + stack.pop_back(); + stack.pop_back(); + } + break; + + case OP_2DUP: + { + // (x1 x2 -- x1 x2 x1 x2) + if (stack.size() < 2) + return false; + valtype vch1 = stacktop(-2); + valtype vch2 = stacktop(-1); + stack.push_back(vch1); + stack.push_back(vch2); + } + break; + + case OP_3DUP: + { + // (x1 x2 x3 -- x1 x2 x3 x1 x2 x3) + if (stack.size() < 3) + return false; + valtype vch1 = stacktop(-3); + valtype vch2 = stacktop(-2); + valtype vch3 = stacktop(-1); + stack.push_back(vch1); + stack.push_back(vch2); + stack.push_back(vch3); + } + break; + + case OP_2OVER: + { + // (x1 x2 x3 x4 -- x1 x2 x3 x4 x1 x2) + if (stack.size() < 4) + return false; + valtype vch1 = stacktop(-4); + valtype vch2 = stacktop(-3); + stack.push_back(vch1); + stack.push_back(vch2); + } + break; + + case OP_2ROT: + { + // (x1 x2 x3 x4 x5 x6 -- x3 x4 x5 x6 x1 x2) + if (stack.size() < 6) + return false; + valtype vch1 = stacktop(-6); + valtype vch2 = stacktop(-5); + stack.erase(stack.end()-6, stack.end()-4); + stack.push_back(vch1); + stack.push_back(vch2); + } + break; + + case OP_2SWAP: + { + // (x1 x2 x3 x4 -- x3 x4 x1 x2) + if (stack.size() < 4) + return false; + swap(stacktop(-4), stacktop(-2)); + swap(stacktop(-3), stacktop(-1)); + } + break; + + case OP_IFDUP: + { + // (x - 0 | x x) + if (stack.size() < 1) + return false; + valtype vch = stacktop(-1); + if (CastToBool(vch)) + stack.push_back(vch); + } + break; + + case OP_DEPTH: + { + // -- stacksize + CBigNum bn(stack.size()); + stack.push_back(bn.getvch()); + } + break; + + case OP_DROP: + { + // (x -- ) + if (stack.size() < 1) + return false; + stack.pop_back(); + } + break; + + case OP_DUP: + { + // (x -- x x) + if (stack.size() < 1) + return false; + valtype vch = stacktop(-1); + stack.push_back(vch); + } + break; + + case OP_NIP: + { + // (x1 x2 -- x2) + if (stack.size() < 2) + return false; + stack.erase(stack.end() - 2); + } + break; + + case OP_OVER: + { + // (x1 x2 -- x1 x2 x1) + if (stack.size() < 2) + return false; + valtype vch = stacktop(-2); + stack.push_back(vch); + } + break; + + case OP_PICK: + case OP_ROLL: + { + // (xn ... x2 x1 x0 n - xn ... x2 x1 x0 xn) + // (xn ... x2 x1 x0 n - ... x2 x1 x0 xn) + if (stack.size() < 2) + return false; + int n = CBigNum(stacktop(-1)).getint(); + stack.pop_back(); + if (n < 0 || n >= stack.size()) + return false; + valtype vch = stacktop(-n-1); + if (opcode == OP_ROLL) + stack.erase(stack.end()-n-1); + stack.push_back(vch); + } + break; + + case OP_ROT: + { + // (x1 x2 x3 -- x2 x3 x1) + // x2 x1 x3 after first swap + // x2 x3 x1 after second swap + if (stack.size() < 3) + return false; + swap(stacktop(-3), stacktop(-2)); + swap(stacktop(-2), stacktop(-1)); + } + break; + + case OP_SWAP: + { + // (x1 x2 -- x2 x1) + if (stack.size() < 2) + return false; + swap(stacktop(-2), stacktop(-1)); + } + break; + + case OP_TUCK: + { + // (x1 x2 -- x2 x1 x2) + if (stack.size() < 2) + return false; + valtype vch = stacktop(-1); + stack.insert(stack.end()-2, vch); + } + break; + + + // + // Splice ops + // + case OP_CAT: + { + // (x1 x2 -- out) + if (stack.size() < 2) + return false; + valtype& vch1 = stacktop(-2); + valtype& vch2 = stacktop(-1); + vch1.insert(vch1.end(), vch2.begin(), vch2.end()); + stack.pop_back(); + } + break; + + case OP_SUBSTR: + { + // (in begin size -- out) + if (stack.size() < 3) + return false; + valtype& vch = stacktop(-3); + int nBegin = CBigNum(stacktop(-2)).getint(); + int nEnd = nBegin + CBigNum(stacktop(-1)).getint(); + if (nBegin < 0 || nEnd < nBegin) + return false; + if (nBegin > vch.size()) + nBegin = vch.size(); + if (nEnd > vch.size()) + nEnd = vch.size(); + vch.erase(vch.begin() + nEnd, vch.end()); + vch.erase(vch.begin(), vch.begin() + nBegin); + stack.pop_back(); + stack.pop_back(); + } + break; + + case OP_LEFT: + case OP_RIGHT: + { + // (in size -- out) + if (stack.size() < 2) + return false; + valtype& vch = stacktop(-2); + int nSize = CBigNum(stacktop(-1)).getint(); + if (nSize < 0) + return false; + if (nSize > vch.size()) + nSize = vch.size(); + if (opcode == OP_LEFT) + vch.erase(vch.begin() + nSize, vch.end()); + else + vch.erase(vch.begin(), vch.end() - nSize); + stack.pop_back(); + } + break; + + case OP_SIZE: + { + // (in -- in size) + if (stack.size() < 1) + return false; + CBigNum bn(stacktop(-1).size()); + stack.push_back(bn.getvch()); + } + break; + + + // + // Bitwise logic + // + case OP_INVERT: + { + // (in - out) + if (stack.size() < 1) + return false; + valtype& vch = stacktop(-1); + for (int i = 0; i < vch.size(); i++) + vch[i] = ~vch[i]; + } + break; + + case OP_AND: + case OP_OR: + case OP_XOR: + { + // (x1 x2 - out) + if (stack.size() < 2) + return false; + valtype& vch1 = stacktop(-2); + valtype& vch2 = stacktop(-1); + MakeSameSize(vch1, vch2); + if (opcode == OP_AND) + { + for (int i = 0; i < vch1.size(); i++) + vch1[i] &= vch2[i]; + } + else if (opcode == OP_OR) + { + for (int i = 0; i < vch1.size(); i++) + vch1[i] |= vch2[i]; + } + else if (opcode == OP_XOR) + { + for (int i = 0; i < vch1.size(); i++) + vch1[i] ^= vch2[i]; + } + stack.pop_back(); + } + break; + + case OP_EQUAL: + case OP_EQUALVERIFY: + //case OP_NOTEQUAL: // use OP_NUMNOTEQUAL + { + // (x1 x2 - bool) + if (stack.size() < 2) + return false; + valtype& vch1 = stacktop(-2); + valtype& vch2 = stacktop(-1); + bool fEqual = (vch1 == vch2); + // OP_NOTEQUAL is disabled because it would be too easy to say + // something like n != 1 and have some wiseguy pass in 1 with extra + // zero bytes after it (numerically, 0x01 == 0x0001 == 0x000001) + //if (opcode == OP_NOTEQUAL) + // fEqual = !fEqual; + stack.pop_back(); + stack.pop_back(); + stack.push_back(fEqual ? vchTrue : vchFalse); + if (opcode == OP_EQUALVERIFY) + { + if (fEqual) + stack.pop_back(); + else + pc = pend; + } + } + break; + + + // + // Numeric + // + case OP_1ADD: + case OP_1SUB: + case OP_2MUL: + case OP_2DIV: + case OP_NEGATE: + case OP_ABS: + case OP_NOT: + case OP_0NOTEQUAL: + { + // (in -- out) + if (stack.size() < 1) + return false; + CBigNum bn(stacktop(-1)); + switch (opcode) + { + case OP_1ADD: bn += bnOne; break; + case OP_1SUB: bn -= bnOne; break; + case OP_2MUL: bn <<= 1; break; + case OP_2DIV: bn >>= 1; break; + case OP_NEGATE: bn = -bn; break; + case OP_ABS: if (bn < bnZero) bn = -bn; break; + case OP_NOT: bn = (bn == bnZero); break; + case OP_0NOTEQUAL: bn = (bn != bnZero); break; + } + stack.pop_back(); + stack.push_back(bn.getvch()); + } + break; + + case OP_ADD: + case OP_SUB: + case OP_MUL: + case OP_DIV: + case OP_MOD: + case OP_LSHIFT: + case OP_RSHIFT: + case OP_BOOLAND: + case OP_BOOLOR: + case OP_NUMEQUAL: + case OP_NUMEQUALVERIFY: + case OP_NUMNOTEQUAL: + case OP_LESSTHAN: + case OP_GREATERTHAN: + case OP_LESSTHANOREQUAL: + case OP_GREATERTHANOREQUAL: + case OP_MIN: + case OP_MAX: + { + // (x1 x2 -- out) + if (stack.size() < 2) + return false; + CBigNum bn1(stacktop(-2)); + CBigNum bn2(stacktop(-1)); + CBigNum bn; + switch (opcode) + { + case OP_ADD: + bn = bn1 + bn2; + break; + + case OP_SUB: + bn = bn1 - bn2; + break; + + case OP_MUL: + if (!BN_mul(&bn, &bn1, &bn2, pctx)) + return false; + break; + + case OP_DIV: + if (!BN_div(&bn, NULL, &bn1, &bn2, pctx)) + return false; + break; + + case OP_MOD: + if (!BN_mod(&bn, &bn1, &bn2, pctx)) + return false; + break; + + case OP_LSHIFT: + if (bn2 < bnZero) + return false; + bn = bn1 << bn2.getulong(); + break; + + case OP_RSHIFT: + if (bn2 < bnZero) + return false; + bn = bn1 >> bn2.getulong(); + break; + + case OP_BOOLAND: bn = (bn1 != bnZero && bn2 != bnZero); break; + case OP_BOOLOR: bn = (bn1 != bnZero || bn2 != bnZero); break; + case OP_NUMEQUAL: bn = (bn1 == bn2); break; + case OP_NUMEQUALVERIFY: bn = (bn1 == bn2); break; + case OP_NUMNOTEQUAL: bn = (bn1 != bn2); break; + case OP_LESSTHAN: bn = (bn1 < bn2); break; + case OP_GREATERTHAN: bn = (bn1 > bn2); break; + case OP_LESSTHANOREQUAL: bn = (bn1 <= bn2); break; + case OP_GREATERTHANOREQUAL: bn = (bn1 >= bn2); break; + case OP_MIN: bn = (bn1 < bn2 ? bn1 : bn2); break; + case OP_MAX: bn = (bn1 > bn2 ? bn1 : bn2); break; + } + stack.pop_back(); + stack.pop_back(); + stack.push_back(bn.getvch()); + + if (opcode == OP_NUMEQUALVERIFY) + { + if (CastToBool(stacktop(-1))) + stack.pop_back(); + else + pc = pend; + } + } + break; + + case OP_WITHIN: + { + // (x min max -- out) + if (stack.size() < 3) + return false; + CBigNum bn1(stacktop(-3)); + CBigNum bn2(stacktop(-2)); + CBigNum bn3(stacktop(-1)); + bool fValue = (bn2 <= bn1 && bn1 < bn3); + stack.pop_back(); + stack.pop_back(); + stack.pop_back(); + stack.push_back(fValue ? vchTrue : vchFalse); + } + break; + + + // + // Crypto + // + case OP_RIPEMD160: + case OP_SHA1: + case OP_SHA256: + case OP_HASH160: + case OP_HASH256: + { + // (in -- hash) + if (stack.size() < 1) + return false; + valtype& vch = stacktop(-1); + valtype vchHash(opcode == OP_RIPEMD160 || opcode == OP_SHA1 || opcode == OP_HASH160 ? 20 : 32); + if (opcode == OP_RIPEMD160) + RIPEMD160(&vch[0], vch.size(), &vchHash[0]); + else if (opcode == OP_SHA1) + SHA1(&vch[0], vch.size(), &vchHash[0]); + else if (opcode == OP_SHA256) + SHA256(&vch[0], vch.size(), &vchHash[0]); + else if (opcode == OP_HASH160) + { + uint160 hash160 = Hash160(vch); + memcpy(&vchHash[0], &hash160, sizeof(hash160)); + } + else if (opcode == OP_HASH256) + { + uint256 hash = Hash(vch.begin(), vch.end()); + memcpy(&vchHash[0], &hash, sizeof(hash)); + } + stack.pop_back(); + stack.push_back(vchHash); + } + break; + + case OP_CODESEPARATOR: + { + // Hash starts after the code separator + pbegincodehash = pc; + } + break; + + case OP_CHECKSIG: + case OP_CHECKSIGVERIFY: + { + // (sig pubkey -- bool) + if (stack.size() < 2) + return false; + + valtype& vchSig = stacktop(-2); + valtype& vchPubKey = stacktop(-1); + + ////// debug print + //PrintHex(vchSig.begin(), vchSig.end(), "sig: %s\n"); + //PrintHex(vchPubKey.begin(), vchPubKey.end(), "pubkey: %s\n"); + + // Subset of script starting at the most recent codeseparator + CScript scriptCode(pbegincodehash, pend); + + // Drop the signature, since there's no way for a signature to sign itself + scriptCode.FindAndDelete(CScript(vchSig)); + + bool fSuccess = CheckSig(vchSig, vchPubKey, scriptCode, txTo, nIn, nHashType); + + stack.pop_back(); + stack.pop_back(); + stack.push_back(fSuccess ? vchTrue : vchFalse); + if (opcode == OP_CHECKSIGVERIFY) + { + if (fSuccess) + stack.pop_back(); + else + pc = pend; + } + } + break; + + case OP_CHECKMULTISIG: + case OP_CHECKMULTISIGVERIFY: + { + // ([sig ...] num_of_signatures [pubkey ...] num_of_pubkeys -- bool) + + int i = 1; + if (stack.size() < i) + return false; + + int nKeysCount = CBigNum(stacktop(-i)).getint(); + if (nKeysCount < 0) + return false; + int ikey = ++i; + i += nKeysCount; + if (stack.size() < i) + return false; + + int nSigsCount = CBigNum(stacktop(-i)).getint(); + if (nSigsCount < 0 || nSigsCount > nKeysCount) + return false; + int isig = ++i; + i += nSigsCount; + if (stack.size() < i) + return false; + + // Subset of script starting at the most recent codeseparator + CScript scriptCode(pbegincodehash, pend); + + // Drop the signatures, since there's no way for a signature to sign itself + for (int i = 0; i < nSigsCount; i++) + { + valtype& vchSig = stacktop(-isig-i); + scriptCode.FindAndDelete(CScript(vchSig)); + } + + bool fSuccess = true; + while (fSuccess && nSigsCount > 0) + { + valtype& vchSig = stacktop(-isig); + valtype& vchPubKey = stacktop(-ikey); + + // Check signature + if (CheckSig(vchSig, vchPubKey, scriptCode, txTo, nIn, nHashType)) + { + isig++; + nSigsCount--; + } + ikey++; + nKeysCount--; + + // If there are more signatures left than keys left, + // then too many signatures have failed + if (nSigsCount > nKeysCount) + fSuccess = false; + } + + while (i-- > 0) + stack.pop_back(); + stack.push_back(fSuccess ? vchTrue : vchFalse); + + if (opcode == OP_CHECKMULTISIGVERIFY) + { + if (fSuccess) + stack.pop_back(); + else + pc = pend; + } + } + break; + + default: + return false; + } + } + + + if (pvStackRet) + *pvStackRet = stack; + return (stack.empty() ? false : CastToBool(stack.back())); +} + +#undef top + + + + + + + + + +uint256 SignatureHash(CScript scriptCode, const CTransaction& txTo, unsigned int nIn, int nHashType) +{ + if (nIn >= txTo.vin.size()) + { + printf("ERROR: SignatureHash() : nIn=%d out of range\n", nIn); + return 1; + } + CTransaction txTmp(txTo); + + // In case concatenating two scripts ends up with two codeseparators, + // or an extra one at the end, this prevents all those possible incompatibilities. + scriptCode.FindAndDelete(CScript(OP_CODESEPARATOR)); + + // Blank out other inputs' signatures + for (int i = 0; i < txTmp.vin.size(); i++) + txTmp.vin[i].scriptSig = CScript(); + txTmp.vin[nIn].scriptSig = scriptCode; + + // Blank out some of the outputs + if ((nHashType & 0x1f) == SIGHASH_NONE) + { + // Wildcard payee + txTmp.vout.clear(); + + // Let the others update at will + for (int i = 0; i < txTmp.vin.size(); i++) + if (i != nIn) + txTmp.vin[i].nSequence = 0; + } + else if ((nHashType & 0x1f) == SIGHASH_SINGLE) + { + // Only lockin the txout payee at same index as txin + unsigned int nOut = nIn; + if (nOut >= txTmp.vout.size()) + { + printf("ERROR: SignatureHash() : nOut=%d out of range\n", nOut); + return 1; + } + txTmp.vout.resize(nOut+1); + for (int i = 0; i < nOut; i++) + txTmp.vout[i].SetNull(); + + // Let the others update at will + for (int i = 0; i < txTmp.vin.size(); i++) + if (i != nIn) + txTmp.vin[i].nSequence = 0; + } + + // Blank out other inputs completely, not recommended for open transactions + if (nHashType & SIGHASH_ANYONECANPAY) + { + txTmp.vin[0] = txTmp.vin[nIn]; + txTmp.vin.resize(1); + } + + // Serialize and hash + CDataStream ss(SER_GETHASH); + ss.reserve(10000); + ss << txTmp << nHashType; + return Hash(ss.begin(), ss.end()); +} + + +bool CheckSig(vector vchSig, vector vchPubKey, CScript scriptCode, + const CTransaction& txTo, unsigned int nIn, int nHashType) +{ + CKey key; + if (!key.SetPubKey(vchPubKey)) + return false; + + // Hash type is one byte tacked on to the end of the signature + if (vchSig.empty()) + return false; + if (nHashType == 0) + nHashType = vchSig.back(); + else if (nHashType != vchSig.back()) + return false; + vchSig.pop_back(); + + if (key.Verify(SignatureHash(scriptCode, txTo, nIn, nHashType), vchSig)) + return true; + + return false; +} + + + + + + + + + + + +bool Solver(const CScript& scriptPubKey, vector >& vSolutionRet) +{ + // Templates + static vector vTemplates; + if (vTemplates.empty()) + { + // Standard tx, sender provides pubkey, receiver adds signature + vTemplates.push_back(CScript() << OP_PUBKEY << OP_CHECKSIG); + + // Short account number tx, sender provides hash of pubkey, receiver provides signature and pubkey + vTemplates.push_back(CScript() << OP_DUP << OP_HASH160 << OP_PUBKEYHASH << OP_EQUALVERIFY << OP_CHECKSIG); + } + + // Scan templates + const CScript& script1 = scriptPubKey; + foreach(const CScript& script2, vTemplates) + { + vSolutionRet.clear(); + opcodetype opcode1, opcode2; + vector vch1, vch2; + + // Compare + CScript::const_iterator pc1 = script1.begin(); + CScript::const_iterator pc2 = script2.begin(); + loop + { + bool f1 = script1.GetOp(pc1, opcode1, vch1); + bool f2 = script2.GetOp(pc2, opcode2, vch2); + if (!f1 && !f2) + { + // Success + reverse(vSolutionRet.begin(), vSolutionRet.end()); + return true; + } + else if (f1 != f2) + { + break; + } + else if (opcode2 == OP_PUBKEY) + { + if (vch1.size() <= sizeof(uint256)) + break; + vSolutionRet.push_back(make_pair(opcode2, vch1)); + } + else if (opcode2 == OP_PUBKEYHASH) + { + if (vch1.size() != sizeof(uint160)) + break; + vSolutionRet.push_back(make_pair(opcode2, vch1)); + } + else if (opcode1 != opcode2) + { + break; + } + } + } + + vSolutionRet.clear(); + return false; +} + + +bool Solver(const CScript& scriptPubKey, uint256 hash, int nHashType, CScript& scriptSigRet) +{ + scriptSigRet.clear(); + + vector > vSolution; + if (!Solver(scriptPubKey, vSolution)) + return false; + + // Compile solution + CRITICAL_BLOCK(cs_mapKeys) + { + foreach(PAIRTYPE(opcodetype, valtype)& item, vSolution) + { + if (item.first == OP_PUBKEY) + { + // Sign + const valtype& vchPubKey = item.second; + if (!mapKeys.count(vchPubKey)) + return false; + if (hash != 0) + { + vector vchSig; + if (!CKey::Sign(mapKeys[vchPubKey], hash, vchSig)) + return false; + vchSig.push_back((unsigned char)nHashType); + scriptSigRet << vchSig; + } + } + else if (item.first == OP_PUBKEYHASH) + { + // Sign and give pubkey + map::iterator mi = mapPubKeys.find(uint160(item.second)); + if (mi == mapPubKeys.end()) + return false; + const vector& vchPubKey = (*mi).second; + if (!mapKeys.count(vchPubKey)) + return false; + if (hash != 0) + { + vector vchSig; + if (!CKey::Sign(mapKeys[vchPubKey], hash, vchSig)) + return false; + vchSig.push_back((unsigned char)nHashType); + scriptSigRet << vchSig << vchPubKey; + } + } + } + } + + return true; +} + + +bool IsMine(const CScript& scriptPubKey) +{ + CScript scriptSig; + return Solver(scriptPubKey, 0, 0, scriptSig); +} + + +bool ExtractPubKey(const CScript& scriptPubKey, bool fMineOnly, vector& vchPubKeyRet) +{ + vchPubKeyRet.clear(); + + vector > vSolution; + if (!Solver(scriptPubKey, vSolution)) + return false; + + CRITICAL_BLOCK(cs_mapKeys) + { + foreach(PAIRTYPE(opcodetype, valtype)& item, vSolution) + { + valtype vchPubKey; + if (item.first == OP_PUBKEY) + { + vchPubKey = item.second; + } + else if (item.first == OP_PUBKEYHASH) + { + map::iterator mi = mapPubKeys.find(uint160(item.second)); + if (mi == mapPubKeys.end()) + continue; + vchPubKey = (*mi).second; + } + if (!fMineOnly || mapKeys.count(vchPubKey)) + { + vchPubKeyRet = vchPubKey; + return true; + } + } + } + return false; +} + + +bool ExtractHash160(const CScript& scriptPubKey, uint160& hash160Ret) +{ + hash160Ret = 0; + + vector > vSolution; + if (!Solver(scriptPubKey, vSolution)) + return false; + + foreach(PAIRTYPE(opcodetype, valtype)& item, vSolution) + { + if (item.first == OP_PUBKEYHASH) + { + hash160Ret = uint160(item.second); + return true; + } + } + return false; +} + + +bool SignSignature(const CTransaction& txFrom, CTransaction& txTo, unsigned int nIn, int nHashType, CScript scriptPrereq) +{ + assert(nIn < txTo.vin.size()); + CTxIn& txin = txTo.vin[nIn]; + assert(txin.prevout.n < txFrom.vout.size()); + const CTxOut& txout = txFrom.vout[txin.prevout.n]; + + // Leave out the signature from the hash, since a signature can't sign itself. + // The checksig op will also drop the signatures from its hash. + uint256 hash = SignatureHash(scriptPrereq + txout.scriptPubKey, txTo, nIn, nHashType); + + if (!Solver(txout.scriptPubKey, hash, nHashType, txin.scriptSig)) + return false; + + txin.scriptSig = scriptPrereq + txin.scriptSig; + + // Test solution + if (scriptPrereq.empty()) + if (!EvalScript(txin.scriptSig + CScript(OP_CODESEPARATOR) + txout.scriptPubKey, txTo, nIn)) + return false; + + return true; +} + + +bool VerifySignature(const CTransaction& txFrom, const CTransaction& txTo, unsigned int nIn, int nHashType) +{ + assert(nIn < txTo.vin.size()); + const CTxIn& txin = txTo.vin[nIn]; + if (txin.prevout.n >= txFrom.vout.size()) + return false; + const CTxOut& txout = txFrom.vout[txin.prevout.n]; + + if (txin.prevout.hash != txFrom.GetHash()) + return false; + + return EvalScript(txin.scriptSig + CScript(OP_CODESEPARATOR) + txout.scriptPubKey, txTo, nIn, nHashType); +} diff --git a/script.h b/script.h new file mode 100644 index 00000000..0d977734 --- /dev/null +++ b/script.h @@ -0,0 +1,597 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +class CTransaction; + +enum +{ + SIGHASH_ALL = 1, + SIGHASH_NONE = 2, + SIGHASH_SINGLE = 3, + SIGHASH_ANYONECANPAY = 0x80, +}; + + + +enum opcodetype +{ + // push value + OP_0=0, + OP_FALSE=OP_0, + OP_PUSHDATA1=76, + OP_PUSHDATA2, + OP_PUSHDATA4, + OP_1NEGATE, + OP_RESERVED, + OP_1, + OP_TRUE=OP_1, + OP_2, + OP_3, + OP_4, + OP_5, + OP_6, + OP_7, + OP_8, + OP_9, + OP_10, + OP_11, + OP_12, + OP_13, + OP_14, + OP_15, + OP_16, + + // control + OP_NOP, + OP_VER, + OP_IF, + OP_NOTIF, + OP_VERIF, + OP_VERNOTIF, + OP_ELSE, + OP_ENDIF, + OP_VERIFY, + OP_RETURN, + + // stack ops + OP_TOALTSTACK, + OP_FROMALTSTACK, + OP_2DROP, + OP_2DUP, + OP_3DUP, + OP_2OVER, + OP_2ROT, + OP_2SWAP, + OP_IFDUP, + OP_DEPTH, + OP_DROP, + OP_DUP, + OP_NIP, + OP_OVER, + OP_PICK, + OP_ROLL, + OP_ROT, + OP_SWAP, + OP_TUCK, + + // splice ops + OP_CAT, + OP_SUBSTR, + OP_LEFT, + OP_RIGHT, + OP_SIZE, + + // bit logic + OP_INVERT, + OP_AND, + OP_OR, + OP_XOR, + OP_EQUAL, + OP_EQUALVERIFY, + OP_RESERVED1, + OP_RESERVED2, + + // numeric + OP_1ADD, + OP_1SUB, + OP_2MUL, + OP_2DIV, + OP_NEGATE, + OP_ABS, + OP_NOT, + OP_0NOTEQUAL, + + OP_ADD, + OP_SUB, + OP_MUL, + OP_DIV, + OP_MOD, + OP_LSHIFT, + OP_RSHIFT, + + OP_BOOLAND, + OP_BOOLOR, + OP_NUMEQUAL, + OP_NUMEQUALVERIFY, + OP_NUMNOTEQUAL, + OP_LESSTHAN, + OP_GREATERTHAN, + OP_LESSTHANOREQUAL, + OP_GREATERTHANOREQUAL, + OP_MIN, + OP_MAX, + + OP_WITHIN, + + // crypto + OP_RIPEMD160, + OP_SHA1, + OP_SHA256, + OP_HASH160, + OP_HASH256, + OP_CODESEPARATOR, + OP_CHECKSIG, + OP_CHECKSIGVERIFY, + OP_CHECKMULTISIG, + OP_CHECKMULTISIGVERIFY, + + + // multi-byte opcodes + OP_SINGLEBYTE_END = 0xF0, + OP_DOUBLEBYTE_BEGIN = 0xF000, + + // template matching params + OP_PUBKEY, + OP_PUBKEYHASH, + + + + OP_INVALIDOPCODE = 0xFFFF, +}; + + + + + + + + +inline const char* GetOpName(opcodetype opcode) +{ + switch (opcode) + { + // push value + case OP_0 : return "0"; + case OP_PUSHDATA1 : return "OP_PUSHDATA1"; + case OP_PUSHDATA2 : return "OP_PUSHDATA2"; + case OP_PUSHDATA4 : return "OP_PUSHDATA4"; + case OP_1NEGATE : return "-1"; + case OP_RESERVED : return "OP_RESERVED"; + case OP_1 : return "1"; + case OP_2 : return "2"; + case OP_3 : return "3"; + case OP_4 : return "4"; + case OP_5 : return "5"; + case OP_6 : return "6"; + case OP_7 : return "7"; + case OP_8 : return "8"; + case OP_9 : return "9"; + case OP_10 : return "10"; + case OP_11 : return "11"; + case OP_12 : return "12"; + case OP_13 : return "13"; + case OP_14 : return "14"; + case OP_15 : return "15"; + case OP_16 : return "16"; + + // control + case OP_NOP : return "OP_NOP"; + case OP_VER : return "OP_VER"; + case OP_IF : return "OP_IF"; + case OP_NOTIF : return "OP_NOTIF"; + case OP_VERIF : return "OP_VERIF"; + case OP_VERNOTIF : return "OP_VERNOTIF"; + case OP_ELSE : return "OP_ELSE"; + case OP_ENDIF : return "OP_ENDIF"; + case OP_VERIFY : return "OP_VERIFY"; + case OP_RETURN : return "OP_RETURN"; + + // stack ops + case OP_TOALTSTACK : return "OP_TOALTSTACK"; + case OP_FROMALTSTACK : return "OP_FROMALTSTACK"; + case OP_2DROP : return "OP_2DROP"; + case OP_2DUP : return "OP_2DUP"; + case OP_3DUP : return "OP_3DUP"; + case OP_2OVER : return "OP_2OVER"; + case OP_2ROT : return "OP_2ROT"; + case OP_2SWAP : return "OP_2SWAP"; + case OP_IFDUP : return "OP_IFDUP"; + case OP_DEPTH : return "OP_DEPTH"; + case OP_DROP : return "OP_DROP"; + case OP_DUP : return "OP_DUP"; + case OP_NIP : return "OP_NIP"; + case OP_OVER : return "OP_OVER"; + case OP_PICK : return "OP_PICK"; + case OP_ROLL : return "OP_ROLL"; + case OP_ROT : return "OP_ROT"; + case OP_SWAP : return "OP_SWAP"; + case OP_TUCK : return "OP_TUCK"; + + // splice ops + case OP_CAT : return "OP_CAT"; + case OP_SUBSTR : return "OP_SUBSTR"; + case OP_LEFT : return "OP_LEFT"; + case OP_RIGHT : return "OP_RIGHT"; + case OP_SIZE : return "OP_SIZE"; + + // bit logic + case OP_INVERT : return "OP_INVERT"; + case OP_AND : return "OP_AND"; + case OP_OR : return "OP_OR"; + case OP_XOR : return "OP_XOR"; + case OP_EQUAL : return "OP_EQUAL"; + case OP_EQUALVERIFY : return "OP_EQUALVERIFY"; + case OP_RESERVED1 : return "OP_RESERVED1"; + case OP_RESERVED2 : return "OP_RESERVED2"; + + // numeric + case OP_1ADD : return "OP_1ADD"; + case OP_1SUB : return "OP_1SUB"; + case OP_2MUL : return "OP_2MUL"; + case OP_2DIV : return "OP_2DIV"; + case OP_NEGATE : return "OP_NEGATE"; + case OP_ABS : return "OP_ABS"; + case OP_NOT : return "OP_NOT"; + case OP_0NOTEQUAL : return "OP_0NOTEQUAL"; + case OP_ADD : return "OP_ADD"; + case OP_SUB : return "OP_SUB"; + case OP_MUL : return "OP_MUL"; + case OP_DIV : return "OP_DIV"; + case OP_MOD : return "OP_MOD"; + case OP_LSHIFT : return "OP_LSHIFT"; + case OP_RSHIFT : return "OP_RSHIFT"; + case OP_BOOLAND : return "OP_BOOLAND"; + case OP_BOOLOR : return "OP_BOOLOR"; + case OP_NUMEQUAL : return "OP_NUMEQUAL"; + case OP_NUMEQUALVERIFY : return "OP_NUMEQUALVERIFY"; + case OP_NUMNOTEQUAL : return "OP_NUMNOTEQUAL"; + case OP_LESSTHAN : return "OP_LESSTHAN"; + case OP_GREATERTHAN : return "OP_GREATERTHAN"; + case OP_LESSTHANOREQUAL : return "OP_LESSTHANOREQUAL"; + case OP_GREATERTHANOREQUAL : return "OP_GREATERTHANOREQUAL"; + case OP_MIN : return "OP_MIN"; + case OP_MAX : return "OP_MAX"; + case OP_WITHIN : return "OP_WITHIN"; + + // crypto + case OP_RIPEMD160 : return "OP_RIPEMD160"; + case OP_SHA1 : return "OP_SHA1"; + case OP_SHA256 : return "OP_SHA256"; + case OP_HASH160 : return "OP_HASH160"; + case OP_HASH256 : return "OP_HASH256"; + case OP_CODESEPARATOR : return "OP_CODESEPARATOR"; + case OP_CHECKSIG : return "OP_CHECKSIG"; + case OP_CHECKSIGVERIFY : return "OP_CHECKSIGVERIFY"; + case OP_CHECKMULTISIG : return "OP_CHECKMULTISIG"; + case OP_CHECKMULTISIGVERIFY : return "OP_CHECKMULTISIGVERIFY"; + + + + // multi-byte opcodes + case OP_SINGLEBYTE_END : return "OP_SINGLEBYTE_END"; + case OP_DOUBLEBYTE_BEGIN : return "OP_DOUBLEBYTE_BEGIN"; + case OP_PUBKEY : return "OP_PUBKEY"; + case OP_PUBKEYHASH : return "OP_PUBKEYHASH"; + + + + case OP_INVALIDOPCODE : return "OP_INVALIDOPCODE"; + default: + return "UNKNOWN_OPCODE"; + } +}; + + + + +inline string ValueString(const vector& vch) +{ + if (vch.size() <= 4) + return strprintf("%d", CBigNum(vch).getint()); + else + return HexNumStr(vch.begin(), vch.end()); + //return string("(") + HexStr(vch.begin(), vch.end()) + string(")"); +} + +inline string StackString(const vector >& vStack) +{ + string str; + foreach(const vector& vch, vStack) + { + if (!str.empty()) + str += " "; + str += ValueString(vch); + } + return str; +} + + + + + + + + + +class CScript : public vector +{ +protected: + CScript& push_int64(int64 n) + { + if (n == -1 || (n >= 1 && n <= 16)) + { + push_back(n + (OP_1 - 1)); + } + else + { + CBigNum bn(n); + *this << bn.getvch(); + } + return (*this); + } + + CScript& push_uint64(uint64 n) + { + if (n == -1 || (n >= 1 && n <= 16)) + { + push_back(n + (OP_1 - 1)); + } + else + { + CBigNum bn(n); + *this << bn.getvch(); + } + return (*this); + } + +public: + CScript() { } + CScript(const CScript& b) : vector(b.begin(), b.end()) { } + CScript(const_iterator pbegin, const_iterator pend) : vector(pbegin, pend) { } +#ifndef _MSC_VER + CScript(const unsigned char* pbegin, const unsigned char* pend) : vector(pbegin, pend) { } +#endif + + CScript& operator+=(const CScript& b) + { + insert(end(), b.begin(), b.end()); + return *this; + } + + friend CScript operator+(const CScript& a, const CScript& b) + { + CScript ret = a; + ret += b; + return (ret); + } + + + explicit CScript(char b) { operator<<(b); } + explicit CScript(short b) { operator<<(b); } + explicit CScript(int b) { operator<<(b); } + explicit CScript(long b) { operator<<(b); } + explicit CScript(int64 b) { operator<<(b); } + explicit CScript(unsigned char b) { operator<<(b); } + explicit CScript(unsigned int b) { operator<<(b); } + explicit CScript(unsigned short b) { operator<<(b); } + explicit CScript(unsigned long b) { operator<<(b); } + explicit CScript(uint64 b) { operator<<(b); } + + explicit CScript(opcodetype b) { operator<<(b); } + explicit CScript(const uint256& b) { operator<<(b); } + explicit CScript(const CBigNum& b) { operator<<(b); } + explicit CScript(const vector& b) { operator<<(b); } + + + CScript& operator<<(char b) { return (push_int64(b)); } + CScript& operator<<(short b) { return (push_int64(b)); } + CScript& operator<<(int b) { return (push_int64(b)); } + CScript& operator<<(long b) { return (push_int64(b)); } + CScript& operator<<(int64 b) { return (push_int64(b)); } + CScript& operator<<(unsigned char b) { return (push_uint64(b)); } + CScript& operator<<(unsigned int b) { return (push_uint64(b)); } + CScript& operator<<(unsigned short b) { return (push_uint64(b)); } + CScript& operator<<(unsigned long b) { return (push_uint64(b)); } + CScript& operator<<(uint64 b) { return (push_uint64(b)); } + + CScript& operator<<(opcodetype opcode) + { + if (opcode <= OP_SINGLEBYTE_END) + { + insert(end(), (unsigned char)opcode); + } + else + { + assert(opcode >= OP_DOUBLEBYTE_BEGIN); + insert(end(), (unsigned char)(opcode >> 8)); + insert(end(), (unsigned char)(opcode & 0xFF)); + } + return (*this); + } + + CScript& operator<<(const uint160& b) + { + insert(end(), sizeof(b)); + insert(end(), (unsigned char*)&b, (unsigned char*)&b + sizeof(b)); + return (*this); + } + + CScript& operator<<(const uint256& b) + { + insert(end(), sizeof(b)); + insert(end(), (unsigned char*)&b, (unsigned char*)&b + sizeof(b)); + return (*this); + } + + CScript& operator<<(const CBigNum& b) + { + *this << b.getvch(); + return (*this); + } + + CScript& operator<<(const vector& b) + { + if (b.size() < OP_PUSHDATA1) + { + insert(end(), (unsigned char)b.size()); + } + else if (b.size() <= 0xff) + { + insert(end(), OP_PUSHDATA1); + insert(end(), (unsigned char)b.size()); + } + else + { + insert(end(), OP_PUSHDATA2); + unsigned short nSize = b.size(); + insert(end(), (unsigned char*)&nSize, (unsigned char*)&nSize + sizeof(nSize)); + } + insert(end(), b.begin(), b.end()); + return (*this); + } + + CScript& operator<<(const CScript& b) + { + // I'm not sure if this should push the script or concatenate scripts. + // If there's ever a use for pushing a script onto a script, delete this member fn + assert(("warning: pushing a CScript onto a CScript with << is probably not intended, use + to concatenate", false)); + return (*this); + } + + + bool GetOp(iterator& pc, opcodetype& opcodeRet, vector& vchRet) + { + // This is why people hate C++ + const_iterator pc2 = pc; + bool fRet = GetOp(pc2, opcodeRet, vchRet); + pc = begin() + (pc2 - begin()); + return fRet; + } + + bool GetOp(const_iterator& pc, opcodetype& opcodeRet, vector& vchRet) const + { + opcodeRet = OP_INVALIDOPCODE; + vchRet.clear(); + if (pc >= end()) + return false; + + // Read instruction + unsigned int opcode = *pc++; + if (opcode >= OP_SINGLEBYTE_END) + { + if (pc + 1 > end()) + return false; + opcode <<= 8; + opcode |= *pc++; + } + + // Immediate operand + if (opcode <= OP_PUSHDATA4) + { + unsigned int nSize = opcode; + if (opcode == OP_PUSHDATA1) + { + if (pc + 1 > end()) + return false; + nSize = *pc++; + } + else if (opcode == OP_PUSHDATA2) + { + if (pc + 2 > end()) + return false; + nSize = 0; + memcpy(&nSize, &pc[0], 2); + pc += 2; + } + else if (opcode == OP_PUSHDATA4) + { + if (pc + 4 > end()) + return false; + memcpy(&nSize, &pc[0], 4); + pc += 4; + } + if (pc + nSize > end()) + return false; + vchRet.assign(pc, pc + nSize); + pc += nSize; + } + + opcodeRet = (opcodetype)opcode; + return true; + } + + + void FindAndDelete(const CScript& b) + { + iterator pc = begin(); + opcodetype opcode; + vector vchPushValue; + int count = 0; + do + { + while (end() - pc >= b.size() && memcmp(&pc[0], &b[0], b.size()) == 0) + { + erase(pc, pc + b.size()); + count++; + } + } + while (GetOp(pc, opcode, vchPushValue)); + //printf("FindAndDeleted deleted %d items\n", count); /// debug + } + + + void PrintHex() const + { + printf("CScript(%s)\n", HexStr(begin(), end()).c_str()); + } + + string ToString() const + { + string str; + opcodetype opcode; + vector vch; + const_iterator it = begin(); + while (GetOp(it, opcode, vch)) + { + if (!str.empty()) + str += " "; + if (opcode <= OP_PUSHDATA4) + str += ValueString(vch); + else + str += GetOpName(opcode); + } + return str; + } + + void print() const + { + printf("%s\n", ToString().c_str()); + } +}; + + + + + + + + +bool EvalScript(const CScript& script, const CTransaction& txTo, unsigned int nIn, int nHashType=0, + vector >* pvStackRet=NULL); +uint256 SignatureHash(CScript scriptCode, const CTransaction& txTo, unsigned int nIn, int nHashType); +bool IsMine(const CScript& scriptPubKey); +bool ExtractPubKey(const CScript& scriptPubKey, bool fMineOnly, vector& vchPubKeyRet); +bool ExtractHash160(const CScript& scriptPubKey, uint160& hash160Ret); +bool SignSignature(const CTransaction& txFrom, CTransaction& txTo, unsigned int nIn, int nHashType=SIGHASH_ALL, CScript scriptPrereq=CScript()); +bool VerifySignature(const CTransaction& txFrom, const CTransaction& txTo, unsigned int nIn, int nHashType=0); diff --git a/serialize.h b/serialize.h new file mode 100644 index 00000000..b7ab86d2 --- /dev/null +++ b/serialize.h @@ -0,0 +1,1158 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include +#include +#include +#if defined(_MSC_VER) || defined(__BORLANDC__) +typedef __int64 int64; +typedef unsigned __int64 uint64; +#else +typedef long long int64; +typedef unsigned long long uint64; +#endif +#if defined(_MSC_VER) && _MSC_VER < 1300 +#define for if (false) ; else for +#endif +class CScript; +class CDataStream; +class CAutoFile; + +static const int VERSION = 105; + + + + + +///////////////////////////////////////////////////////////////// +// +// Templates for serializing to anything that looks like a stream, +// i.e. anything that supports .read(char*, int) and .write(char*, int) +// + +enum +{ + // primary actions + SER_NETWORK = (1 << 0), + SER_DISK = (1 << 1), + SER_GETHASH = (1 << 2), + + // modifiers + SER_SKIPSIG = (1 << 16), + SER_BLOCKHEADERONLY = (1 << 17), +}; + +#define IMPLEMENT_SERIALIZE(statements) \ + unsigned int GetSerializeSize(int nType=0, int nVersion=VERSION) const \ + { \ + CSerActionGetSerializeSize ser_action; \ + const bool fGetSize = true; \ + const bool fWrite = false; \ + const bool fRead = false; \ + unsigned int nSerSize = 0; \ + ser_streamplaceholder s; \ + s.nType = nType; \ + s.nVersion = nVersion; \ + {statements} \ + return nSerSize; \ + } \ + template \ + void Serialize(Stream& s, int nType=0, int nVersion=VERSION) const \ + { \ + CSerActionSerialize ser_action; \ + const bool fGetSize = false; \ + const bool fWrite = true; \ + const bool fRead = false; \ + unsigned int nSerSize = 0; \ + {statements} \ + } \ + template \ + void Unserialize(Stream& s, int nType=0, int nVersion=VERSION) \ + { \ + CSerActionUnserialize ser_action; \ + const bool fGetSize = false; \ + const bool fWrite = false; \ + const bool fRead = true; \ + unsigned int nSerSize = 0; \ + {statements} \ + } + +#define READWRITE(obj) (nSerSize += ::SerReadWrite(s, (obj), nType, nVersion, ser_action)) + + + + + + +// +// Basic types +// +#define WRITEDATA(s, obj) s.write((char*)&(obj), sizeof(obj)) +#define READDATA(s, obj) s.read((char*)&(obj), sizeof(obj)) + +inline unsigned int GetSerializeSize(char a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(signed char a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(unsigned char a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(signed short a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(unsigned short a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(signed int a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(unsigned int a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(signed long a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(unsigned long a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(int64 a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(uint64 a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(float a, int, int=0) { return sizeof(a); } +inline unsigned int GetSerializeSize(double a, int, int=0) { return sizeof(a); } + +template inline void Serialize(Stream& s, char a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, signed char a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, unsigned char a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, signed short a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, unsigned short a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, signed int a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, unsigned int a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, signed long a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, unsigned long a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, int64 a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, uint64 a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, float a, int, int=0) { WRITEDATA(s, a); } +template inline void Serialize(Stream& s, double a, int, int=0) { WRITEDATA(s, a); } + +template inline void Unserialize(Stream& s, char& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, signed char& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, unsigned char& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, signed short& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, unsigned short& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, signed int& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, unsigned int& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, signed long& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, unsigned long& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, int64& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, uint64& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, float& a, int, int=0) { READDATA(s, a); } +template inline void Unserialize(Stream& s, double& a, int, int=0) { READDATA(s, a); } + +inline unsigned int GetSerializeSize(bool a, int, int=0) { return sizeof(char); } +template inline void Serialize(Stream& s, bool a, int, int=0) { char f=a; WRITEDATA(s, f); } +template inline void Unserialize(Stream& s, bool& a, int, int=0) { char f; READDATA(s, f); a=f; } + + + + + + +// +// Compact size +// size < 253 -- 1 byte +// size <= USHRT_MAX -- 3 bytes (253 + 2 bytes) +// size <= UINT_MAX -- 5 bytes (254 + 4 bytes) +// size > UINT_MAX -- 9 bytes (255 + 8 bytes) +// +inline unsigned int GetSizeOfCompactSize(uint64 nSize) +{ + if (nSize < UCHAR_MAX-2) return sizeof(unsigned char); + else if (nSize <= USHRT_MAX) return sizeof(unsigned char) + sizeof(unsigned short); + else if (nSize <= UINT_MAX) return sizeof(unsigned char) + sizeof(unsigned int); + else return sizeof(unsigned char) + sizeof(uint64); +} + +template +void WriteCompactSize(Stream& os, uint64 nSize) +{ + if (nSize < UCHAR_MAX-2) + { + unsigned char chSize = nSize; + WRITEDATA(os, chSize); + } + else if (nSize <= USHRT_MAX) + { + unsigned char chSize = UCHAR_MAX-2; + unsigned short xSize = nSize; + WRITEDATA(os, chSize); + WRITEDATA(os, xSize); + } + else if (nSize <= UINT_MAX) + { + unsigned char chSize = UCHAR_MAX-1; + unsigned int xSize = nSize; + WRITEDATA(os, chSize); + WRITEDATA(os, xSize); + } + else + { + unsigned char chSize = UCHAR_MAX; + WRITEDATA(os, chSize); + WRITEDATA(os, nSize); + } + return; +} + +template +uint64 ReadCompactSize(Stream& is) +{ + unsigned char chSize; + READDATA(is, chSize); + if (chSize < UCHAR_MAX-2) + { + return chSize; + } + else if (chSize == UCHAR_MAX-2) + { + unsigned short nSize; + READDATA(is, nSize); + return nSize; + } + else if (chSize == UCHAR_MAX-1) + { + unsigned int nSize; + READDATA(is, nSize); + return nSize; + } + else + { + uint64 nSize; + READDATA(is, nSize); + return nSize; + } +} + + + +// +// Wrapper for serializing arrays and POD +// There's a clever template way to make arrays serialize normally, but MSVC6 doesn't support it +// +#define FLATDATA(obj) REF(CFlatData((char*)&(obj), (char*)&(obj) + sizeof(obj))) +class CFlatData +{ +protected: + char* pbegin; + char* pend; +public: + CFlatData(void* pbeginIn, void* pendIn) : pbegin((char*)pbeginIn), pend((char*)pendIn) { } + char* begin() { return pbegin; } + const char* begin() const { return pbegin; } + char* end() { return pend; } + const char* end() const { return pend; } + + unsigned int GetSerializeSize(int, int=0) const + { + return pend - pbegin; + } + + template + void Serialize(Stream& s, int, int=0) const + { + s.write(pbegin, pend - pbegin); + } + + template + void Unserialize(Stream& s, int, int=0) + { + s.read(pbegin, pend - pbegin); + } +}; + + + +// +// string stored as a fixed length field +// +template +class CFixedFieldString +{ +protected: + const string* pcstr; + string* pstr; +public: + explicit CFixedFieldString(const string& str) : pcstr(&str), pstr(NULL) { } + explicit CFixedFieldString(string& str) : pcstr(&str), pstr(&str) { } + + unsigned int GetSerializeSize(int, int=0) const + { + return LEN; + } + + template + void Serialize(Stream& s, int, int=0) const + { + char pszBuf[LEN]; + strncpy(pszBuf, pcstr->c_str(), LEN); + s.write(pszBuf, LEN); + } + + template + void Unserialize(Stream& s, int, int=0) + { + if (pstr == NULL) + throw std::ios_base::failure("CFixedFieldString::Unserialize : trying to unserialize to const string"); + char pszBuf[LEN+1]; + s.read(pszBuf, LEN); + pszBuf[LEN] = '\0'; + *pstr = pszBuf; + } +}; + + + + + +// +// Forward declarations +// + +// string +template unsigned int GetSerializeSize(const basic_string& str, int, int=0); +template void Serialize(Stream& os, const basic_string& str, int, int=0); +template void Unserialize(Stream& is, basic_string& str, int, int=0); + +// vector +template unsigned int GetSerializeSize_impl(const std::vector& v, int nType, int nVersion, const boost::true_type&); +template unsigned int GetSerializeSize_impl(const std::vector& v, int nType, int nVersion, const boost::false_type&); +template inline unsigned int GetSerializeSize(const std::vector& v, int nType, int nVersion=VERSION); +template void Serialize_impl(Stream& os, const std::vector& v, int nType, int nVersion, const boost::true_type&); +template void Serialize_impl(Stream& os, const std::vector& v, int nType, int nVersion, const boost::false_type&); +template inline void Serialize(Stream& os, const std::vector& v, int nType, int nVersion=VERSION); +template void Unserialize_impl(Stream& is, std::vector& v, int nType, int nVersion, const boost::true_type&); +template void Unserialize_impl(Stream& is, std::vector& v, int nType, int nVersion, const boost::false_type&); +template inline void Unserialize(Stream& is, std::vector& v, int nType, int nVersion=VERSION); + +// others derived from vector +extern inline unsigned int GetSerializeSize(const CScript& v, int nType, int nVersion=VERSION); +template void Serialize(Stream& os, const CScript& v, int nType, int nVersion=VERSION); +template void Unserialize(Stream& is, CScript& v, int nType, int nVersion=VERSION); + +// pair +template unsigned int GetSerializeSize(const std::pair& item, int nType, int nVersion=VERSION); +template void Serialize(Stream& os, const std::pair& item, int nType, int nVersion=VERSION); +template void Unserialize(Stream& is, std::pair& item, int nType, int nVersion=VERSION); + +// map +template unsigned int GetSerializeSize(const std::map& m, int nType, int nVersion=VERSION); +template void Serialize(Stream& os, const std::map& m, int nType, int nVersion=VERSION); +template void Unserialize(Stream& is, std::map& m, int nType, int nVersion=VERSION); + +// set +template unsigned int GetSerializeSize(const std::set& m, int nType, int nVersion=VERSION); +template void Serialize(Stream& os, const std::set& m, int nType, int nVersion=VERSION); +template void Unserialize(Stream& is, std::set& m, int nType, int nVersion=VERSION); + + + + + +// +// If none of the specialized versions above matched, default to calling member function. +// "int nType" is changed to "long nType" to keep from getting an ambiguous overload error. +// The compiler will only cast int to long if none of the other templates matched. +// Thanks to Boost serialization for this idea. +// +template +inline unsigned int GetSerializeSize(const T& a, long nType, int nVersion=VERSION) +{ + return a.GetSerializeSize((int)nType, nVersion); +} + +template +inline void Serialize(Stream& os, const T& a, long nType, int nVersion=VERSION) +{ + a.Serialize(os, (int)nType, nVersion); +} + +template +inline void Unserialize(Stream& is, T& a, long nType, int nVersion=VERSION) +{ + a.Unserialize(is, (int)nType, nVersion); +} + + + + + +// +// string +// +template +unsigned int GetSerializeSize(const basic_string& str, int, int) +{ + return GetSizeOfCompactSize(str.size()) + str.size() * sizeof(str[0]); +} + +template +void Serialize(Stream& os, const basic_string& str, int, int) +{ + WriteCompactSize(os, str.size()); + if (!str.empty()) + os.write((char*)&str[0], str.size() * sizeof(str[0])); +} + +template +void Unserialize(Stream& is, basic_string& str, int, int) +{ + unsigned int nSize = ReadCompactSize(is); + str.resize(nSize); + if (nSize != 0) + is.read((char*)&str[0], nSize * sizeof(str[0])); +} + + + +// +// vector +// +template +unsigned int GetSerializeSize_impl(const std::vector& v, int nType, int nVersion, const boost::true_type&) +{ + return (GetSizeOfCompactSize(v.size()) + v.size() * sizeof(T)); +} + +template +unsigned int GetSerializeSize_impl(const std::vector& v, int nType, int nVersion, const boost::false_type&) +{ + unsigned int nSize = GetSizeOfCompactSize(v.size()); + for (typename std::vector::const_iterator vi = v.begin(); vi != v.end(); ++vi) + nSize += GetSerializeSize((*vi), nType, nVersion); + return nSize; +} + +template +inline unsigned int GetSerializeSize(const std::vector& v, int nType, int nVersion) +{ + return GetSerializeSize_impl(v, nType, nVersion, boost::is_fundamental()); +} + + +template +void Serialize_impl(Stream& os, const std::vector& v, int nType, int nVersion, const boost::true_type&) +{ + WriteCompactSize(os, v.size()); + if (!v.empty()) + os.write((char*)&v[0], v.size() * sizeof(T)); +} + +template +void Serialize_impl(Stream& os, const std::vector& v, int nType, int nVersion, const boost::false_type&) +{ + WriteCompactSize(os, v.size()); + for (typename std::vector::const_iterator vi = v.begin(); vi != v.end(); ++vi) + ::Serialize(os, (*vi), nType, nVersion); +} + +template +inline void Serialize(Stream& os, const std::vector& v, int nType, int nVersion) +{ + Serialize_impl(os, v, nType, nVersion, boost::is_fundamental()); +} + + +template +void Unserialize_impl(Stream& is, std::vector& v, int nType, int nVersion, const boost::true_type&) +{ + //unsigned int nSize = ReadCompactSize(is); + //v.resize(nSize); + //is.read((char*)&v[0], nSize * sizeof(T)); + + // Limit size per read so bogus size value won't cause out of memory + v.clear(); + unsigned int nSize = ReadCompactSize(is); + unsigned int i = 0; + while (i < nSize) + { + unsigned int blk = min(nSize - i, 1 + 4999999 / sizeof(T)); + v.resize(i + blk); + is.read((char*)&v[i], blk * sizeof(T)); + i += blk; + } +} + +template +void Unserialize_impl(Stream& is, std::vector& v, int nType, int nVersion, const boost::false_type&) +{ + //unsigned int nSize = ReadCompactSize(is); + //v.resize(nSize); + //for (std::vector::iterator vi = v.begin(); vi != v.end(); ++vi) + // Unserialize(is, (*vi), nType, nVersion); + + v.clear(); + unsigned int nSize = ReadCompactSize(is); + unsigned int i = 0; + unsigned int nMid = 0; + while (nMid < nSize) + { + nMid += 5000000 / sizeof(T); + if (nMid > nSize) + nMid = nSize; + v.resize(nMid); + for (; i < nMid; i++) + Unserialize(is, v[i], nType, nVersion); + } +} + +template +inline void Unserialize(Stream& is, std::vector& v, int nType, int nVersion) +{ + Unserialize_impl(is, v, nType, nVersion, boost::is_fundamental()); +} + + + +// +// others derived from vector +// +inline unsigned int GetSerializeSize(const CScript& v, int nType, int nVersion) +{ + return GetSerializeSize((const vector&)v, nType, nVersion); +} + +template +void Serialize(Stream& os, const CScript& v, int nType, int nVersion) +{ + Serialize(os, (const vector&)v, nType, nVersion); +} + +template +void Unserialize(Stream& is, CScript& v, int nType, int nVersion) +{ + Unserialize(is, (vector&)v, nType, nVersion); +} + + + +// +// pair +// +template +unsigned int GetSerializeSize(const std::pair& item, int nType, int nVersion) +{ + return GetSerializeSize(item.first, nType, nVersion) + GetSerializeSize(item.second, nType, nVersion); +} + +template +void Serialize(Stream& os, const std::pair& item, int nType, int nVersion) +{ + Serialize(os, item.first, nType, nVersion); + Serialize(os, item.second, nType, nVersion); +} + +template +void Unserialize(Stream& is, std::pair& item, int nType, int nVersion) +{ + Unserialize(is, item.first, nType, nVersion); + Unserialize(is, item.second, nType, nVersion); +} + + + +// +// map +// +template +unsigned int GetSerializeSize(const std::map& m, int nType, int nVersion) +{ + unsigned int nSize = GetSizeOfCompactSize(m.size()); + for (typename std::map::const_iterator mi = m.begin(); mi != m.end(); ++mi) + nSize += GetSerializeSize((*mi), nType, nVersion); + return nSize; +} + +template +void Serialize(Stream& os, const std::map& m, int nType, int nVersion) +{ + WriteCompactSize(os, m.size()); + for (typename std::map::const_iterator mi = m.begin(); mi != m.end(); ++mi) + Serialize(os, (*mi), nType, nVersion); +} + +template +void Unserialize(Stream& is, std::map& m, int nType, int nVersion) +{ + m.clear(); + unsigned int nSize = ReadCompactSize(is); + typename std::map::iterator mi = m.begin(); + for (unsigned int i = 0; i < nSize; i++) + { + pair item; + Unserialize(is, item, nType, nVersion); + mi = m.insert(mi, item); + } +} + + + +// +// set +// +template +unsigned int GetSerializeSize(const std::set& m, int nType, int nVersion) +{ + unsigned int nSize = GetSizeOfCompactSize(m.size()); + for (typename std::set::const_iterator it = m.begin(); it != m.end(); ++it) + nSize += GetSerializeSize((*it), nType, nVersion); + return nSize; +} + +template +void Serialize(Stream& os, const std::set& m, int nType, int nVersion) +{ + WriteCompactSize(os, m.size()); + for (typename std::set::const_iterator it = m.begin(); it != m.end(); ++it) + Serialize(os, (*it), nType, nVersion); +} + +template +void Unserialize(Stream& is, std::set& m, int nType, int nVersion) +{ + m.clear(); + unsigned int nSize = ReadCompactSize(is); + typename std::set::iterator it = m.begin(); + for (unsigned int i = 0; i < nSize; i++) + { + K key; + Unserialize(is, key, nType, nVersion); + it = m.insert(it, key); + } +} + + + +// +// Support for IMPLEMENT_SERIALIZE and READWRITE macro +// +class CSerActionGetSerializeSize { }; +class CSerActionSerialize { }; +class CSerActionUnserialize { }; + +template +inline unsigned int SerReadWrite(Stream& s, const T& obj, int nType, int nVersion, CSerActionGetSerializeSize ser_action) +{ + return ::GetSerializeSize(obj, nType, nVersion); +} + +template +inline unsigned int SerReadWrite(Stream& s, const T& obj, int nType, int nVersion, CSerActionSerialize ser_action) +{ + ::Serialize(s, obj, nType, nVersion); + return 0; +} + +template +inline unsigned int SerReadWrite(Stream& s, T& obj, int nType, int nVersion, CSerActionUnserialize ser_action) +{ + ::Unserialize(s, obj, nType, nVersion); + return 0; +} + +struct ser_streamplaceholder +{ + int nType; + int nVersion; +}; + + + + + + + + + +// +// Allocator that clears its contents before deletion +// +template +struct secure_allocator : public std::allocator +{ + // MSVC8 default copy constructor is broken + typedef std::allocator base; + typedef typename base::size_type size_type; + typedef typename base::difference_type difference_type; + typedef typename base::pointer pointer; + typedef typename base::const_pointer const_pointer; + typedef typename base::reference reference; + typedef typename base::const_reference const_reference; + typedef typename base::value_type value_type; + secure_allocator() throw() {} + secure_allocator(const secure_allocator& a) throw() : base(a) {} + ~secure_allocator() throw() {} + template struct rebind + { typedef secure_allocator<_Other> other; }; + + void deallocate(T* p, std::size_t n) + { + if (p != NULL) + memset(p, 0, sizeof(T) * n); + allocator::deallocate(p, n); + } +}; + + + +// +// Double ended buffer combining vector and stream-like interfaces. +// >> and << read and write unformatted data using the above serialization templates. +// Fills with data in linear time; some stringstream implementations take N^2 time. +// +class CDataStream +{ +protected: + typedef vector > vector_type; + vector_type vch; + unsigned int nReadPos; + short state; + short exceptmask; +public: + int nType; + int nVersion; + + typedef vector_type::allocator_type allocator_type; + typedef vector_type::size_type size_type; + typedef vector_type::difference_type difference_type; + typedef vector_type::reference reference; + typedef vector_type::const_reference const_reference; + typedef vector_type::value_type value_type; + typedef vector_type::iterator iterator; + typedef vector_type::const_iterator const_iterator; + typedef vector_type::reverse_iterator reverse_iterator; + + explicit CDataStream(int nTypeIn=0, int nVersionIn=VERSION) + { + Init(nTypeIn, nVersionIn); + } + + CDataStream(const_iterator pbegin, const_iterator pend, int nTypeIn=0, int nVersionIn=VERSION) : vch(pbegin, pend) + { + Init(nTypeIn, nVersionIn); + } + +#if !defined(_MSC_VER) || _MSC_VER >= 1300 + CDataStream(const char* pbegin, const char* pend, int nTypeIn=0, int nVersionIn=VERSION) : vch(pbegin, pend) + { + Init(nTypeIn, nVersionIn); + } +#endif + + CDataStream(const vector_type& vchIn, int nTypeIn=0, int nVersionIn=VERSION) : vch(vchIn.begin(), vchIn.end()) + { + Init(nTypeIn, nVersionIn); + } + + CDataStream(const vector& vchIn, int nTypeIn=0, int nVersionIn=VERSION) : vch(vchIn.begin(), vchIn.end()) + { + Init(nTypeIn, nVersionIn); + } + + CDataStream(const vector& vchIn, int nTypeIn=0, int nVersionIn=VERSION) : vch((char*)&vchIn.begin()[0], (char*)&vchIn.end()[0]) + { + Init(nTypeIn, nVersionIn); + } + + void Init(int nTypeIn=0, int nVersionIn=VERSION) + { + nReadPos = 0; + nType = nTypeIn; + nVersion = nVersionIn; + state = 0; + exceptmask = ios::badbit | ios::failbit; + } + + CDataStream& operator+=(const CDataStream& b) + { + vch.insert(vch.end(), b.begin(), b.end()); + return *this; + } + + friend CDataStream operator+(const CDataStream& a, const CDataStream& b) + { + CDataStream ret = a; + ret += b; + return (ret); + } + + string str() const + { + return (string(begin(), end())); + } + + + // + // Vector subset + // + const_iterator begin() const { return vch.begin() + nReadPos; } + iterator begin() { return vch.begin() + nReadPos; } + const_iterator end() const { return vch.end(); } + iterator end() { return vch.end(); } + size_type size() const { return vch.size() - nReadPos; } + bool empty() const { return vch.size() == nReadPos; } + void resize(size_type n, value_type c=0) { vch.resize(n + nReadPos, c); } + void reserve(size_type n) { vch.reserve(n + nReadPos); } + const_reference operator[](size_type pos) const { return vch[pos + nReadPos]; } + reference operator[](size_type pos) { return vch[pos + nReadPos]; } + void clear() { vch.clear(); nReadPos = 0; } + iterator insert(iterator it, const char& x=char()) { return vch.insert(it, x); } + void insert(iterator it, size_type n, const char& x) { vch.insert(it, n, x); } + + void insert(iterator it, const_iterator first, const_iterator last) + { + if (it == vch.begin() + nReadPos && last - first <= nReadPos) + { + // special case for inserting at the front when there's room + nReadPos -= (last - first); + memcpy(&vch[nReadPos], &first[0], last - first); + } + else + vch.insert(it, first, last); + } + +#if !defined(_MSC_VER) || _MSC_VER >= 1300 + void insert(iterator it, const char* first, const char* last) + { + if (it == vch.begin() + nReadPos && last - first <= nReadPos) + { + // special case for inserting at the front when there's room + nReadPos -= (last - first); + memcpy(&vch[nReadPos], &first[0], last - first); + } + else + vch.insert(it, first, last); + } +#endif + + iterator erase(iterator it) + { + if (it == vch.begin() + nReadPos) + { + // special case for erasing from the front + if (++nReadPos >= vch.size()) + { + // whenever we reach the end, we take the opportunity to clear the buffer + nReadPos = 0; + return vch.erase(vch.begin(), vch.end()); + } + return vch.begin() + nReadPos; + } + else + return vch.erase(it); + } + + iterator erase(iterator first, iterator last) + { + if (first == vch.begin() + nReadPos) + { + // special case for erasing from the front + if (last == vch.end()) + { + nReadPos = 0; + return vch.erase(vch.begin(), vch.end()); + } + else + { + nReadPos = (last - vch.begin()); + return last; + } + } + else + return vch.erase(first, last); + } + + inline void Compact() + { + vch.erase(vch.begin(), vch.begin() + nReadPos); + nReadPos = 0; + } + + bool Rewind(size_type n) + { + // Rewind by n characters if the buffer hasn't been compacted yet + if (n > nReadPos) + return false; + nReadPos -= n; + return true; + } + + + // + // Stream subset + // + void setstate(short bits, const char* psz) + { + state |= bits; + if (state & exceptmask) + throw std::ios_base::failure(psz); + } + + bool eof() const { return size() == 0; } + bool fail() const { return state & (ios::badbit | ios::failbit); } + bool good() const { return !eof() && (state == 0); } + void clear(short n) { state = n; } // name conflict with vector clear() + short exceptions() { return exceptmask; } + short exceptions(short mask) { short prev = exceptmask; exceptmask = mask; setstate(0, "CDataStream"); return prev; } + CDataStream* rdbuf() { return this; } + int in_avail() { return size(); } + + void SetType(int n) { nType = n; } + int GetType() { return nType; } + void SetVersion(int n) { nVersion = n; } + int GetVersion() { return nVersion; } + void ReadVersion() { *this >> nVersion; } + void WriteVersion() { *this << nVersion; } + + CDataStream& read(char* pch, int nSize) + { + // Read from the beginning of the buffer + assert(nSize >= 0); + unsigned int nReadPosNext = nReadPos + nSize; + if (nReadPosNext >= vch.size()) + { + if (nReadPosNext > vch.size()) + { + setstate(ios::failbit, "CDataStream::read() : end of data"); + memset(pch, 0, nSize); + nSize = vch.size() - nReadPos; + } + memcpy(pch, &vch[nReadPos], nSize); + nReadPos = 0; + vch.clear(); + return (*this); + } + memcpy(pch, &vch[nReadPos], nSize); + nReadPos = nReadPosNext; + return (*this); + } + + CDataStream& ignore(int nSize) + { + // Ignore from the beginning of the buffer + assert(nSize >= 0); + unsigned int nReadPosNext = nReadPos + nSize; + if (nReadPosNext >= vch.size()) + { + if (nReadPosNext > vch.size()) + { + setstate(ios::failbit, "CDataStream::ignore() : end of data"); + nSize = vch.size() - nReadPos; + } + nReadPos = 0; + vch.clear(); + return (*this); + } + nReadPos = nReadPosNext; + return (*this); + } + + CDataStream& write(const char* pch, int nSize) + { + // Write to the end of the buffer + assert(nSize >= 0); + vch.insert(vch.end(), pch, pch + nSize); + return (*this); + } + + template + void Serialize(Stream& s, int nType=0, int nVersion=VERSION) const + { + // Special case: stream << stream concatenates like stream += stream + if (!vch.empty()) + s.write((char*)&vch[0], vch.size() * sizeof(vch[0])); + } + + template + unsigned int GetSerializeSize(const T& obj) + { + // Tells the size of the object if serialized to this stream + return ::GetSerializeSize(obj, nType, nVersion); + } + + template + CDataStream& operator<<(const T& obj) + { + // Serialize to this stream + ::Serialize(*this, obj, nType, nVersion); + return (*this); + } + + template + CDataStream& operator>>(T& obj) + { + // Unserialize from this stream + ::Unserialize(*this, obj, nType, nVersion); + return (*this); + } +}; + +#ifdef TESTCDATASTREAM +// VC6sp6 +// CDataStream: +// n=1000 0 seconds +// n=2000 0 seconds +// n=4000 0 seconds +// n=8000 0 seconds +// n=16000 0 seconds +// n=32000 0 seconds +// n=64000 1 seconds +// n=128000 1 seconds +// n=256000 2 seconds +// n=512000 4 seconds +// n=1024000 8 seconds +// n=2048000 16 seconds +// n=4096000 32 seconds +// stringstream: +// n=1000 1 seconds +// n=2000 1 seconds +// n=4000 13 seconds +// n=8000 87 seconds +// n=16000 400 seconds +// n=32000 1660 seconds +// n=64000 6749 seconds +// n=128000 27241 seconds +// n=256000 109804 seconds +#include +int main(int argc, char *argv[]) +{ + vector vch(0xcc, 250); + printf("CDataStream:\n"); + for (int n = 1000; n <= 4500000; n *= 2) + { + CDataStream ss; + time_t nStart = time(NULL); + for (int i = 0; i < n; i++) + ss.write((char*)&vch[0], vch.size()); + printf("n=%-10d %d seconds\n", n, time(NULL) - nStart); + } + printf("stringstream:\n"); + for (int n = 1000; n <= 4500000; n *= 2) + { + stringstream ss; + time_t nStart = time(NULL); + for (int i = 0; i < n; i++) + ss.write((char*)&vch[0], vch.size()); + printf("n=%-10d %d seconds\n", n, time(NULL) - nStart); + } +} +#endif + + + + + + + + + + +// +// Automatic closing wrapper for FILE* +// - Will automatically close the file when it goes out of scope if not null. +// - If you're returning the file pointer, return file.release(). +// - If you need to close the file early, use file.fclose() instead of fclose(file). +// +class CAutoFile +{ +protected: + FILE* file; + short state; + short exceptmask; +public: + int nType; + int nVersion; + + typedef FILE element_type; + + CAutoFile(FILE* filenew=NULL, int nTypeIn=SER_DISK, int nVersionIn=VERSION) + { + file = filenew; + nType = nTypeIn; + nVersion = nVersionIn; + state = 0; + exceptmask = ios::badbit | ios::failbit; + } + + ~CAutoFile() + { + fclose(); + } + + void fclose() + { + if (file != NULL && file != stdin && file != stdout && file != stderr) + ::fclose(file); + file = NULL; + } + + FILE* release() { FILE* ret = file; file = NULL; return ret; } + operator FILE*() { return file; } + FILE* operator->() { return file; } + FILE& operator*() { return *file; } + FILE** operator&() { return &file; } + FILE* operator=(FILE* pnew) { return file = pnew; } + bool operator!() { return (file == NULL); } + + + // + // Stream subset + // + void setstate(short bits, const char* psz) + { + state |= bits; + if (state & exceptmask) + throw std::ios_base::failure(psz); + } + + bool fail() const { return state & (ios::badbit | ios::failbit); } + bool good() const { return state == 0; } + void clear(short n = 0) { state = n; } + short exceptions() { return exceptmask; } + short exceptions(short mask) { short prev = exceptmask; exceptmask = mask; setstate(0, "CAutoFile"); return prev; } + + void SetType(int n) { nType = n; } + int GetType() { return nType; } + void SetVersion(int n) { nVersion = n; } + int GetVersion() { return nVersion; } + void ReadVersion() { *this >> nVersion; } + void WriteVersion() { *this << nVersion; } + + CAutoFile& read(char* pch, int nSize) + { + if (!file) + throw std::ios_base::failure("CAutoFile::read : file handle is NULL"); + if (fread(pch, 1, nSize, file) != nSize) + setstate(ios::failbit, feof(file) ? "CAutoFile::read : end of file" : "CAutoFile::read : fread failed"); + return (*this); + } + + CAutoFile& write(const char* pch, int nSize) + { + if (!file) + throw std::ios_base::failure("CAutoFile::write : file handle is NULL"); + if (fwrite(pch, 1, nSize, file) != nSize) + setstate(ios::failbit, "CAutoFile::write : write failed"); + return (*this); + } + + template + unsigned int GetSerializeSize(const T& obj) + { + // Tells the size of the object if serialized to this stream + return ::GetSerializeSize(obj, nType, nVersion); + } + + template + CAutoFile& operator<<(const T& obj) + { + // Serialize to this stream + if (!file) + throw std::ios_base::failure("CAutoFile::operator<< : file handle is NULL"); + ::Serialize(*this, obj, nType, nVersion); + return (*this); + } + + template + CAutoFile& operator>>(T& obj) + { + // Unserialize from this stream + if (!file) + throw std::ios_base::failure("CAutoFile::operator>> : file handle is NULL"); + ::Unserialize(*this, obj, nType, nVersion); + return (*this); + } +}; diff --git a/sha.cpp b/sha.cpp new file mode 100644 index 00000000..09b62fdc --- /dev/null +++ b/sha.cpp @@ -0,0 +1,554 @@ +// This file is public domain +// SHA routines extracted as a standalone file from: +// Crypto++: a C++ Class Library of Cryptographic Schemes +// Version 5.5.2 (9/24/2007) +// http://www.cryptopp.com + +// sha.cpp - modified by Wei Dai from Steve Reid's public domain sha1.c + +// Steve Reid implemented SHA-1. Wei Dai implemented SHA-2. +// Both are in the public domain. + +#include +#include +#include "sha.h" + +namespace CryptoPP +{ + +// start of Steve Reid's code + +#define blk0(i) (W[i] = data[i]) +#define blk1(i) (W[i&15] = rotlFixed(W[(i+13)&15]^W[(i+8)&15]^W[(i+2)&15]^W[i&15],1)) + +void SHA1::InitState(HashWordType *state) +{ + state[0] = 0x67452301L; + state[1] = 0xEFCDAB89L; + state[2] = 0x98BADCFEL; + state[3] = 0x10325476L; + state[4] = 0xC3D2E1F0L; +} + +#define f1(x,y,z) (z^(x&(y^z))) +#define f2(x,y,z) (x^y^z) +#define f3(x,y,z) ((x&y)|(z&(x|y))) +#define f4(x,y,z) (x^y^z) + +/* (R0+R1), R2, R3, R4 are the different operations used in SHA1 */ +#define R0(v,w,x,y,z,i) z+=f1(w,x,y)+blk0(i)+0x5A827999+rotlFixed(v,5);w=rotlFixed(w,30); +#define R1(v,w,x,y,z,i) z+=f1(w,x,y)+blk1(i)+0x5A827999+rotlFixed(v,5);w=rotlFixed(w,30); +#define R2(v,w,x,y,z,i) z+=f2(w,x,y)+blk1(i)+0x6ED9EBA1+rotlFixed(v,5);w=rotlFixed(w,30); +#define R3(v,w,x,y,z,i) z+=f3(w,x,y)+blk1(i)+0x8F1BBCDC+rotlFixed(v,5);w=rotlFixed(w,30); +#define R4(v,w,x,y,z,i) z+=f4(w,x,y)+blk1(i)+0xCA62C1D6+rotlFixed(v,5);w=rotlFixed(w,30); + +void SHA1::Transform(word32 *state, const word32 *data) +{ + word32 W[16]; + /* Copy context->state[] to working vars */ + word32 a = state[0]; + word32 b = state[1]; + word32 c = state[2]; + word32 d = state[3]; + word32 e = state[4]; + /* 4 rounds of 20 operations each. Loop unrolled. */ + R0(a,b,c,d,e, 0); R0(e,a,b,c,d, 1); R0(d,e,a,b,c, 2); R0(c,d,e,a,b, 3); + R0(b,c,d,e,a, 4); R0(a,b,c,d,e, 5); R0(e,a,b,c,d, 6); R0(d,e,a,b,c, 7); + R0(c,d,e,a,b, 8); R0(b,c,d,e,a, 9); R0(a,b,c,d,e,10); R0(e,a,b,c,d,11); + R0(d,e,a,b,c,12); R0(c,d,e,a,b,13); R0(b,c,d,e,a,14); R0(a,b,c,d,e,15); + R1(e,a,b,c,d,16); R1(d,e,a,b,c,17); R1(c,d,e,a,b,18); R1(b,c,d,e,a,19); + R2(a,b,c,d,e,20); R2(e,a,b,c,d,21); R2(d,e,a,b,c,22); R2(c,d,e,a,b,23); + R2(b,c,d,e,a,24); R2(a,b,c,d,e,25); R2(e,a,b,c,d,26); R2(d,e,a,b,c,27); + R2(c,d,e,a,b,28); R2(b,c,d,e,a,29); R2(a,b,c,d,e,30); R2(e,a,b,c,d,31); + R2(d,e,a,b,c,32); R2(c,d,e,a,b,33); R2(b,c,d,e,a,34); R2(a,b,c,d,e,35); + R2(e,a,b,c,d,36); R2(d,e,a,b,c,37); R2(c,d,e,a,b,38); R2(b,c,d,e,a,39); + R3(a,b,c,d,e,40); R3(e,a,b,c,d,41); R3(d,e,a,b,c,42); R3(c,d,e,a,b,43); + R3(b,c,d,e,a,44); R3(a,b,c,d,e,45); R3(e,a,b,c,d,46); R3(d,e,a,b,c,47); + R3(c,d,e,a,b,48); R3(b,c,d,e,a,49); R3(a,b,c,d,e,50); R3(e,a,b,c,d,51); + R3(d,e,a,b,c,52); R3(c,d,e,a,b,53); R3(b,c,d,e,a,54); R3(a,b,c,d,e,55); + R3(e,a,b,c,d,56); R3(d,e,a,b,c,57); R3(c,d,e,a,b,58); R3(b,c,d,e,a,59); + R4(a,b,c,d,e,60); R4(e,a,b,c,d,61); R4(d,e,a,b,c,62); R4(c,d,e,a,b,63); + R4(b,c,d,e,a,64); R4(a,b,c,d,e,65); R4(e,a,b,c,d,66); R4(d,e,a,b,c,67); + R4(c,d,e,a,b,68); R4(b,c,d,e,a,69); R4(a,b,c,d,e,70); R4(e,a,b,c,d,71); + R4(d,e,a,b,c,72); R4(c,d,e,a,b,73); R4(b,c,d,e,a,74); R4(a,b,c,d,e,75); + R4(e,a,b,c,d,76); R4(d,e,a,b,c,77); R4(c,d,e,a,b,78); R4(b,c,d,e,a,79); + /* Add the working vars back into context.state[] */ + state[0] += a; + state[1] += b; + state[2] += c; + state[3] += d; + state[4] += e; +} + +// end of Steve Reid's code + +// ************************************************************* + +void SHA224::InitState(HashWordType *state) +{ + static const word32 s[8] = {0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939, 0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4}; + memcpy(state, s, sizeof(s)); +} + +void SHA256::InitState(HashWordType *state) +{ + static const word32 s[8] = {0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19}; + memcpy(state, s, sizeof(s)); +} + +static const word32 SHA256_K[64] = { + 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, + 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5, + 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, + 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, + 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, + 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, + 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, + 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967, + 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, + 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, + 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, + 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, + 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, + 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3, + 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, + 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2 +}; + +#define blk2(i) (W[i&15]+=s1(W[(i-2)&15])+W[(i-7)&15]+s0(W[(i-15)&15])) + +#define Ch(x,y,z) (z^(x&(y^z))) +#define Maj(x,y,z) ((x&y)|(z&(x|y))) + +#define a(i) T[(0-i)&7] +#define b(i) T[(1-i)&7] +#define c(i) T[(2-i)&7] +#define d(i) T[(3-i)&7] +#define e(i) T[(4-i)&7] +#define f(i) T[(5-i)&7] +#define g(i) T[(6-i)&7] +#define h(i) T[(7-i)&7] + +#define R(i) h(i)+=S1(e(i))+Ch(e(i),f(i),g(i))+SHA256_K[i+j]+(j?blk2(i):blk0(i));\ + d(i)+=h(i);h(i)+=S0(a(i))+Maj(a(i),b(i),c(i)) + +// for SHA256 +#define S0(x) (rotrFixed(x,2)^rotrFixed(x,13)^rotrFixed(x,22)) +#define S1(x) (rotrFixed(x,6)^rotrFixed(x,11)^rotrFixed(x,25)) +#define s0(x) (rotrFixed(x,7)^rotrFixed(x,18)^(x>>3)) +#define s1(x) (rotrFixed(x,17)^rotrFixed(x,19)^(x>>10)) + +void SHA256::Transform(word32 *state, const word32 *data) +{ + word32 W[16]; + word32 T[8]; + /* Copy context->state[] to working vars */ + memcpy(T, state, sizeof(T)); + /* 64 operations, partially loop unrolled */ + for (unsigned int j=0; j<64; j+=16) + { + R( 0); R( 1); R( 2); R( 3); + R( 4); R( 5); R( 6); R( 7); + R( 8); R( 9); R(10); R(11); + R(12); R(13); R(14); R(15); + } + /* Add the working vars back into context.state[] */ + state[0] += a(0); + state[1] += b(0); + state[2] += c(0); + state[3] += d(0); + state[4] += e(0); + state[5] += f(0); + state[6] += g(0); + state[7] += h(0); +} + +/* +// smaller but slower +void SHA256_Transform(word32 *state, const word32 *data) +{ + word32 T[20]; + word32 W[32]; + unsigned int i = 0, j = 0; + word32 *t = T+8; + + memcpy(t, state, 8*4); + word32 e = t[4], a = t[0]; + + do + { + word32 w = data[j]; + W[j] = w; + w += K[j]; + w += t[7]; + w += S1(e); + w += Ch(e, t[5], t[6]); + e = t[3] + w; + t[3] = t[3+8] = e; + w += S0(t[0]); + a = w + Maj(a, t[1], t[2]); + t[-1] = t[7] = a; + --t; + ++j; + if (j%8 == 0) + t += 8; + } while (j<16); + + do + { + i = j&0xf; + word32 w = s1(W[i+16-2]) + s0(W[i+16-15]) + W[i] + W[i+16-7]; + W[i+16] = W[i] = w; + w += K[j]; + w += t[7]; + w += S1(e); + w += Ch(e, t[5], t[6]); + e = t[3] + w; + t[3] = t[3+8] = e; + w += S0(t[0]); + a = w + Maj(a, t[1], t[2]); + t[-1] = t[7] = a; + + w = s1(W[(i+1)+16-2]) + s0(W[(i+1)+16-15]) + W[(i+1)] + W[(i+1)+16-7]; + W[(i+1)+16] = W[(i+1)] = w; + w += K[j+1]; + w += (t-1)[7]; + w += S1(e); + w += Ch(e, (t-1)[5], (t-1)[6]); + e = (t-1)[3] + w; + (t-1)[3] = (t-1)[3+8] = e; + w += S0((t-1)[0]); + a = w + Maj(a, (t-1)[1], (t-1)[2]); + (t-1)[-1] = (t-1)[7] = a; + + t-=2; + j+=2; + if (j%8 == 0) + t += 8; + } while (j<64); + + state[0] += a; + state[1] += t[1]; + state[2] += t[2]; + state[3] += t[3]; + state[4] += e; + state[5] += t[5]; + state[6] += t[6]; + state[7] += t[7]; +} +*/ + +#undef S0 +#undef S1 +#undef s0 +#undef s1 +#undef R + +// ************************************************************* + +#ifdef WORD64_AVAILABLE + +void SHA384::InitState(HashWordType *state) +{ + static const word64 s[8] = { + W64LIT(0xcbbb9d5dc1059ed8), W64LIT(0x629a292a367cd507), + W64LIT(0x9159015a3070dd17), W64LIT(0x152fecd8f70e5939), + W64LIT(0x67332667ffc00b31), W64LIT(0x8eb44a8768581511), + W64LIT(0xdb0c2e0d64f98fa7), W64LIT(0x47b5481dbefa4fa4)}; + memcpy(state, s, sizeof(s)); +} + +void SHA512::InitState(HashWordType *state) +{ + static const word64 s[8] = { + W64LIT(0x6a09e667f3bcc908), W64LIT(0xbb67ae8584caa73b), + W64LIT(0x3c6ef372fe94f82b), W64LIT(0xa54ff53a5f1d36f1), + W64LIT(0x510e527fade682d1), W64LIT(0x9b05688c2b3e6c1f), + W64LIT(0x1f83d9abfb41bd6b), W64LIT(0x5be0cd19137e2179)}; + memcpy(state, s, sizeof(s)); +} + +CRYPTOPP_ALIGN_DATA(16) static const word64 SHA512_K[80] CRYPTOPP_SECTION_ALIGN16 = { + W64LIT(0x428a2f98d728ae22), W64LIT(0x7137449123ef65cd), + W64LIT(0xb5c0fbcfec4d3b2f), W64LIT(0xe9b5dba58189dbbc), + W64LIT(0x3956c25bf348b538), W64LIT(0x59f111f1b605d019), + W64LIT(0x923f82a4af194f9b), W64LIT(0xab1c5ed5da6d8118), + W64LIT(0xd807aa98a3030242), W64LIT(0x12835b0145706fbe), + W64LIT(0x243185be4ee4b28c), W64LIT(0x550c7dc3d5ffb4e2), + W64LIT(0x72be5d74f27b896f), W64LIT(0x80deb1fe3b1696b1), + W64LIT(0x9bdc06a725c71235), W64LIT(0xc19bf174cf692694), + W64LIT(0xe49b69c19ef14ad2), W64LIT(0xefbe4786384f25e3), + W64LIT(0x0fc19dc68b8cd5b5), W64LIT(0x240ca1cc77ac9c65), + W64LIT(0x2de92c6f592b0275), W64LIT(0x4a7484aa6ea6e483), + W64LIT(0x5cb0a9dcbd41fbd4), W64LIT(0x76f988da831153b5), + W64LIT(0x983e5152ee66dfab), W64LIT(0xa831c66d2db43210), + W64LIT(0xb00327c898fb213f), W64LIT(0xbf597fc7beef0ee4), + W64LIT(0xc6e00bf33da88fc2), W64LIT(0xd5a79147930aa725), + W64LIT(0x06ca6351e003826f), W64LIT(0x142929670a0e6e70), + W64LIT(0x27b70a8546d22ffc), W64LIT(0x2e1b21385c26c926), + W64LIT(0x4d2c6dfc5ac42aed), W64LIT(0x53380d139d95b3df), + W64LIT(0x650a73548baf63de), W64LIT(0x766a0abb3c77b2a8), + W64LIT(0x81c2c92e47edaee6), W64LIT(0x92722c851482353b), + W64LIT(0xa2bfe8a14cf10364), W64LIT(0xa81a664bbc423001), + W64LIT(0xc24b8b70d0f89791), W64LIT(0xc76c51a30654be30), + W64LIT(0xd192e819d6ef5218), W64LIT(0xd69906245565a910), + W64LIT(0xf40e35855771202a), W64LIT(0x106aa07032bbd1b8), + W64LIT(0x19a4c116b8d2d0c8), W64LIT(0x1e376c085141ab53), + W64LIT(0x2748774cdf8eeb99), W64LIT(0x34b0bcb5e19b48a8), + W64LIT(0x391c0cb3c5c95a63), W64LIT(0x4ed8aa4ae3418acb), + W64LIT(0x5b9cca4f7763e373), W64LIT(0x682e6ff3d6b2b8a3), + W64LIT(0x748f82ee5defb2fc), W64LIT(0x78a5636f43172f60), + W64LIT(0x84c87814a1f0ab72), W64LIT(0x8cc702081a6439ec), + W64LIT(0x90befffa23631e28), W64LIT(0xa4506cebde82bde9), + W64LIT(0xbef9a3f7b2c67915), W64LIT(0xc67178f2e372532b), + W64LIT(0xca273eceea26619c), W64LIT(0xd186b8c721c0c207), + W64LIT(0xeada7dd6cde0eb1e), W64LIT(0xf57d4f7fee6ed178), + W64LIT(0x06f067aa72176fba), W64LIT(0x0a637dc5a2c898a6), + W64LIT(0x113f9804bef90dae), W64LIT(0x1b710b35131c471b), + W64LIT(0x28db77f523047d84), W64LIT(0x32caab7b40c72493), + W64LIT(0x3c9ebe0a15c9bebc), W64LIT(0x431d67c49c100d4c), + W64LIT(0x4cc5d4becb3e42b6), W64LIT(0x597f299cfc657e2a), + W64LIT(0x5fcb6fab3ad6faec), W64LIT(0x6c44198c4a475817) +}; + +#if CRYPTOPP_BOOL_SSE2_ASM_AVAILABLE && CRYPTOPP_BOOL_X86 +// put assembly version in separate function, otherwise MSVC 2005 SP1 doesn't generate correct code for the non-assembly version +CRYPTOPP_NAKED static void CRYPTOPP_FASTCALL SHA512_SSE2_Transform(word64 *state, const word64 *data) +{ +#ifdef __GNUC__ + __asm__ __volatile__ + ( + ".intel_syntax noprefix;" + AS1( push ebx) + AS2( mov ebx, eax) +#else + AS1( push ebx) + AS1( push esi) + AS1( push edi) + AS2( lea ebx, SHA512_K) +#endif + + AS2( mov eax, esp) + AS2( and esp, 0xfffffff0) + AS2( sub esp, 27*16) // 17*16 for expanded data, 20*8 for state + AS1( push eax) + AS2( xor eax, eax) + AS2( lea edi, [esp+4+8*8]) // start at middle of state buffer. will decrement pointer each round to avoid copying + AS2( lea esi, [esp+4+20*8+8]) // 16-byte alignment, then add 8 + + AS2( movq mm4, [ecx+0*8]) + AS2( movq [edi+0*8], mm4) + AS2( movq mm0, [ecx+1*8]) + AS2( movq [edi+1*8], mm0) + AS2( movq mm0, [ecx+2*8]) + AS2( movq [edi+2*8], mm0) + AS2( movq mm0, [ecx+3*8]) + AS2( movq [edi+3*8], mm0) + AS2( movq mm5, [ecx+4*8]) + AS2( movq [edi+4*8], mm5) + AS2( movq mm0, [ecx+5*8]) + AS2( movq [edi+5*8], mm0) + AS2( movq mm0, [ecx+6*8]) + AS2( movq [edi+6*8], mm0) + AS2( movq mm0, [ecx+7*8]) + AS2( movq [edi+7*8], mm0) + ASJ( jmp, 0, f) + +#define SSE2_S0_S1(r, a, b, c) \ + AS2( movq mm6, r)\ + AS2( psrlq r, a)\ + AS2( movq mm7, r)\ + AS2( psllq mm6, 64-c)\ + AS2( pxor mm7, mm6)\ + AS2( psrlq r, b-a)\ + AS2( pxor mm7, r)\ + AS2( psllq mm6, c-b)\ + AS2( pxor mm7, mm6)\ + AS2( psrlq r, c-b)\ + AS2( pxor r, mm7)\ + AS2( psllq mm6, b-a)\ + AS2( pxor r, mm6) + +#define SSE2_s0(r, a, b, c) \ + AS2( movdqa xmm6, r)\ + AS2( psrlq r, a)\ + AS2( movdqa xmm7, r)\ + AS2( psllq xmm6, 64-c)\ + AS2( pxor xmm7, xmm6)\ + AS2( psrlq r, b-a)\ + AS2( pxor xmm7, r)\ + AS2( psrlq r, c-b)\ + AS2( pxor r, xmm7)\ + AS2( psllq xmm6, c-a)\ + AS2( pxor r, xmm6) + +#define SSE2_s1(r, a, b, c) \ + AS2( movdqa xmm6, r)\ + AS2( psrlq r, a)\ + AS2( movdqa xmm7, r)\ + AS2( psllq xmm6, 64-c)\ + AS2( pxor xmm7, xmm6)\ + AS2( psrlq r, b-a)\ + AS2( pxor xmm7, r)\ + AS2( psllq xmm6, c-b)\ + AS2( pxor xmm7, xmm6)\ + AS2( psrlq r, c-b)\ + AS2( pxor r, xmm7) + + ASL(SHA512_Round) + // k + w is in mm0, a is in mm4, e is in mm5 + AS2( paddq mm0, [edi+7*8]) // h + AS2( movq mm2, [edi+5*8]) // f + AS2( movq mm3, [edi+6*8]) // g + AS2( pxor mm2, mm3) + AS2( pand mm2, mm5) + SSE2_S0_S1(mm5,14,18,41) + AS2( pxor mm2, mm3) + AS2( paddq mm0, mm2) // h += Ch(e,f,g) + AS2( paddq mm5, mm0) // h += S1(e) + AS2( movq mm2, [edi+1*8]) // b + AS2( movq mm1, mm2) + AS2( por mm2, mm4) + AS2( pand mm2, [edi+2*8]) // c + AS2( pand mm1, mm4) + AS2( por mm1, mm2) + AS2( paddq mm1, mm5) // temp = h + Maj(a,b,c) + AS2( paddq mm5, [edi+3*8]) // e = d + h + AS2( movq [edi+3*8], mm5) + AS2( movq [edi+11*8], mm5) + SSE2_S0_S1(mm4,28,34,39) // S0(a) + AS2( paddq mm4, mm1) // a = temp + S0(a) + AS2( movq [edi-8], mm4) + AS2( movq [edi+7*8], mm4) + AS1( ret) + + // first 16 rounds + ASL(0) + AS2( movq mm0, [edx+eax*8]) + AS2( movq [esi+eax*8], mm0) + AS2( movq [esi+eax*8+16*8], mm0) + AS2( paddq mm0, [ebx+eax*8]) + ASC( call, SHA512_Round) + AS1( inc eax) + AS2( sub edi, 8) + AS2( test eax, 7) + ASJ( jnz, 0, b) + AS2( add edi, 8*8) + AS2( cmp eax, 16) + ASJ( jne, 0, b) + + // rest of the rounds + AS2( movdqu xmm0, [esi+(16-2)*8]) + ASL(1) + // data expansion, W[i-2] already in xmm0 + AS2( movdqu xmm3, [esi]) + AS2( paddq xmm3, [esi+(16-7)*8]) + AS2( movdqa xmm2, [esi+(16-15)*8]) + SSE2_s1(xmm0, 6, 19, 61) + AS2( paddq xmm0, xmm3) + SSE2_s0(xmm2, 1, 7, 8) + AS2( paddq xmm0, xmm2) + AS2( movdq2q mm0, xmm0) + AS2( movhlps xmm1, xmm0) + AS2( paddq mm0, [ebx+eax*8]) + AS2( movlps [esi], xmm0) + AS2( movlps [esi+8], xmm1) + AS2( movlps [esi+8*16], xmm0) + AS2( movlps [esi+8*17], xmm1) + // 2 rounds + ASC( call, SHA512_Round) + AS2( sub edi, 8) + AS2( movdq2q mm0, xmm1) + AS2( paddq mm0, [ebx+eax*8+8]) + ASC( call, SHA512_Round) + // update indices and loop + AS2( add esi, 16) + AS2( add eax, 2) + AS2( sub edi, 8) + AS2( test eax, 7) + ASJ( jnz, 1, b) + // do housekeeping every 8 rounds + AS2( mov esi, 0xf) + AS2( and esi, eax) + AS2( lea esi, [esp+4+20*8+8+esi*8]) + AS2( add edi, 8*8) + AS2( cmp eax, 80) + ASJ( jne, 1, b) + +#define SSE2_CombineState(i) \ + AS2( movq mm0, [edi+i*8])\ + AS2( paddq mm0, [ecx+i*8])\ + AS2( movq [ecx+i*8], mm0) + + SSE2_CombineState(0) + SSE2_CombineState(1) + SSE2_CombineState(2) + SSE2_CombineState(3) + SSE2_CombineState(4) + SSE2_CombineState(5) + SSE2_CombineState(6) + SSE2_CombineState(7) + + AS1( pop esp) + AS1( emms) + +#if defined(__GNUC__) + AS1( pop ebx) + ".att_syntax prefix;" + : + : "a" (SHA512_K), "c" (state), "d" (data) + : "%esi", "%edi", "memory", "cc" + ); +#else + AS1( pop edi) + AS1( pop esi) + AS1( pop ebx) + AS1( ret) +#endif +} +#endif // #if CRYPTOPP_BOOL_SSE2_ASM_AVAILABLE + +void SHA512::Transform(word64 *state, const word64 *data) +{ +#if CRYPTOPP_BOOL_SSE2_ASM_AVAILABLE && CRYPTOPP_BOOL_X86 + if (HasSSE2()) + { + SHA512_SSE2_Transform(state, data); + return; + } +#endif + +#define S0(x) (rotrFixed(x,28)^rotrFixed(x,34)^rotrFixed(x,39)) +#define S1(x) (rotrFixed(x,14)^rotrFixed(x,18)^rotrFixed(x,41)) +#define s0(x) (rotrFixed(x,1)^rotrFixed(x,8)^(x>>7)) +#define s1(x) (rotrFixed(x,19)^rotrFixed(x,61)^(x>>6)) + +#define R(i) h(i)+=S1(e(i))+Ch(e(i),f(i),g(i))+SHA512_K[i+j]+(j?blk2(i):blk0(i));\ + d(i)+=h(i);h(i)+=S0(a(i))+Maj(a(i),b(i),c(i)) + + word64 W[16]; + word64 T[8]; + /* Copy context->state[] to working vars */ + memcpy(T, state, sizeof(T)); + /* 80 operations, partially loop unrolled */ + for (unsigned int j=0; j<80; j+=16) + { + R( 0); R( 1); R( 2); R( 3); + R( 4); R( 5); R( 6); R( 7); + R( 8); R( 9); R(10); R(11); + R(12); R(13); R(14); R(15); + } + /* Add the working vars back into context.state[] */ + state[0] += a(0); + state[1] += b(0); + state[2] += c(0); + state[3] += d(0); + state[4] += e(0); + state[5] += f(0); + state[6] += g(0); + state[7] += h(0); +} + +#endif + +} diff --git a/sha.h b/sha.h new file mode 100644 index 00000000..4e56b56d --- /dev/null +++ b/sha.h @@ -0,0 +1,177 @@ +// This file is public domain +// SHA routines extracted as a standalone file from: +// Crypto++: a C++ Class Library of Cryptographic Schemes +// Version 5.5.2 (9/24/2007) +// http://www.cryptopp.com +#ifndef CRYPTOPP_SHA_H +#define CRYPTOPP_SHA_H +#include + +namespace CryptoPP +{ + +// +// Dependencies +// + +typedef unsigned char byte; +typedef unsigned short word16; +typedef unsigned int word32; +#if defined(_MSC_VER) || defined(__BORLANDC__) +typedef unsigned __int64 word64; +#else +typedef unsigned long long word64; +#endif + +template inline T rotlFixed(T x, unsigned int y) +{ + assert(y < sizeof(T)*8); + return T((x<>(sizeof(T)*8-y))); +} + +template inline T rotrFixed(T x, unsigned int y) +{ + assert(y < sizeof(T)*8); + return T((x>>y) | (x<<(sizeof(T)*8-y))); +} + +// ************** endian reversal *************** + +#ifdef _MSC_VER + #if _MSC_VER >= 1400 + #define CRYPTOPP_FAST_ROTATE(x) 1 + #elif _MSC_VER >= 1300 + #define CRYPTOPP_FAST_ROTATE(x) ((x) == 32 | (x) == 64) + #else + #define CRYPTOPP_FAST_ROTATE(x) ((x) == 32) + #endif +#elif (defined(__MWERKS__) && TARGET_CPU_PPC) || \ + (defined(__GNUC__) && (defined(_ARCH_PWR2) || defined(_ARCH_PWR) || defined(_ARCH_PPC) || defined(_ARCH_PPC64) || defined(_ARCH_COM))) + #define CRYPTOPP_FAST_ROTATE(x) ((x) == 32) +#elif defined(__GNUC__) && (CRYPTOPP_BOOL_X64 || CRYPTOPP_BOOL_X86) // depend on GCC's peephole optimization to generate rotate instructions + #define CRYPTOPP_FAST_ROTATE(x) 1 +#else + #define CRYPTOPP_FAST_ROTATE(x) 0 +#endif + +inline byte ByteReverse(byte value) +{ + return value; +} + +inline word16 ByteReverse(word16 value) +{ +#ifdef CRYPTOPP_BYTESWAP_AVAILABLE + return bswap_16(value); +#elif defined(_MSC_VER) && _MSC_VER >= 1300 + return _byteswap_ushort(value); +#else + return rotlFixed(value, 8U); +#endif +} + +inline word32 ByteReverse(word32 value) +{ +#if defined(__GNUC__) + __asm__ ("bswap %0" : "=r" (value) : "0" (value)); + return value; +#elif defined(CRYPTOPP_BYTESWAP_AVAILABLE) + return bswap_32(value); +#elif defined(__MWERKS__) && TARGET_CPU_PPC + return (word32)__lwbrx(&value,0); +#elif _MSC_VER >= 1400 || (_MSC_VER >= 1300 && !defined(_DLL)) + return _byteswap_ulong(value); +#elif CRYPTOPP_FAST_ROTATE(32) + // 5 instructions with rotate instruction, 9 without + return (rotrFixed(value, 8U) & 0xff00ff00) | (rotlFixed(value, 8U) & 0x00ff00ff); +#else + // 6 instructions with rotate instruction, 8 without + value = ((value & 0xFF00FF00) >> 8) | ((value & 0x00FF00FF) << 8); + return rotlFixed(value, 16U); +#endif +} + +#ifdef WORD64_AVAILABLE +inline word64 ByteReverse(word64 value) +{ +#if defined(__GNUC__) && defined(__x86_64__) + __asm__ ("bswap %0" : "=r" (value) : "0" (value)); + return value; +#elif defined(CRYPTOPP_BYTESWAP_AVAILABLE) + return bswap_64(value); +#elif defined(_MSC_VER) && _MSC_VER >= 1300 + return _byteswap_uint64(value); +#elif defined(CRYPTOPP_SLOW_WORD64) + return (word64(ByteReverse(word32(value))) << 32) | ByteReverse(word32(value>>32)); +#else + value = ((value & W64LIT(0xFF00FF00FF00FF00)) >> 8) | ((value & W64LIT(0x00FF00FF00FF00FF)) << 8); + value = ((value & W64LIT(0xFFFF0000FFFF0000)) >> 16) | ((value & W64LIT(0x0000FFFF0000FFFF)) << 16); + return rotlFixed(value, 32U); +#endif +} +#endif + + +// +// SHA +// + +// http://www.weidai.com/scan-mirror/md.html#SHA-1 +class SHA1 +{ +public: + typedef word32 HashWordType; + static void InitState(word32 *state); + static void Transform(word32 *digest, const word32 *data); + static const char * StaticAlgorithmName() {return "SHA-1";} +}; + +typedef SHA1 SHA; // for backwards compatibility + +// implements the SHA-256 standard +class SHA256 +{ +public: + typedef word32 HashWordType; + static void InitState(word32 *state); + static void Transform(word32 *digest, const word32 *data); + static const char * StaticAlgorithmName() {return "SHA-256";} +}; + +// implements the SHA-224 standard +class SHA224 +{ +public: + typedef word32 HashWordType; + static void InitState(word32 *state); + static void Transform(word32 *digest, const word32 *data) {SHA256::Transform(digest, data);} + static const char * StaticAlgorithmName() {return "SHA-224";} +}; + +#ifdef WORD64_AVAILABLE + +// implements the SHA-512 standard +class SHA512 +{ +public: + typedef word64 HashWordType; + static void InitState(word64 *state); + static void Transform(word64 *digest, const word64 *data); + static const char * StaticAlgorithmName() {return "SHA-512";} +}; + +// implements the SHA-384 standard +class SHA384 +{ +public: + typedef word64 HashWordType; + static void InitState(word64 *state); + static void Transform(word64 *digest, const word64 *data) {SHA512::Transform(digest, data);} + static const char * StaticAlgorithmName() {return "SHA-384";} +}; + +#endif + +} + +#endif diff --git a/ui.cpp b/ui.cpp new file mode 100644 index 00000000..96c5a83d --- /dev/null +++ b/ui.cpp @@ -0,0 +1,3290 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" +#ifdef _MSC_VER +#include +#endif + + + +DEFINE_EVENT_TYPE(wxEVT_CROSSTHREADCALL) +DEFINE_EVENT_TYPE(wxEVT_REPLY1) +DEFINE_EVENT_TYPE(wxEVT_REPLY2) +DEFINE_EVENT_TYPE(wxEVT_REPLY3) +DEFINE_EVENT_TYPE(wxEVT_TABLEADDED) +DEFINE_EVENT_TYPE(wxEVT_TABLEUPDATED) +DEFINE_EVENT_TYPE(wxEVT_TABLEDELETED) + +CMainFrame* pframeMain = NULL; +map mapAddressBook; + + +void ThreadRequestProductDetails(void* parg); +void ThreadRandSendTest(void* parg); +bool fRandSendTest = false; +void RandSend(); +extern int g_isPainting; + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// Util +// + +void HandleCtrlA(wxKeyEvent& event) +{ + // Ctrl-a select all + wxTextCtrl* textCtrl = (wxTextCtrl*)event.GetEventObject(); + if (event.GetModifiers() == wxMOD_CONTROL && event.GetKeyCode() == 'A') + textCtrl->SetSelection(-1, -1); + event.Skip(); +} + +bool Is24HourTime() +{ + //char pszHourFormat[256]; + //pszHourFormat[0] = '\0'; + //GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_ITIME, pszHourFormat, 256); + //return (pszHourFormat[0] != '0'); + return true; +} + +string DateStr(int64 nTime) +{ + return (string)wxDateTime((time_t)nTime).FormatDate(); +} + +string DateTimeStr(int64 nTime) +{ + wxDateTime datetime((time_t)nTime); + if (Is24HourTime()) + return (string)datetime.Format("%x %H:%M"); + else + return (string)datetime.Format("%x ") + itostr((datetime.GetHour() + 11) % 12 + 1) + (string)datetime.Format(":%M %p"); +} + +wxString GetItemText(wxListCtrl* listCtrl, int nIndex, int nColumn) +{ + // Helper to simplify access to listctrl + wxListItem item; + item.m_itemId = nIndex; + item.m_col = nColumn; + item.m_mask = wxLIST_MASK_TEXT; + if (!listCtrl->GetItem(item)) + return ""; + return item.GetText(); +} + +int InsertLine(wxListCtrl* listCtrl, const wxString& str0, const wxString& str1) +{ + int nIndex = listCtrl->InsertItem(listCtrl->GetItemCount(), str0); + listCtrl->SetItem(nIndex, 1, str1); + return nIndex; +} + +int InsertLine(wxListCtrl* listCtrl, const wxString& str0, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4) +{ + int nIndex = listCtrl->InsertItem(listCtrl->GetItemCount(), str0); + listCtrl->SetItem(nIndex, 1, str1); + listCtrl->SetItem(nIndex, 2, str2); + listCtrl->SetItem(nIndex, 3, str3); + listCtrl->SetItem(nIndex, 4, str4); + return nIndex; +} + +int InsertLine(wxListCtrl* listCtrl, void* pdata, const wxString& str0, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4) +{ + int nIndex = listCtrl->InsertItem(listCtrl->GetItemCount(), str0); + listCtrl->SetItemPtrData(nIndex, (wxUIntPtr)pdata); + listCtrl->SetItem(nIndex, 1, str1); + listCtrl->SetItem(nIndex, 2, str2); + listCtrl->SetItem(nIndex, 3, str3); + listCtrl->SetItem(nIndex, 4, str4); + return nIndex; +} + +void SetSelection(wxListCtrl* listCtrl, int nIndex) +{ + int nSize = listCtrl->GetItemCount(); + long nState = (wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED); + for (int i = 0; i < nSize; i++) + listCtrl->SetItemState(i, (i == nIndex ? nState : 0), nState); +} + +int GetSelection(wxListCtrl* listCtrl) +{ + int nSize = listCtrl->GetItemCount(); + for (int i = 0; i < nSize; i++) + if (listCtrl->GetItemState(i, wxLIST_STATE_FOCUSED)) + return i; + return -1; +} + + +string HtmlEscape(const char* psz, bool fMultiLine=false) +{ + int len = 0; + for (const char* p = psz; *p; p++) + { + if (*p == '<') len += 4; + else if (*p == '>') len += 4; + else if (*p == '&') len += 5; + else if (*p == '"') len += 6; + else if (*p == ' ' && p > psz && p[-1] == ' ' && p[1] == ' ') len += 6; + else if (*p == '\n' && fMultiLine) len += 5; + else + len++; + } + string str; + str.reserve(len); + for (const char* p = psz; *p; p++) + { + if (*p == '<') str += "<"; + else if (*p == '>') str += ">"; + else if (*p == '&') str += "&"; + else if (*p == '"') str += """; + else if (*p == ' ' && p > psz && p[-1] == ' ' && p[1] == ' ') str += " "; + else if (*p == '\n' && fMultiLine) str += "
\n"; + else + str += *p; + } + return str; +} + +string HtmlEscape(const string& str, bool fMultiLine=false) +{ + return HtmlEscape(str.c_str(), fMultiLine); +} + +void AddToMyProducts(CProduct product) +{ + CProduct& productInsert = mapMyProducts[product.GetHash()]; + productInsert = product; + InsertLine(pframeMain->m_listCtrlProductsSent, &productInsert, + product.mapValue["category"], + product.mapValue["title"].substr(0, 100), + product.mapValue["description"].substr(0, 100), + product.mapValue["price"], + ""); +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// Custom events +// + +set setCallbackAvailable; +CCriticalSection cs_setCallbackAvailable; + +void AddCallbackAvailable(void* p) +{ + CRITICAL_BLOCK(cs_setCallbackAvailable) + setCallbackAvailable.insert(p); +} + +void RemoveCallbackAvailable(void* p) +{ + CRITICAL_BLOCK(cs_setCallbackAvailable) + setCallbackAvailable.erase(p); +} + +bool IsCallbackAvailable(void* p) +{ + CRITICAL_BLOCK(cs_setCallbackAvailable) + return setCallbackAvailable.count(p); + return false; +} + +template +void AddPendingCustomEvent(wxEvtHandler* pevthandler, int nEventID, const T pbeginIn, const T pendIn) +{ + if (!pevthandler) + return; + + const char* pbegin = (pendIn != pbeginIn) ? &pbeginIn[0] : NULL; + const char* pend = pbegin + (pendIn - pbeginIn) * sizeof(pbeginIn[0]); + wxCommandEvent event(nEventID); + wxString strData(wxChar(0), (pend - pbegin) / sizeof(wxChar) + 1); + memcpy(&strData[0], pbegin, pend - pbegin); + event.SetString(strData); + event.SetInt(pend - pbegin); + + pevthandler->AddPendingEvent(event); +} + +template +void AddPendingCustomEvent(wxEvtHandler* pevthandler, int nEventID, const T& obj) +{ + CDataStream ss; + ss << obj; + AddPendingCustomEvent(pevthandler, nEventID, ss.begin(), ss.end()); +} + +void AddPendingReplyEvent1(void* pevthandler, CDataStream& vRecv) +{ + if (IsCallbackAvailable(pevthandler)) + AddPendingCustomEvent((wxEvtHandler*)pevthandler, wxEVT_REPLY1, vRecv.begin(), vRecv.end()); +} + +void AddPendingReplyEvent2(void* pevthandler, CDataStream& vRecv) +{ + if (IsCallbackAvailable(pevthandler)) + AddPendingCustomEvent((wxEvtHandler*)pevthandler, wxEVT_REPLY2, vRecv.begin(), vRecv.end()); +} + +void AddPendingReplyEvent3(void* pevthandler, CDataStream& vRecv) +{ + if (IsCallbackAvailable(pevthandler)) + AddPendingCustomEvent((wxEvtHandler*)pevthandler, wxEVT_REPLY3, vRecv.begin(), vRecv.end()); +} + +CDataStream GetStreamFromEvent(const wxCommandEvent& event) +{ + wxString strData = event.GetString(); + return CDataStream(strData.begin(), strData.begin() + event.GetInt(), SER_NETWORK); +} + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CMainFrame +// + +CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) +{ + Connect(wxEVT_CROSSTHREADCALL, wxCommandEventHandler(CMainFrame::OnCrossThreadCall), NULL, this); + + // Init + fRefreshListCtrl = false; + fRefreshListCtrlRunning = false; + fOnSetFocusAddress = false; + pindexBestLast = NULL; + m_choiceFilter->SetSelection(0); + m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); + m_listCtrl->SetFocus(); + SetIcon(wxICON(bitcoin)); + m_menuOptions->Check(wxID_OPTIONSGENERATEBITCOINS, fGenerateBitcoins); + + // Init toolbar with transparency masked bitmaps + m_toolBar->ClearTools(); + + //// shouldn't have to do mask separately anymore, bitmap alpha support added in wx 2.8.9, + wxBitmap bmpSend(wxT("send20"), wxBITMAP_TYPE_RESOURCE); + bmpSend.SetMask(new wxMask(wxBitmap(wxT("send20mask"), wxBITMAP_TYPE_RESOURCE))); + m_toolBar->AddTool(wxID_BUTTONSEND, wxT("&Send Coins"), bmpSend, wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + + wxBitmap bmpAddressBook(wxT("addressbook20"), wxBITMAP_TYPE_RESOURCE); + bmpAddressBook.SetMask(new wxMask(wxBitmap(wxT("addressbook20mask"), wxBITMAP_TYPE_RESOURCE))); + m_toolBar->AddTool(wxID_BUTTONRECEIVE, wxT("&Address Book"), bmpAddressBook, wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + + m_toolBar->Realize(); + + // Init column headers + int nDateWidth = DateTimeStr(1229413914).size() * 6 + 8; + m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, 0); + m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, 0); + m_listCtrl->InsertColumn(2, "Status", wxLIST_FORMAT_LEFT, 90); + m_listCtrl->InsertColumn(3, "Date", wxLIST_FORMAT_LEFT, nDateWidth); + m_listCtrl->InsertColumn(4, "Description", wxLIST_FORMAT_LEFT, 409 - nDateWidth); + m_listCtrl->InsertColumn(5, "Debit", wxLIST_FORMAT_RIGHT, 79); + m_listCtrl->InsertColumn(6, "Credit", wxLIST_FORMAT_RIGHT, 79); + + //m_listCtrlProductsSent->InsertColumn(0, "Category", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlProductsSent->InsertColumn(1, "Title", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlProductsSent->InsertColumn(2, "Description", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlProductsSent->InsertColumn(3, "Price", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlProductsSent->InsertColumn(4, "", wxLIST_FORMAT_LEFT, 100); + + //m_listCtrlOrdersSent->InsertColumn(0, "Time", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersSent->InsertColumn(1, "Price", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersSent->InsertColumn(2, "", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersSent->InsertColumn(3, "", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersSent->InsertColumn(4, "", wxLIST_FORMAT_LEFT, 100); + + //m_listCtrlOrdersReceived->InsertColumn(0, "Time", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersReceived->InsertColumn(1, "Price", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersReceived->InsertColumn(2, "Payment Status", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersReceived->InsertColumn(3, "", wxLIST_FORMAT_LEFT, 100); + //m_listCtrlOrdersReceived->InsertColumn(4, "", wxLIST_FORMAT_LEFT, 100); + + // Init status bar + int pnWidths[3] = { -100, 81, 286 }; + m_statusBar->SetFieldsCount(3, pnWidths); + + // Fill your address text box + vector vchPubKey; + if (CWalletDB("r").ReadDefaultKey(vchPubKey)) + m_textCtrlAddress->SetValue(PubKeyToAddress(vchPubKey)); + + // Fill listctrl with wallet transactions + RefreshListCtrl(); +} + +CMainFrame::~CMainFrame() +{ + pframeMain = NULL; +} + +void Shutdown(void* parg) +{ + static CCriticalSection cs_Shutdown; + CRITICAL_BLOCK(cs_Shutdown) + { + fShutdown = true; + nTransactionsUpdated++; + DBFlush(false); + StopNode(); + DBFlush(true); + + printf("Bitcoin exiting\n"); + exit(0); + } +} + +void CMainFrame::OnClose(wxCloseEvent& event) +{ + Destroy(); + _beginthread(Shutdown, 0, NULL); +} + +void CMainFrame::OnMouseEvents(wxMouseEvent& event) +{ + RandAddSeed(); + RAND_add(&event.m_x, sizeof(event.m_x), 0.25); + RAND_add(&event.m_y, sizeof(event.m_y), 0.25); +} + +void CMainFrame::OnListColBeginDrag(wxListEvent& event) +{ + // Hidden columns not resizeable + if (event.GetColumn() <= 1 && !fDebug) + event.Veto(); +} + +void CMainFrame::InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5, const wxString& str6) +{ + string str0 = strSort; + long nData = *(long*)&hashKey; + + if (fNew) + { + nIndex = m_listCtrl->InsertItem(0, str0); + } + else + { + if (nIndex == -1) + { + // Find item + while ((nIndex = m_listCtrl->FindItem(nIndex, nData)) != -1) + if (GetItemText(m_listCtrl, nIndex, 1) == hashKey.ToString()) + break; + if (nIndex == -1) + { + printf("CMainFrame::InsertLine : Couldn't find item to be updated\n"); + return; + } + } + + // If sort key changed, must delete and reinsert to make it relocate + if (GetItemText(m_listCtrl, nIndex, 0) != str0) + { + m_listCtrl->DeleteItem(nIndex); + nIndex = m_listCtrl->InsertItem(0, str0); + } + } + + m_listCtrl->SetItem(nIndex, 1, hashKey.ToString()); + m_listCtrl->SetItem(nIndex, 2, str2); + m_listCtrl->SetItem(nIndex, 3, str3); + m_listCtrl->SetItem(nIndex, 4, str4); + m_listCtrl->SetItem(nIndex, 5, str5); + m_listCtrl->SetItem(nIndex, 6, str6); + m_listCtrl->SetItemData(nIndex, nData); +} + +string FormatTxStatus(const CWalletTx& wtx) +{ + // Status + int nDepth = wtx.GetDepthInMainChain(); + if (!wtx.IsFinal()) + return strprintf("Open for %d blocks", nBestHeight - wtx.nLockTime); + else if (nDepth < 6) + return strprintf("%d/unconfirmed", nDepth); + else + return strprintf("%d blocks", nDepth); +} + +string SingleLine(const string& strIn) +{ + string strOut; + bool fOneSpace = false; + foreach(int c, strIn) + { + if (isspace(c)) + { + fOneSpace = true; + } + else if (c > ' ') + { + if (fOneSpace && !strOut.empty()) + strOut += ' '; + strOut += c; + fOneSpace = false; + } + } + return strOut; +} + +void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) +{ + int64 nTime = wtx.nTimeDisplayed = wtx.GetTxTime(); + int64 nCredit = wtx.GetCredit(); + int64 nDebit = wtx.GetDebit(); + int64 nNet = nCredit - nDebit; + uint256 hash = wtx.GetHash(); + string strStatus = FormatTxStatus(wtx); + map mapValue = wtx.mapValue; + + // Find the block the tx is in + CBlockIndex* pindex = NULL; + map::iterator mi = mapBlockIndex.find(wtx.hashBlock); + if (mi != mapBlockIndex.end()) + pindex = (*mi).second; + + // Sort order, unrecorded transactions sort to the top + string strSort = strprintf("%010d-%01d-%010u", + (pindex ? pindex->nHeight : INT_MAX), + (wtx.IsCoinBase() ? 1 : 0), + wtx.nTimeReceived); + + // Insert line + if (nNet > 0 || wtx.IsCoinBase()) + { + // + // Credit + // + string strDescription; + + if (wtx.IsCoinBase()) + { + // Coinbase + strDescription = "Generated"; + if (nCredit == 0) + { + int64 nUnmatured = 0; + foreach(const CTxOut& txout, wtx.vout) + nUnmatured += txout.GetCredit(); + if (wtx.IsInMainChain()) + strDescription += strprintf(" (%s matures in %d more blocks)", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); + else + strDescription += " (not accepted)"; + } + } + else if (!mapValue["from"].empty() || !mapValue["message"].empty()) + { + // Online transaction + if (!mapValue["from"].empty()) + strDescription += "From: " + mapValue["from"]; + if (!mapValue["message"].empty()) + { + if (!strDescription.empty()) + strDescription += " - "; + strDescription += mapValue["message"]; + } + } + else + { + // Offline transaction + foreach(const CTxOut& txout, wtx.vout) + { + if (txout.IsMine()) + { + vector vchPubKey; + if (ExtractPubKey(txout.scriptPubKey, true, vchPubKey)) + { + string strAddress = PubKeyToAddress(vchPubKey); + if (mapAddressBook.count(strAddress)) + { + //strDescription += "Received payment to "; + //strDescription += "Received with address "; + strDescription += "From: unknown, To: "; + strDescription += strAddress; + /// The labeling feature is just too confusing, so I hid it + /// by putting it at the end where it runs off the screen. + /// It can still be seen by widening the column, or in the + /// details dialog. + if (!mapAddressBook[strAddress].empty()) + strDescription += " (" + mapAddressBook[strAddress] + ")"; + } + } + break; + } + } + } + + InsertLine(fNew, nIndex, hash, strSort, + strStatus, + nTime ? DateTimeStr(nTime) : "", + SingleLine(strDescription), + "", + FormatMoney(nNet, true)); + } + else + { + bool fAllFromMe = true; + foreach(const CTxIn& txin, wtx.vin) + fAllFromMe = fAllFromMe && txin.IsMine(); + + bool fAllToMe = true; + foreach(const CTxOut& txout, wtx.vout) + fAllToMe = fAllToMe && txout.IsMine(); + + if (fAllFromMe && fAllToMe) + { + // Payment to self + int64 nValue = wtx.vout[0].nValue; + InsertLine(fNew, nIndex, hash, strSort, + strStatus, + nTime ? DateTimeStr(nTime) : "", + "Payment to yourself", + FormatMoney(nNet - nValue, true), + FormatMoney(nValue, true)); + } + else if (fAllFromMe) + { + // + // Debit + // + int64 nTxFee = nDebit - wtx.GetValueOut(); + for (int nOut = 0; nOut < wtx.vout.size(); nOut++) + { + const CTxOut& txout = wtx.vout[nOut]; + if (txout.IsMine()) + continue; + + string strAddress; + if (!mapValue["to"].empty()) + { + // Online transaction + strAddress = mapValue["to"]; + } + else + { + // Offline transaction + uint160 hash160; + if (ExtractHash160(txout.scriptPubKey, hash160)) + strAddress = Hash160ToAddress(hash160); + } + + string strDescription = "To: "; + if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) + strDescription += mapAddressBook[strAddress] + " "; + strDescription += strAddress; + if (!mapValue["message"].empty()) + { + if (!strDescription.empty()) + strDescription += " - "; + strDescription += mapValue["message"]; + } + + int64 nValue = txout.nValue; + if (nOut == 0 && nTxFee > 0) + nValue += nTxFee; + + InsertLine(fNew, nIndex, hash, strprintf("%s-%d", strSort.c_str(), nOut), + strStatus, + nTime ? DateTimeStr(nTime) : "", + SingleLine(strDescription), + FormatMoney(-nValue, true), + ""); + } + } + else + { + // + // Mixed debit transaction, can't break down payees + // + bool fAllMine = true; + foreach(const CTxOut& txout, wtx.vout) + fAllMine = fAllMine && txout.IsMine(); + foreach(const CTxIn& txin, wtx.vin) + fAllMine = fAllMine && txin.IsMine(); + + InsertLine(fNew, nIndex, hash, strSort, + strStatus, + nTime ? DateTimeStr(nTime) : "", + "", + FormatMoney(nNet, true), + ""); + } + } +} + +void CMainFrame::RefreshStatus() +{ + static int nLastTop; + int nTop = m_listCtrl->GetTopItem(); + if (nTop == nLastTop && pindexBestLast == pindexBest) + return; + + TRY_CRITICAL_BLOCK(cs_mapWallet) + { + int nStart = nTop; + int nEnd = min(nStart + 100, m_listCtrl->GetItemCount()); + if (pindexBestLast == pindexBest) + { + if (nStart >= nLastTop && nStart < nLastTop + 100) + nStart = nLastTop + 100; + if (nEnd >= nLastTop && nEnd < nLastTop + 100) + nEnd = nLastTop; + } + nLastTop = nTop; + pindexBestLast = pindexBest; + + for (int nIndex = nStart; nIndex < nEnd; nIndex++) + { + uint256 hash((string)GetItemText(m_listCtrl, nIndex, 1)); + map::iterator mi = mapWallet.find(hash); + if (mi == mapWallet.end()) + { + printf("CMainFrame::RefreshStatus() : tx not found in mapWallet\n"); + continue; + } + const CWalletTx& wtx = (*mi).second; + if (wtx.IsCoinBase() || wtx.GetTxTime() != wtx.nTimeDisplayed) + InsertTransaction(wtx, false, nIndex); + else + m_listCtrl->SetItem(nIndex, 2, FormatTxStatus(wtx)); + } + } +} + +void CMainFrame::RefreshListCtrl() +{ + fRefreshListCtrl = true; + ::wxWakeUpIdle(); +} + +void CMainFrame::OnIdle(wxIdleEvent& event) +{ + if (fRefreshListCtrl) + { + // Collect list of wallet transactions and sort newest first + bool fEntered = false; + vector > vSorted; + TRY_CRITICAL_BLOCK(cs_mapWallet) + { + printf("RefreshListCtrl starting\n"); + fEntered = true; + fRefreshListCtrl = false; + vWalletUpdated.clear(); + + // Do the newest transactions first + vSorted.reserve(mapWallet.size()); + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + const CWalletTx& wtx = (*it).second; + unsigned int nTime = UINT_MAX - wtx.GetTxTime(); + vSorted.push_back(make_pair(nTime, (*it).first)); + } + m_listCtrl->DeleteAllItems(); + } + if (!fEntered) + return; + + sort(vSorted.begin(), vSorted.end()); + + // Fill list control + for (int i = 0; i < vSorted.size();) + { + if (fShutdown) + return; + bool fEntered = false; + TRY_CRITICAL_BLOCK(cs_mapWallet) + { + fEntered = true; + uint256& hash = vSorted[i++].second; + map::iterator mi = mapWallet.find(hash); + if (mi != mapWallet.end()) + InsertTransaction((*mi).second, true); + } + if (!fEntered || i == 100 || i % 500 == 0) + wxYield(); + } + + printf("RefreshListCtrl done\n"); + } + else + { + // Check for time updates + static int64 nLastTime; + if (GetTime() > nLastTime + 30) + { + TRY_CRITICAL_BLOCK(cs_mapWallet) + { + nLastTime = GetTime(); + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + CWalletTx& wtx = (*it).second; + if (wtx.nTimeDisplayed && wtx.nTimeDisplayed != wtx.GetTxTime()) + InsertTransaction(wtx, false); + } + } + } + } +} + +void CMainFrame::OnPaint(wxPaintEvent& event) +{ + event.Skip(); +} + +void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) +{ + // Update listctrl contents + if (!vWalletUpdated.empty()) + { + TRY_CRITICAL_BLOCK(cs_mapWallet) + { + pair item; + foreach(item, vWalletUpdated) + { + bool fNew = item.second; + map::iterator mi = mapWallet.find(item.first); + if (mi != mapWallet.end()) + { + printf("vWalletUpdated: %s %s\n", (*mi).second.GetHash().ToString().substr(0,6).c_str(), fNew ? "new" : ""); + InsertTransaction((*mi).second, fNew); + } + } + m_listCtrl->ScrollList(0, INT_MAX); + vWalletUpdated.clear(); + } + } + + // Update status column of visible items only + RefreshStatus(); + + // Update status bar + string strGen = ""; + if (fGenerateBitcoins) + strGen = " Generating"; + if (fGenerateBitcoins && vNodes.empty()) + strGen = "(not connected)"; + m_statusBar->SetStatusText(strGen, 1); + + string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, m_listCtrl->GetItemCount()); + m_statusBar->SetStatusText(strStatus, 2); + + // Balance total + TRY_CRITICAL_BLOCK(cs_mapWallet) + m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); + + m_listCtrl->OnPaint(event); +} + +void CrossThreadCall(wxCommandEvent& event) +{ + if (pframeMain) + pframeMain->GetEventHandler()->AddPendingEvent(event); +} + +void CrossThreadCall(int nID, void* pdata) +{ + wxCommandEvent event; + event.SetInt(nID); + event.SetClientData(pdata); + if (pframeMain) + pframeMain->GetEventHandler()->AddPendingEvent(event); +} + +void CMainFrame::OnCrossThreadCall(wxCommandEvent& event) +{ + void* pdata = event.GetClientData(); + switch (event.GetInt()) + { + case UICALL_ADDORDER: + { + break; + } + + case UICALL_UPDATEORDER: + { + break; + } + } +} + +void CMainFrame::OnMenuFileExit(wxCommandEvent& event) +{ + Close(true); +} + +void CMainFrame::OnMenuOptionsGenerate(wxCommandEvent& event) +{ + fGenerateBitcoins = event.IsChecked(); + nTransactionsUpdated++; + CWalletDB().WriteSetting("fGenerateBitcoins", fGenerateBitcoins); + + if (fGenerateBitcoins) + if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) + printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); + + Refresh(); + wxPaintEvent eventPaint; + AddPendingEvent(eventPaint); +} + +void CMainFrame::OnMenuOptionsChangeYourAddress(wxCommandEvent& event) +{ + OnButtonChange(event); +} + +void CMainFrame::OnMenuOptionsOptions(wxCommandEvent& event) +{ + COptionsDialog dialog(this); + dialog.ShowModal(); +} + +void CMainFrame::OnMenuHelpAbout(wxCommandEvent& event) +{ + CAboutDialog dialog(this); + dialog.ShowModal(); +} + +void CMainFrame::OnButtonSend(wxCommandEvent& event) +{ + /// debug test + if (fRandSendTest) + { + RandSend(); + return; + } + + // Toolbar: Send + CSendDialog dialog(this); + dialog.ShowModal(); +} + +void CMainFrame::OnButtonAddressBook(wxCommandEvent& event) +{ + // Toolbar: Address Book + CAddressBookDialog dialogAddr(this, "", false); + if (dialogAddr.ShowModal() == 2) + { + // Send + CSendDialog dialogSend(this, dialogAddr.GetAddress()); + dialogSend.ShowModal(); + } +} + +void CMainFrame::OnSetFocusAddress(wxFocusEvent& event) +{ + // Automatically select-all when entering window + m_textCtrlAddress->SetSelection(-1, -1); + fOnSetFocusAddress = true; + event.Skip(); +} + +void CMainFrame::OnMouseEventsAddress(wxMouseEvent& event) +{ + if (fOnSetFocusAddress) + m_textCtrlAddress->SetSelection(-1, -1); + fOnSetFocusAddress = false; + event.Skip(); +} + +void CMainFrame::OnButtonCopy(wxCommandEvent& event) +{ + // Copy address box to clipboard + if (wxTheClipboard->Open()) + { + wxTheClipboard->SetData(new wxTextDataObject(m_textCtrlAddress->GetValue())); + wxTheClipboard->Close(); + } +} + +void CMainFrame::OnButtonChange(wxCommandEvent& event) +{ + CYourAddressDialog dialog(this, string(m_textCtrlAddress->GetValue())); + if (!dialog.ShowModal()) + return; + string strAddress = (string)dialog.GetAddress(); + if (strAddress != m_textCtrlAddress->GetValue()) + { + uint160 hash160; + if (!AddressToHash160(strAddress, hash160)) + return; + if (!mapPubKeys.count(hash160)) + return; + CWalletDB().WriteDefaultKey(mapPubKeys[hash160]); + m_textCtrlAddress->SetValue(strAddress); + } +} + +void CMainFrame::OnListItemActivatedAllTransactions(wxListEvent& event) +{ + uint256 hash((string)GetItemText(m_listCtrl, event.GetIndex(), 1)); + CWalletTx wtx; + CRITICAL_BLOCK(cs_mapWallet) + { + map::iterator mi = mapWallet.find(hash); + if (mi == mapWallet.end()) + { + printf("CMainFrame::OnListItemActivatedAllTransactions() : tx not found in mapWallet\n"); + return; + } + wtx = (*mi).second; + } + CTxDetailsDialog dialog(this, wtx); + dialog.ShowModal(); + //CTxDetailsDialog* pdialog = new CTxDetailsDialog(this, wtx); + //pdialog->Show(); +} + +void CMainFrame::OnListItemActivatedProductsSent(wxListEvent& event) +{ + CProduct& product = *(CProduct*)event.GetItem().GetData(); + CEditProductDialog* pdialog = new CEditProductDialog(this); + pdialog->SetProduct(product); + pdialog->Show(); +} + +void CMainFrame::OnListItemActivatedOrdersSent(wxListEvent& event) +{ + CWalletTx& order = *(CWalletTx*)event.GetItem().GetData(); + CViewOrderDialog* pdialog = new CViewOrderDialog(this, order, false); + pdialog->Show(); +} + +void CMainFrame::OnListItemActivatedOrdersReceived(wxListEvent& event) +{ + CWalletTx& order = *(CWalletTx*)event.GetItem().GetData(); + CViewOrderDialog* pdialog = new CViewOrderDialog(this, order, true); + pdialog->Show(); +} + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CTxDetailsDialog +// + +CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetailsDialogBase(parent) +{ + string strHTML; + strHTML.reserve(4000); + strHTML += ""; + + int64 nTime = wtx.GetTxTime(); + int64 nCredit = wtx.GetCredit(); + int64 nDebit = wtx.GetDebit(); + int64 nNet = nCredit - nDebit; + + + + strHTML += "Status: " + FormatTxStatus(wtx) + "
"; + strHTML += "Date: " + (nTime ? DateTimeStr(nTime) : "") + "
"; + + + // + // From + // + if (wtx.IsCoinBase()) + { + strHTML += "Source: Generated
"; + } + else if (!wtx.mapValue["from"].empty()) + { + // Online transaction + if (!wtx.mapValue["from"].empty()) + strHTML += "From: " + HtmlEscape(wtx.mapValue["from"]) + "
"; + } + else + { + // Offline transaction + if (nNet > 0) + { + // Credit + foreach(const CTxOut& txout, wtx.vout) + { + if (txout.IsMine()) + { + vector vchPubKey; + if (ExtractPubKey(txout.scriptPubKey, true, vchPubKey)) + { + string strAddress = PubKeyToAddress(vchPubKey); + if (mapAddressBook.count(strAddress)) + { + strHTML += "From: unknown
"; + strHTML += "To: "; + strHTML += HtmlEscape(strAddress); + if (!mapAddressBook[strAddress].empty()) + strHTML += " (yours, label: " + mapAddressBook[strAddress] + ")"; + else + strHTML += " (yours)"; + strHTML += "
"; + } + } + break; + } + } + } + } + + + // + // To + // + string strAddress; + if (!wtx.mapValue["to"].empty()) + { + // Online transaction + strAddress = wtx.mapValue["to"]; + strHTML += "To: "; + if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) + strHTML += mapAddressBook[strAddress] + " "; + strHTML += HtmlEscape(strAddress) + "
"; + } + + + // + // Amount + // + if (wtx.IsCoinBase() && nCredit == 0) + { + // + // Coinbase + // + int64 nUnmatured = 0; + foreach(const CTxOut& txout, wtx.vout) + nUnmatured += txout.GetCredit(); + if (wtx.IsInMainChain()) + strHTML += strprintf("Credit: (%s matures in %d more blocks)
", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); + else + strHTML += "Credit: (not accepted)
"; + } + else if (nNet > 0) + { + // + // Credit + // + strHTML += "Credit: " + FormatMoney(nNet) + "
"; + } + else + { + bool fAllFromMe = true; + foreach(const CTxIn& txin, wtx.vin) + fAllFromMe = fAllFromMe && txin.IsMine(); + + bool fAllToMe = true; + foreach(const CTxOut& txout, wtx.vout) + fAllToMe = fAllToMe && txout.IsMine(); + + if (fAllFromMe) + { + // + // Debit + // + foreach(const CTxOut& txout, wtx.vout) + { + if (txout.IsMine()) + continue; + + if (wtx.mapValue["to"].empty()) + { + // Offline transaction + uint160 hash160; + if (ExtractHash160(txout.scriptPubKey, hash160)) + { + string strAddress = Hash160ToAddress(hash160); + strHTML += "To: "; + if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) + strHTML += mapAddressBook[strAddress] + " "; + strHTML += strAddress; + strHTML += "
"; + } + } + + strHTML += "Debit: " + FormatMoney(-txout.nValue) + "
"; + } + + if (fAllToMe) + { + // Payment to self + int64 nValue = wtx.vout[0].nValue; + strHTML += "Debit: " + FormatMoney(-nValue) + "
"; + strHTML += "Credit: " + FormatMoney(nValue) + "
"; + } + + int64 nTxFee = nDebit - wtx.GetValueOut(); + if (nTxFee > 0) + strHTML += "Transaction fee: " + FormatMoney(-nTxFee) + "
"; + } + else + { + // + // Mixed debit transaction + // + foreach(const CTxIn& txin, wtx.vin) + if (txin.IsMine()) + strHTML += "Debit: " + FormatMoney(-txin.GetDebit()) + "
"; + foreach(const CTxOut& txout, wtx.vout) + if (txout.IsMine()) + strHTML += "Credit: " + FormatMoney(txout.GetCredit()) + "
"; + } + } + + strHTML += "Net amount: " + FormatMoney(nNet, true) + "
"; + + + // + // Message + // + if (!wtx.mapValue["message"].empty()) + strHTML += "
Message:
" + HtmlEscape(wtx.mapValue["message"], true) + "
"; + + if (wtx.IsCoinBase()) + strHTML += "
Generated coins must wait 120 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, it will change to \"not accepted\" and not be spendable. This may occasionally happen if another node generates a block within a few seconds of yours.
"; + + + // + // Debug view + // + if (fDebug) + { + strHTML += "

debug print

"; + foreach(const CTxIn& txin, wtx.vin) + if (txin.IsMine()) + strHTML += "Debit: " + FormatMoney(-txin.GetDebit()) + "
"; + foreach(const CTxOut& txout, wtx.vout) + if (txout.IsMine()) + strHTML += "Credit: " + FormatMoney(txout.GetCredit()) + "
"; + + strHTML += "Inputs:
"; + CRITICAL_BLOCK(cs_mapWallet) + { + foreach(const CTxIn& txin, wtx.vin) + { + COutPoint prevout = txin.prevout; + map::iterator mi = mapWallet.find(prevout.hash); + if (mi != mapWallet.end()) + { + const CWalletTx& prev = (*mi).second; + if (prevout.n < prev.vout.size()) + { + strHTML += HtmlEscape(prev.ToString(), true); + strHTML += "    " + FormatTxStatus(prev) + ", "; + strHTML = strHTML + "IsMine=" + (prev.vout[prevout.n].IsMine() ? "true" : "false") + "
"; + } + } + } + } + + strHTML += "


Transaction:
"; + strHTML += HtmlEscape(wtx.ToString(), true); + } + + + + strHTML += "
"; + string(strHTML.begin(), strHTML.end()).swap(strHTML); + m_htmlWin->SetPage(strHTML); + m_buttonOK->SetFocus(); +} + +void CTxDetailsDialog::OnButtonOK(wxCommandEvent& event) +{ + Close(); + //Destroy(); +} + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// COptionsDialog +// + +COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) +{ + m_textCtrlTransactionFee->SetValue(FormatMoney(nTransactionFee)); + m_buttonOK->SetFocus(); +} + +void COptionsDialog::OnKillFocusTransactionFee(wxFocusEvent& event) +{ + int64 nTmp = nTransactionFee; + ParseMoney(m_textCtrlTransactionFee->GetValue(), nTmp); + m_textCtrlTransactionFee->SetValue(FormatMoney(nTmp)); +} + +void COptionsDialog::OnButtonOK(wxCommandEvent& event) +{ + // nTransactionFee + int64 nPrevTransactionFee = nTransactionFee; + if (ParseMoney(m_textCtrlTransactionFee->GetValue(), nTransactionFee) && nTransactionFee != nPrevTransactionFee) + CWalletDB().WriteSetting("nTransactionFee", nTransactionFee); + + Close(); +} + +void COptionsDialog::OnButtonCancel(wxCommandEvent& event) +{ + Close(); +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CAboutDialog +// + +CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) +{ + m_staticTextVersion->SetLabel(strprintf("version 0.%d.%d Alpha", VERSION/100, VERSION%100)); + + // Workaround until upgrade to wxWidgets supporting UTF-8 + wxString str = m_staticTextMain->GetLabel(); + if (str.Find('Â') != wxNOT_FOUND) + str.Remove(str.Find('Â'), 1); + m_staticTextMain->SetLabel(str); +} + +void CAboutDialog::OnButtonOK(wxCommandEvent& event) +{ + Close(); +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CSendDialog +// + +CSendDialog::CSendDialog(wxWindow* parent, const wxString& strAddress) : CSendDialogBase(parent) +{ + // Init + m_textCtrlAddress->SetValue(strAddress); + m_choiceTransferType->SetSelection(0); + m_bitmapCheckMark->Show(false); + //// todo: should add a display of your balance for convenience + + // Set Icon + wxBitmap bmpSend(wxT("send16"), wxBITMAP_TYPE_RESOURCE); + bmpSend.SetMask(new wxMask(wxBitmap(wxT("send16masknoshadow"), wxBITMAP_TYPE_RESOURCE))); + wxIcon iconSend; + iconSend.CopyFromBitmap(bmpSend); + SetIcon(iconSend); + + wxCommandEvent event; + OnTextAddress(event); + + // Fixup the tab order + m_buttonPaste->MoveAfterInTabOrder(m_buttonCancel); + m_buttonAddress->MoveAfterInTabOrder(m_buttonPaste); + this->Layout(); +} + +void CSendDialog::OnTextAddress(wxCommandEvent& event) +{ + // Check mark + bool fBitcoinAddress = IsValidBitcoinAddress(m_textCtrlAddress->GetValue()); + m_bitmapCheckMark->Show(fBitcoinAddress); + + // Grey out message if bitcoin address + bool fEnable = !fBitcoinAddress; + m_staticTextFrom->Enable(fEnable); + m_textCtrlFrom->Enable(fEnable); + m_staticTextMessage->Enable(fEnable); + m_textCtrlMessage->Enable(fEnable); + m_textCtrlMessage->SetBackgroundColour(wxSystemSettings::GetColour(fEnable ? wxSYS_COLOUR_WINDOW : wxSYS_COLOUR_BTNFACE)); +} + +void CSendDialog::OnKillFocusAmount(wxFocusEvent& event) +{ + // Reformat the amount + if (m_textCtrlAmount->GetValue().Trim().empty()) + return; + int64 nTmp; + if (ParseMoney(m_textCtrlAmount->GetValue(), nTmp)) + m_textCtrlAmount->SetValue(FormatMoney(nTmp)); +} + +void CSendDialog::OnButtonAddressBook(wxCommandEvent& event) +{ + // Open address book + CAddressBookDialog dialog(this, m_textCtrlAddress->GetValue(), true); + if (dialog.ShowModal()) + m_textCtrlAddress->SetValue(dialog.GetAddress()); +} + +void CSendDialog::OnButtonPaste(wxCommandEvent& event) +{ + // Copy clipboard to address box + if (wxTheClipboard->Open()) + { + if (wxTheClipboard->IsSupported(wxDF_TEXT)) + { + wxTextDataObject data; + wxTheClipboard->GetData(data); + m_textCtrlAddress->SetValue(data.GetText()); + } + wxTheClipboard->Close(); + } +} + +void CSendDialog::OnButtonSend(wxCommandEvent& event) +{ + CWalletTx wtx; + string strAddress = (string)m_textCtrlAddress->GetValue(); + + // Parse amount + int64 nValue = 0; + if (!ParseMoney(m_textCtrlAmount->GetValue(), nValue) || nValue <= 0) + { + wxMessageBox("Error in amount ", "Send Coins"); + return; + } + if (nValue > GetBalance()) + { + wxMessageBox("Amount exceeds your balance ", "Send Coins"); + return; + } + if (nValue + nTransactionFee > GetBalance()) + { + wxMessageBox(string("Total exceeds your balance when the ") + FormatMoney(nTransactionFee) + " transaction fee is included ", "Send Coins"); + return; + } + + // Parse bitcoin address + uint160 hash160; + bool fBitcoinAddress = AddressToHash160(strAddress, hash160); + + if (fBitcoinAddress) + { + // Send to bitcoin address + CScript scriptPubKey; + scriptPubKey << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; + + if (!SendMoney(scriptPubKey, nValue, wtx)) + return; + + wxMessageBox("Payment sent ", "Sending..."); + } + else + { + // Parse IP address + CAddress addr(strAddress.c_str()); + if (addr.ip == 0) + { + wxMessageBox("Invalid address ", "Send Coins"); + return; + } + + // Message + wtx.mapValue["to"] = strAddress; + wtx.mapValue["from"] = m_textCtrlFrom->GetValue(); + wtx.mapValue["message"] = m_textCtrlMessage->GetValue(); + + // Send to IP address + CSendingDialog* pdialog = new CSendingDialog(this, addr, nValue, wtx); + if (!pdialog->ShowModal()) + return; + } + + if (!mapAddressBook.count(strAddress)) + SetAddressBookName(strAddress, ""); + + EndModal(true); +} + +void CSendDialog::OnButtonCancel(wxCommandEvent& event) +{ + // Cancel + EndModal(false); +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CSendingDialog +// + +CSendingDialog::CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 nPriceIn, const CWalletTx& wtxIn) : CSendingDialogBase(NULL) // we have to give null so parent can't destroy us +{ + addr = addrIn; + nPrice = nPriceIn; + wtx = wtxIn; + start = wxDateTime::UNow(); + strStatus = ""; + fCanCancel = true; + fAbort = false; + fSuccess = false; + fUIDone = false; + fWorkDone = false; + + SetTitle(strprintf("Sending %s to %s...", FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); + m_textCtrlStatus->SetValue(""); + + _beginthread(SendingDialogStartTransfer, 0, this); +} + +CSendingDialog::~CSendingDialog() +{ + printf("~CSendingDialog()\n"); +} + +void CSendingDialog::Close() +{ + // Last one out turn out the lights. + // fWorkDone signals that work side is done and UI thread should call destroy. + // fUIDone signals that UI window has closed and work thread should call destroy. + // This allows the window to disappear and end modality when cancelled + // without making the user wait for ConnectNode to return. The dialog object + // hangs around in the background until the work thread exits. + if (IsModal()) + EndModal(fSuccess); + else + Show(false); + if (fWorkDone) + Destroy(); + else + fUIDone = true; +} + +void CSendingDialog::OnClose(wxCloseEvent& event) +{ + if (!event.CanVeto() || fWorkDone || fAbort || !fCanCancel) + { + Close(); + } + else + { + event.Veto(); + wxCommandEvent cmdevent; + OnButtonCancel(cmdevent); + } +} + +void CSendingDialog::OnButtonOK(wxCommandEvent& event) +{ + if (fWorkDone) + Close(); +} + +void CSendingDialog::OnButtonCancel(wxCommandEvent& event) +{ + if (fCanCancel) + fAbort = true; +} + +void CSendingDialog::OnPaint(wxPaintEvent& event) +{ + if (strStatus.size() > 130) + m_textCtrlStatus->SetValue(string("\n") + strStatus); + else + m_textCtrlStatus->SetValue(string("\n\n") + strStatus); + m_staticTextSending->SetFocus(); + if (!fCanCancel) + m_buttonCancel->Enable(false); + if (fWorkDone) + { + m_buttonOK->Enable(true); + m_buttonOK->SetFocus(); + m_buttonCancel->Enable(false); + } + if (fAbort && fCanCancel && IsShown()) + { + strStatus = "CANCELLED"; + m_buttonOK->Enable(true); + m_buttonOK->SetFocus(); + m_buttonCancel->Enable(false); + m_buttonCancel->SetLabel("Cancelled"); + Close(); + wxMessageBox("Transfer cancelled ", "Sending...", wxOK, this); + } + event.Skip(); + + /// debug test + if (fRandSendTest && fWorkDone && fSuccess) + { + Close(); + Sleep(1000); + RandSend(); + } +} + + +// +// Everything from here on is not in the UI thread and must only communicate +// with the rest of the dialog through variables and calling repaint. +// + +void CSendingDialog::Repaint() +{ + Refresh(); + wxPaintEvent event; + AddPendingEvent(event); +} + +bool CSendingDialog::Status() +{ + if (fUIDone) + { + Destroy(); + return false; + } + if (fAbort && fCanCancel) + { + strStatus = "CANCELLED"; + Repaint(); + fWorkDone = true; + return false; + } + return true; +} + +bool CSendingDialog::Status(const string& str) +{ + if (!Status()) + return false; + strStatus = str; + Repaint(); + return true; +} + +bool CSendingDialog::Error(const string& str) +{ + fCanCancel = false; + fWorkDone = true; + Status(string("Error: ") + str); + return false; +} + +void SendingDialogStartTransfer(void* parg) +{ + ((CSendingDialog*)parg)->StartTransfer(); +} + +void CSendingDialog::StartTransfer() +{ + // Make sure we have enough money + if (nPrice + nTransactionFee > GetBalance()) + { + Error("You don't have enough money"); + return; + } + + // We may have connected already for product details + if (!Status("Connecting...")) + return; + CNode* pnode = ConnectNode(addr, 5 * 60); + if (!pnode) + { + Error("Unable to connect"); + return; + } + + // Send order to seller, with response going to OnReply2 via event handler + if (!Status("Requesting public key...")) + return; + pnode->PushRequest("checkorder", wtx, SendingDialogOnReply2, this); +} + +void SendingDialogOnReply2(void* parg, CDataStream& vRecv) +{ + ((CSendingDialog*)parg)->OnReply2(vRecv); +} + +void CSendingDialog::OnReply2(CDataStream& vRecv) +{ + if (!Status("Received public key...")) + return; + + CScript scriptPubKey; + int nRet; + try + { + vRecv >> nRet; + if (nRet > 0) + { + string strMessage; + vRecv >> strMessage; + Error("Transfer was not accepted"); + //// todo: enlarge the window and enable a hidden white box to put seller's message + return; + } + vRecv >> scriptPubKey; + } + catch (...) + { + //// what do we want to do about this? + Error("Invalid response received"); + return; + } + + // Should already be connected + CNode* pnode = ConnectNode(addr, 5 * 60); + if (!pnode) + { + Error("Lost connection"); + return; + } + + // Pause to give the user a chance to cancel + while (wxDateTime::UNow() < start + wxTimeSpan(0, 0, 0, 2 * 1000)) + { + Sleep(200); + if (!Status()) + return; + } + + CRITICAL_BLOCK(cs_main) + { + // Pay + if (!Status("Creating transaction...")) + return; + if (nPrice + nTransactionFee > GetBalance()) + { + Error("You don't have enough money"); + return; + } + int64 nFeeRequired; + if (!CreateTransaction(scriptPubKey, nPrice, wtx, nFeeRequired)) + { + if (nPrice + nFeeRequired > GetBalance()) + Error(strprintf("This is an oversized transaction that requires a transaction fee of %s", FormatMoney(nFeeRequired).c_str())); + else + Error("Transaction creation failed"); + return; + } + + // Last chance to cancel + Sleep(50); + if (!Status()) + return; + fCanCancel = false; + if (fAbort) + { + fCanCancel = true; + if (!Status()) + return; + fCanCancel = false; + } + if (!Status("Sending payment...")) + return; + + // Commit + if (!CommitTransactionSpent(wtx)) + { + Error("Error finalizing payment"); + return; + } + + // Send payment tx to seller, with response going to OnReply3 via event handler + pnode->PushRequest("submitorder", wtx, SendingDialogOnReply3, this); + + // Accept and broadcast transaction + if (!wtx.AcceptTransaction()) + printf("ERROR: CSendingDialog : wtxNew.AcceptTransaction() %s failed\n", wtx.GetHash().ToString().c_str()); + wtx.RelayWalletTransaction(); + + Status("Waiting for confirmation..."); + MainFrameRepaint(); + } +} + +void SendingDialogOnReply3(void* parg, CDataStream& vRecv) +{ + ((CSendingDialog*)parg)->OnReply3(vRecv); +} + +void CSendingDialog::OnReply3(CDataStream& vRecv) +{ + int nRet; + try + { + vRecv >> nRet; + if (nRet > 0) + { + Error("The payment was sent, but the recipient was unable to verify it.\n" + "The transaction is recorded and will credit to the recipient if it is valid,\n" + "but without comment information."); + return; + } + } + catch (...) + { + //// what do we want to do about this? + Error("Payment was sent, but an invalid response was received"); + return; + } + + fSuccess = true; + fWorkDone = true; + Status("Payment completed"); +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CYourAddressDialog +// + +CYourAddressDialog::CYourAddressDialog(wxWindow* parent, const string& strInitSelected) : CYourAddressDialogBase(parent) +{ + // Init column headers + m_listCtrl->InsertColumn(0, "Label", wxLIST_FORMAT_LEFT, 200); + m_listCtrl->InsertColumn(1, "Bitcoin Address", wxLIST_FORMAT_LEFT, 350); + m_listCtrl->SetFocus(); + + // Fill listctrl with address book data + CRITICAL_BLOCK(cs_mapKeys) + { + foreach(const PAIRTYPE(string, string)& item, mapAddressBook) + { + string strAddress = item.first; + string strName = item.second; + uint160 hash160; + bool fMine = (AddressToHash160(strAddress, hash160) && mapPubKeys.count(hash160)); + if (fMine) + { + int nIndex = InsertLine(m_listCtrl, strName, strAddress); + if (strAddress == strInitSelected) + m_listCtrl->SetItemState(nIndex, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED); + } + } + } +} + +wxString CYourAddressDialog::GetAddress() +{ + int nIndex = GetSelection(m_listCtrl); + if (nIndex == -1) + return ""; + return GetItemText(m_listCtrl, nIndex, 1); +} + +void CYourAddressDialog::OnListEndLabelEdit(wxListEvent& event) +{ + // Update address book with edited name + if (event.IsEditCancelled()) + return; + string strAddress = (string)GetItemText(m_listCtrl, event.GetIndex(), 1); + SetAddressBookName(strAddress, string(event.GetText())); + pframeMain->RefreshListCtrl(); +} + +void CYourAddressDialog::OnListItemSelected(wxListEvent& event) +{ +} + +void CYourAddressDialog::OnListItemActivated(wxListEvent& event) +{ + // Doubleclick edits item + wxCommandEvent event2; + OnButtonRename(event2); +} + +void CYourAddressDialog::OnButtonRename(wxCommandEvent& event) +{ + // Ask new name + int nIndex = GetSelection(m_listCtrl); + if (nIndex == -1) + return; + string strName = (string)m_listCtrl->GetItemText(nIndex); + string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); + CGetTextFromUserDialog dialog(this, "Edit Address Label", "New Label", strName); + if (!dialog.ShowModal()) + return; + strName = dialog.GetValue(); + + // Change name + SetAddressBookName(strAddress, strName); + m_listCtrl->SetItemText(nIndex, strName); + pframeMain->RefreshListCtrl(); +} + +void CYourAddressDialog::OnButtonNew(wxCommandEvent& event) +{ + // Ask name + CGetTextFromUserDialog dialog(this, "New Bitcoin Address", "Label", ""); + if (!dialog.ShowModal()) + return; + string strName = dialog.GetValue(); + + // Generate new key + string strAddress = PubKeyToAddress(GenerateNewKey()); + SetAddressBookName(strAddress, strName); + + // Add to list and select it + int nIndex = InsertLine(m_listCtrl, strName, strAddress); + SetSelection(m_listCtrl, nIndex); + m_listCtrl->SetFocus(); +} + +void CYourAddressDialog::OnButtonCopy(wxCommandEvent& event) +{ + // Copy address box to clipboard + if (wxTheClipboard->Open()) + { + wxTheClipboard->SetData(new wxTextDataObject(GetAddress())); + wxTheClipboard->Close(); + } +} + +void CYourAddressDialog::OnButtonOK(wxCommandEvent& event) +{ + // OK + EndModal(true); +} + +void CYourAddressDialog::OnButtonCancel(wxCommandEvent& event) +{ + // Cancel + EndModal(false); +} + +void CYourAddressDialog::OnClose(wxCloseEvent& event) +{ + // Close + EndModal(false); +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CAddressBookDialog +// + +CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInitSelected, bool fSendingIn) : CAddressBookDialogBase(parent) +{ + fSending = fSendingIn; + if (!fSending) + m_buttonCancel->Show(false); + + // Init column headers + m_listCtrl->InsertColumn(0, "Name", wxLIST_FORMAT_LEFT, 200); + m_listCtrl->InsertColumn(1, "Address", wxLIST_FORMAT_LEFT, 350); + m_listCtrl->SetFocus(); + + // Set Icon + wxBitmap bmpAddressBook(wxT("addressbook16"), wxBITMAP_TYPE_RESOURCE); + bmpAddressBook.SetMask(new wxMask(wxBitmap(wxT("addressbook16mask"), wxBITMAP_TYPE_RESOURCE))); + wxIcon iconAddressBook; + iconAddressBook.CopyFromBitmap(bmpAddressBook); + SetIcon(iconAddressBook); + + // Fill listctrl with address book data + CRITICAL_BLOCK(cs_mapKeys) + { + foreach(const PAIRTYPE(string, string)& item, mapAddressBook) + { + string strAddress = item.first; + string strName = item.second; + uint160 hash160; + bool fMine = (AddressToHash160(strAddress, hash160) && mapPubKeys.count(hash160)); + if (!fMine) + { + int nIndex = InsertLine(m_listCtrl, strName, strAddress); + if (strAddress == strInitSelected) + m_listCtrl->SetItemState(nIndex, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED); + } + } + } +} + +wxString CAddressBookDialog::GetAddress() +{ + int nIndex = GetSelection(m_listCtrl); + if (nIndex == -1) + return ""; + return GetItemText(m_listCtrl, nIndex, 1); +} + +void CAddressBookDialog::OnListEndLabelEdit(wxListEvent& event) +{ + // Update address book with edited name + if (event.IsEditCancelled()) + return; + string strAddress = (string)GetItemText(m_listCtrl, event.GetIndex(), 1); + SetAddressBookName(strAddress, string(event.GetText())); + pframeMain->RefreshListCtrl(); +} + +void CAddressBookDialog::OnListItemSelected(wxListEvent& event) +{ +} + +void CAddressBookDialog::OnListItemActivated(wxListEvent& event) +{ + if (fSending) + { + // Doubleclick returns selection + EndModal(GetAddress() != "" ? 2 : 0); + } + else + { + // Doubleclick edits item + wxCommandEvent event2; + OnButtonEdit(event2); + } +} + +bool CAddressBookDialog::CheckIfMine(const string& strAddress, const string& strTitle) +{ + uint160 hash160; + bool fMine = (AddressToHash160(strAddress, hash160) && mapPubKeys.count(hash160)); + if (fMine) + wxMessageBox("This is one of your own addresses for receiving payments and cannot be entered in the address book. ", strTitle); + return fMine; +} + +void CAddressBookDialog::OnButtonEdit(wxCommandEvent& event) +{ + // Ask new name + int nIndex = GetSelection(m_listCtrl); + if (nIndex == -1) + return; + string strName = (string)m_listCtrl->GetItemText(nIndex); + string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); + string strAddressOrg = strAddress; + do + { + CGetTextFromUserDialog dialog(this, "Edit Address", "Name", strName, "Address", strAddress); + if (!dialog.ShowModal()) + return; + strName = dialog.GetValue1(); + strAddress = dialog.GetValue2(); + } + while (CheckIfMine(strAddress, "Edit Address")); + + // Change name + if (strAddress != strAddressOrg) + CWalletDB().EraseName(strAddressOrg); + SetAddressBookName(strAddress, strName); + m_listCtrl->SetItem(nIndex, 1, strAddress); + m_listCtrl->SetItemText(nIndex, strName); + pframeMain->RefreshListCtrl(); +} + +void CAddressBookDialog::OnButtonNew(wxCommandEvent& event) +{ + // Ask name + string strName; + string strAddress; + do + { + CGetTextFromUserDialog dialog(this, "New Address", "Name", strName, "Address", strAddress); + if (!dialog.ShowModal()) + return; + strName = dialog.GetValue1(); + strAddress = dialog.GetValue2(); + } + while (CheckIfMine(strAddress, "New Address")); + + // Add to list and select it + SetAddressBookName(strAddress, strName); + int nIndex = InsertLine(m_listCtrl, strName, strAddress); + SetSelection(m_listCtrl, nIndex); + m_listCtrl->SetFocus(); + pframeMain->RefreshListCtrl(); +} + +void CAddressBookDialog::OnButtonDelete(wxCommandEvent& event) +{ + for (int nIndex = m_listCtrl->GetItemCount()-1; nIndex >= 0; nIndex--) + { + if (m_listCtrl->GetItemState(nIndex, wxLIST_STATE_SELECTED)) + { + string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); + CWalletDB().EraseName(strAddress); + m_listCtrl->DeleteItem(nIndex); + } + } + pframeMain->RefreshListCtrl(); +} + +void CAddressBookDialog::OnButtonCopy(wxCommandEvent& event) +{ + // Copy address box to clipboard + if (wxTheClipboard->Open()) + { + wxTheClipboard->SetData(new wxTextDataObject(GetAddress())); + wxTheClipboard->Close(); + } +} + +void CAddressBookDialog::OnButtonOK(wxCommandEvent& event) +{ + // OK + EndModal(GetAddress() != "" ? 1 : 0); +} + +void CAddressBookDialog::OnButtonCancel(wxCommandEvent& event) +{ + // Cancel + EndModal(0); +} + +void CAddressBookDialog::OnClose(wxCloseEvent& event) +{ + // Close + EndModal(0); +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CProductsDialog +// + +bool CompareIntStringPairBestFirst(const pair& item1, const pair& item2) +{ + return (item1.first > item2.first); +} + +CProductsDialog::CProductsDialog(wxWindow* parent) : CProductsDialogBase(parent) +{ + // Init column headers + m_listCtrl->InsertColumn(0, "Title", wxLIST_FORMAT_LEFT, 200); + m_listCtrl->InsertColumn(1, "Price", wxLIST_FORMAT_LEFT, 80); + m_listCtrl->InsertColumn(2, "Seller", wxLIST_FORMAT_LEFT, 80); + m_listCtrl->InsertColumn(3, "Stars", wxLIST_FORMAT_LEFT, 50); + m_listCtrl->InsertColumn(4, "Power", wxLIST_FORMAT_LEFT, 50); + + // Tally top categories + map mapTopCategories; + CRITICAL_BLOCK(cs_mapProducts) + for (map::iterator mi = mapProducts.begin(); mi != mapProducts.end(); ++mi) + mapTopCategories[(*mi).second.mapValue["category"]]++; + + // Sort top categories + vector > vTopCategories; + for (map::iterator mi = mapTopCategories.begin(); mi != mapTopCategories.end(); ++mi) + vTopCategories.push_back(make_pair((*mi).second, (*mi).first)); + sort(vTopCategories.begin(), vTopCategories.end(), CompareIntStringPairBestFirst); + + // Fill categories combo box + int nLimit = 250; + for (vector >::iterator it = vTopCategories.begin(); it != vTopCategories.end() && nLimit-- > 0; ++it) + m_comboBoxCategory->Append((*it).second); + + // Fill window with initial search + //wxCommandEvent event; + //OnButtonSearch(event); +} + +void CProductsDialog::OnCombobox(wxCommandEvent& event) +{ + OnButtonSearch(event); +} + +bool CompareProductsBestFirst(const CProduct* p1, const CProduct* p2) +{ + return (p1->nAtoms > p2->nAtoms); +} + +void CProductsDialog::OnButtonSearch(wxCommandEvent& event) +{ + string strCategory = (string)m_comboBoxCategory->GetValue(); + string strSearch = (string)m_textCtrlSearch->GetValue(); + + // Search products + vector vProductsFound; + CRITICAL_BLOCK(cs_mapProducts) + { + for (map::iterator mi = mapProducts.begin(); mi != mapProducts.end(); ++mi) + { + CProduct& product = (*mi).second; + if (product.mapValue["category"].find(strCategory) != -1) + { + if (product.mapValue["title"].find(strSearch) != -1 || + product.mapValue["description"].find(strSearch) != -1 || + product.mapValue["seller"].find(strSearch) != -1) + { + vProductsFound.push_back(&product); + } + } + } + } + + // Sort + sort(vProductsFound.begin(), vProductsFound.end(), CompareProductsBestFirst); + + // Display + foreach(CProduct* pproduct, vProductsFound) + { + InsertLine(m_listCtrl, + pproduct->mapValue["title"], + pproduct->mapValue["price"], + pproduct->mapValue["seller"], + pproduct->mapValue["stars"], + itostr(pproduct->nAtoms)); + } +} + +void CProductsDialog::OnListItemActivated(wxListEvent& event) +{ + // Doubleclick opens product + CViewProductDialog* pdialog = new CViewProductDialog(this, m_vProduct[event.GetIndex()]); + pdialog->Show(); +} + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CEditProductDialog +// + +CEditProductDialog::CEditProductDialog(wxWindow* parent) : CEditProductDialogBase(parent) +{ + m_textCtrlLabel[0 ] = m_textCtrlLabel0; + m_textCtrlLabel[1 ] = m_textCtrlLabel1; + m_textCtrlLabel[2 ] = m_textCtrlLabel2; + m_textCtrlLabel[3 ] = m_textCtrlLabel3; + m_textCtrlLabel[4 ] = m_textCtrlLabel4; + m_textCtrlLabel[5 ] = m_textCtrlLabel5; + m_textCtrlLabel[6 ] = m_textCtrlLabel6; + m_textCtrlLabel[7 ] = m_textCtrlLabel7; + m_textCtrlLabel[8 ] = m_textCtrlLabel8; + m_textCtrlLabel[9 ] = m_textCtrlLabel9; + m_textCtrlLabel[10] = m_textCtrlLabel10; + m_textCtrlLabel[11] = m_textCtrlLabel11; + m_textCtrlLabel[12] = m_textCtrlLabel12; + m_textCtrlLabel[13] = m_textCtrlLabel13; + m_textCtrlLabel[14] = m_textCtrlLabel14; + m_textCtrlLabel[15] = m_textCtrlLabel15; + m_textCtrlLabel[16] = m_textCtrlLabel16; + m_textCtrlLabel[17] = m_textCtrlLabel17; + m_textCtrlLabel[18] = m_textCtrlLabel18; + m_textCtrlLabel[19] = m_textCtrlLabel19; + + m_textCtrlField[0 ] = m_textCtrlField0; + m_textCtrlField[1 ] = m_textCtrlField1; + m_textCtrlField[2 ] = m_textCtrlField2; + m_textCtrlField[3 ] = m_textCtrlField3; + m_textCtrlField[4 ] = m_textCtrlField4; + m_textCtrlField[5 ] = m_textCtrlField5; + m_textCtrlField[6 ] = m_textCtrlField6; + m_textCtrlField[7 ] = m_textCtrlField7; + m_textCtrlField[8 ] = m_textCtrlField8; + m_textCtrlField[9 ] = m_textCtrlField9; + m_textCtrlField[10] = m_textCtrlField10; + m_textCtrlField[11] = m_textCtrlField11; + m_textCtrlField[12] = m_textCtrlField12; + m_textCtrlField[13] = m_textCtrlField13; + m_textCtrlField[14] = m_textCtrlField14; + m_textCtrlField[15] = m_textCtrlField15; + m_textCtrlField[16] = m_textCtrlField16; + m_textCtrlField[17] = m_textCtrlField17; + m_textCtrlField[18] = m_textCtrlField18; + m_textCtrlField[19] = m_textCtrlField19; + + m_buttonDel[0 ] = m_buttonDel0; + m_buttonDel[1 ] = m_buttonDel1; + m_buttonDel[2 ] = m_buttonDel2; + m_buttonDel[3 ] = m_buttonDel3; + m_buttonDel[4 ] = m_buttonDel4; + m_buttonDel[5 ] = m_buttonDel5; + m_buttonDel[6 ] = m_buttonDel6; + m_buttonDel[7 ] = m_buttonDel7; + m_buttonDel[8 ] = m_buttonDel8; + m_buttonDel[9 ] = m_buttonDel9; + m_buttonDel[10] = m_buttonDel10; + m_buttonDel[11] = m_buttonDel11; + m_buttonDel[12] = m_buttonDel12; + m_buttonDel[13] = m_buttonDel13; + m_buttonDel[14] = m_buttonDel14; + m_buttonDel[15] = m_buttonDel15; + m_buttonDel[16] = m_buttonDel16; + m_buttonDel[17] = m_buttonDel17; + m_buttonDel[18] = m_buttonDel18; + m_buttonDel[19] = m_buttonDel19; + + for (int i = 1; i < FIELDS_MAX; i++) + ShowLine(i, false); + + LayoutAll(); +} + +void CEditProductDialog::LayoutAll() +{ + m_scrolledWindow->Layout(); + m_scrolledWindow->GetSizer()->Fit(m_scrolledWindow); + this->Layout(); +} + +void CEditProductDialog::ShowLine(int i, bool fShow) +{ + m_textCtrlLabel[i]->Show(fShow); + m_textCtrlField[i]->Show(fShow); + m_buttonDel[i]->Show(fShow); +} + +void CEditProductDialog::OnButtonDel0(wxCommandEvent& event) { OnButtonDel(event, 0); } +void CEditProductDialog::OnButtonDel1(wxCommandEvent& event) { OnButtonDel(event, 1); } +void CEditProductDialog::OnButtonDel2(wxCommandEvent& event) { OnButtonDel(event, 2); } +void CEditProductDialog::OnButtonDel3(wxCommandEvent& event) { OnButtonDel(event, 3); } +void CEditProductDialog::OnButtonDel4(wxCommandEvent& event) { OnButtonDel(event, 4); } +void CEditProductDialog::OnButtonDel5(wxCommandEvent& event) { OnButtonDel(event, 5); } +void CEditProductDialog::OnButtonDel6(wxCommandEvent& event) { OnButtonDel(event, 6); } +void CEditProductDialog::OnButtonDel7(wxCommandEvent& event) { OnButtonDel(event, 7); } +void CEditProductDialog::OnButtonDel8(wxCommandEvent& event) { OnButtonDel(event, 8); } +void CEditProductDialog::OnButtonDel9(wxCommandEvent& event) { OnButtonDel(event, 9); } +void CEditProductDialog::OnButtonDel10(wxCommandEvent& event) { OnButtonDel(event, 10); } +void CEditProductDialog::OnButtonDel11(wxCommandEvent& event) { OnButtonDel(event, 11); } +void CEditProductDialog::OnButtonDel12(wxCommandEvent& event) { OnButtonDel(event, 12); } +void CEditProductDialog::OnButtonDel13(wxCommandEvent& event) { OnButtonDel(event, 13); } +void CEditProductDialog::OnButtonDel14(wxCommandEvent& event) { OnButtonDel(event, 14); } +void CEditProductDialog::OnButtonDel15(wxCommandEvent& event) { OnButtonDel(event, 15); } +void CEditProductDialog::OnButtonDel16(wxCommandEvent& event) { OnButtonDel(event, 16); } +void CEditProductDialog::OnButtonDel17(wxCommandEvent& event) { OnButtonDel(event, 17); } +void CEditProductDialog::OnButtonDel18(wxCommandEvent& event) { OnButtonDel(event, 18); } +void CEditProductDialog::OnButtonDel19(wxCommandEvent& event) { OnButtonDel(event, 19); } + +void CEditProductDialog::OnButtonDel(wxCommandEvent& event, int n) +{ + Freeze(); + int x, y; + m_scrolledWindow->GetViewStart(&x, &y); + int i; + for (i = n; i < FIELDS_MAX-1; i++) + { + m_textCtrlLabel[i]->SetValue(m_textCtrlLabel[i+1]->GetValue()); + m_textCtrlField[i]->SetValue(m_textCtrlField[i+1]->GetValue()); + if (!m_buttonDel[i+1]->IsShown()) + break; + } + m_textCtrlLabel[i]->SetValue(""); + m_textCtrlField[i]->SetValue(""); + ShowLine(i, false); + m_buttonAddField->Enable(true); + LayoutAll(); + m_scrolledWindow->Scroll(0, y); + Thaw(); +} + +void CEditProductDialog::OnButtonAddField(wxCommandEvent& event) +{ + for (int i = 0; i < FIELDS_MAX; i++) + { + if (!m_buttonDel[i]->IsShown()) + { + Freeze(); + ShowLine(i, true); + if (i == FIELDS_MAX-1) + m_buttonAddField->Enable(false); + LayoutAll(); + m_scrolledWindow->Scroll(0, 99999); + Thaw(); + break; + } + } +} + +void CEditProductDialog::OnButtonSend(wxCommandEvent& event) +{ + CProduct product; + GetProduct(product); + + // Sign the detailed product + product.vchPubKeyFrom = keyUser.GetPubKey(); + if (!keyUser.Sign(product.GetSigHash(), product.vchSig)) + { + wxMessageBox("Error digitally signing the product "); + return; + } + + // Save detailed product + AddToMyProducts(product); + + // Strip down to summary product + product.mapDetails.clear(); + product.vOrderForm.clear(); + + // Sign the summary product + if (!keyUser.Sign(product.GetSigHash(), product.vchSig)) + { + wxMessageBox("Error digitally signing the product "); + return; + } + + // Verify + if (!product.CheckProduct()) + { + wxMessageBox("Errors found in product "); + return; + } + + // Broadcast + AdvertStartPublish(pnodeLocalHost, MSG_PRODUCT, 0, product); + + Destroy(); +} + +void CEditProductDialog::OnButtonPreview(wxCommandEvent& event) +{ + CProduct product; + GetProduct(product); + CViewProductDialog* pdialog = new CViewProductDialog(this, product); + pdialog->Show(); +} + +void CEditProductDialog::OnButtonCancel(wxCommandEvent& event) +{ + Destroy(); +} + +void CEditProductDialog::SetProduct(const CProduct& productIn) +{ + CProduct product = productIn; + + m_comboBoxCategory->SetValue(product.mapValue["category"]); + m_textCtrlTitle->SetValue(product.mapValue["title"]); + m_textCtrlPrice->SetValue(product.mapValue["price"]); + m_textCtrlDescription->SetValue(product.mapValue["description"]); + m_textCtrlInstructions->SetValue(product.mapValue["instructions"]); + + for (int i = 0; i < FIELDS_MAX; i++) + { + bool fUsed = i < product.vOrderForm.size(); + m_buttonDel[i]->Show(fUsed); + m_textCtrlLabel[i]->Show(fUsed); + m_textCtrlField[i]->Show(fUsed); + if (!fUsed) + continue; + + m_textCtrlLabel[i]->SetValue(product.vOrderForm[i].first); + string strControl = product.vOrderForm[i].second; + if (strControl.substr(0, 5) == "text=") + m_textCtrlField[i]->SetValue(""); + else if (strControl.substr(0, 7) == "choice=") + m_textCtrlField[i]->SetValue(strControl.substr(7)); + else + m_textCtrlField[i]->SetValue(strControl); + } +} + +void CEditProductDialog::GetProduct(CProduct& product) +{ + // map mapValue; + // vector > vOrderForm; + + product.mapValue["category"] = m_comboBoxCategory->GetValue().Trim(); + product.mapValue["title"] = m_textCtrlTitle->GetValue().Trim(); + product.mapValue["price"] = m_textCtrlPrice->GetValue().Trim(); + product.mapValue["description"] = m_textCtrlDescription->GetValue().Trim(); + product.mapValue["instructions"] = m_textCtrlInstructions->GetValue().Trim(); + + for (int i = 0; i < FIELDS_MAX; i++) + { + if (m_buttonDel[i]->IsShown()) + { + string strLabel = (string)m_textCtrlLabel[i]->GetValue().Trim(); + string strControl = (string)m_textCtrlField[i]->GetValue(); + if (strControl.empty()) + strControl = "text="; + else + strControl = "choice=" + strControl; + product.vOrderForm.push_back(make_pair(strLabel, strControl)); + } + } +} + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CViewProductDialog +// + +CViewProductDialog::CViewProductDialog(wxWindow* parent, const CProduct& productIn) : CViewProductDialogBase(parent) +{ + Connect(wxEVT_REPLY1, wxCommandEventHandler(CViewProductDialog::OnReply1), NULL, this); + AddCallbackAvailable(GetEventHandler()); + + // Fill display with product summary while waiting for details + product = productIn; + UpdateProductDisplay(false); + + m_buttonBack->Enable(false); + m_buttonNext->Enable(!product.vOrderForm.empty()); + m_htmlWinReviews->Show(true); + m_scrolledWindow->Show(false); + this->Layout(); + + // Request details from seller + _beginthread(ThreadRequestProductDetails, 0, new pair(product, GetEventHandler())); +} + +CViewProductDialog::~CViewProductDialog() +{ + RemoveCallbackAvailable(GetEventHandler()); +} + +void ThreadRequestProductDetails(void* parg) +{ + // Extract parameters + pair* pitem = (pair*)parg; + CProduct product = pitem->first; + wxEvtHandler* pevthandler = pitem->second; + delete pitem; + + // Connect to seller + CNode* pnode = ConnectNode(product.addr, 5 * 60); + if (!pnode) + { + CDataStream ssEmpty; + AddPendingReplyEvent1(pevthandler, ssEmpty); + return; + } + + // Request detailed product, with response going to OnReply1 via dialog's event handler + pnode->PushRequest("getdetails", product.GetHash(), AddPendingReplyEvent1, (void*)pevthandler); +} + +void CViewProductDialog::OnReply1(wxCommandEvent& event) +{ + CDataStream ss = GetStreamFromEvent(event); + if (ss.empty()) + { + product.mapValue["description"] = "-- CAN'T CONNECT TO SELLER --\n"; + UpdateProductDisplay(true); + return; + } + + int nRet; + CProduct product2; + try + { + ss >> nRet; + if (nRet > 0) + throw false; + ss >> product2; + if (product2.GetHash() != product.GetHash()) + throw false; + if (!product2.CheckSignature()) + throw false; + } + catch (...) + { + product.mapValue["description"] = "-- INVALID RESPONSE --\n"; + UpdateProductDisplay(true); + return; + } + + product = product2; + UpdateProductDisplay(true); +} + +bool CompareReviewsBestFirst(const CReview* p1, const CReview* p2) +{ + return (p1->nAtoms > p2->nAtoms); +} + +void CViewProductDialog::UpdateProductDisplay(bool fDetails) +{ + // Product and reviews + string strHTML; + strHTML.reserve(4000); + strHTML += "\n" + "\n" + "\n" + "\n" + "\n"; + strHTML += "Category: " + HtmlEscape(product.mapValue["category"]) + "
\n"; + strHTML += "Title: " + HtmlEscape(product.mapValue["title"]) + "
\n"; + strHTML += "Price: " + HtmlEscape(product.mapValue["price"]) + "
\n"; + + if (!fDetails) + strHTML += "Loading details...
\n
\n"; + else + strHTML += HtmlEscape(product.mapValue["description"], true) + "
\n
\n"; + + strHTML += "Reviews:
\n
\n"; + + if (!product.vchPubKeyFrom.empty()) + { + CReviewDB reviewdb("r"); + + // Get reviews + vector vReviews; + reviewdb.ReadReviews(product.GetUserHash(), vReviews); + + // Get reviewer's number of atoms + vector vSortedReviews; + vSortedReviews.reserve(vReviews.size()); + for (vector::reverse_iterator it = vReviews.rbegin(); it != vReviews.rend(); ++it) + { + CReview& review = *it; + CUser user; + reviewdb.ReadUser(review.GetUserHash(), user); + review.nAtoms = user.GetAtomCount(); + vSortedReviews.push_back(&review); + } + + reviewdb.Close(); + + // Sort + stable_sort(vSortedReviews.begin(), vSortedReviews.end(), CompareReviewsBestFirst); + + // Format reviews + foreach(CReview* preview, vSortedReviews) + { + CReview& review = *preview; + int nStars = atoi(review.mapValue["stars"].c_str()); + if (nStars < 1 || nStars > 5) + continue; + + strHTML += "" + itostr(nStars) + (nStars == 1 ? " star" : " stars") + ""; + strHTML += "     "; + strHTML += DateStr(atoi64(review.mapValue["date"])) + "
\n"; + strHTML += HtmlEscape(review.mapValue["review"], true); + strHTML += "
\n
\n"; + } + } + + strHTML += "\n\n"; + + // Shrink capacity to fit + string(strHTML.begin(), strHTML.end()).swap(strHTML); + + m_htmlWinReviews->SetPage(strHTML); + + ///// need to find some other indicator to use so can allow empty order form + if (product.vOrderForm.empty()) + return; + + // Order form + m_staticTextInstructions->SetLabel(product.mapValue["instructions"]); + for (int i = 0; i < FIELDS_MAX; i++) + { + m_staticTextLabel[i] = NULL; + m_textCtrlField[i] = NULL; + m_choiceField[i] = NULL; + } + + // Construct flexgridsizer + wxBoxSizer* bSizer21 = (wxBoxSizer*)m_scrolledWindow->GetSizer(); + wxFlexGridSizer* fgSizer; + fgSizer = new wxFlexGridSizer(0, 2, 0, 0); + fgSizer->AddGrowableCol(1); + fgSizer->SetFlexibleDirection(wxBOTH); + fgSizer->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); + + // Construct order form fields + wxWindow* windowLast = NULL; + for (int i = 0; i < product.vOrderForm.size(); i++) + { + string strLabel = product.vOrderForm[i].first; + string strControl = product.vOrderForm[i].second; + + if (strLabel.size() < 20) + strLabel.insert(strLabel.begin(), 20 - strLabel.size(), ' '); + + m_staticTextLabel[i] = new wxStaticText(m_scrolledWindow, wxID_ANY, strLabel, wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); + m_staticTextLabel[i]->Wrap(-1); + fgSizer->Add(m_staticTextLabel[i], 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxALL, 5); + + if (strControl.substr(0, 5) == "text=") + { + m_textCtrlField[i] = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + fgSizer->Add(m_textCtrlField[i], 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5); + windowLast = m_textCtrlField[i]; + } + else if (strControl.substr(0, 7) == "choice=") + { + vector vChoices; + ParseString(strControl.substr(7), ',', vChoices); + + wxArrayString arraystring; + foreach(const string& str, vChoices) + arraystring.Add(str); + + m_choiceField[i] = new wxChoice(m_scrolledWindow, wxID_ANY, wxDefaultPosition, wxDefaultSize, arraystring, 0); + fgSizer->Add(m_choiceField[i], 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + windowLast = m_choiceField[i]; + } + else + { + m_textCtrlField[i] = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + fgSizer->Add(m_textCtrlField[i], 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5); + m_staticTextLabel[i]->Show(false); + m_textCtrlField[i]->Show(false); + } + } + + // Insert after instructions and before submit/cancel buttons + bSizer21->Insert(2, fgSizer, 0, wxEXPAND|wxRIGHT|wxLEFT, 5); + m_scrolledWindow->Layout(); + bSizer21->Fit(m_scrolledWindow); + this->Layout(); + + // Fixup the tab order + m_buttonSubmitForm->MoveAfterInTabOrder(windowLast); + m_buttonCancelForm->MoveAfterInTabOrder(m_buttonSubmitForm); + //m_buttonBack->MoveAfterInTabOrder(m_buttonCancelForm); + //m_buttonNext->MoveAfterInTabOrder(m_buttonBack); + //m_buttonCancel->MoveAfterInTabOrder(m_buttonNext); + this->Layout(); +} + +void CViewProductDialog::GetOrder(CWalletTx& wtx) +{ + wtx.SetNull(); + for (int i = 0; i < product.vOrderForm.size(); i++) + { + string strValue; + if (m_textCtrlField[i]) + strValue = m_textCtrlField[i]->GetValue().Trim(); + else + strValue = m_choiceField[i]->GetStringSelection(); + wtx.vOrderForm.push_back(make_pair(m_staticTextLabel[i]->GetLabel(), strValue)); + } +} + +void CViewProductDialog::OnButtonSubmitForm(wxCommandEvent& event) +{ + m_buttonSubmitForm->Enable(false); + m_buttonCancelForm->Enable(false); + + CWalletTx wtx; + GetOrder(wtx); + + CSendingDialog* pdialog = new CSendingDialog(this, product.addr, atoi64(product.mapValue["price"]), wtx); + if (!pdialog->ShowModal()) + { + m_buttonSubmitForm->Enable(true); + m_buttonCancelForm->Enable(true); + return; + } +} + +void CViewProductDialog::OnButtonCancelForm(wxCommandEvent& event) +{ + Destroy(); +} + +void CViewProductDialog::OnButtonBack(wxCommandEvent& event) +{ + Freeze(); + m_htmlWinReviews->Show(true); + m_scrolledWindow->Show(false); + m_buttonBack->Enable(false); + m_buttonNext->Enable(!product.vOrderForm.empty()); + this->Layout(); + Thaw(); +} + +void CViewProductDialog::OnButtonNext(wxCommandEvent& event) +{ + if (!product.vOrderForm.empty()) + { + Freeze(); + m_htmlWinReviews->Show(false); + m_scrolledWindow->Show(true); + m_buttonBack->Enable(true); + m_buttonNext->Enable(false); + this->Layout(); + Thaw(); + } +} + +void CViewProductDialog::OnButtonCancel(wxCommandEvent& event) +{ + Destroy(); +} + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CViewOrderDialog +// + +CViewOrderDialog::CViewOrderDialog(wxWindow* parent, CWalletTx order, bool fReceived) : CViewOrderDialogBase(parent) +{ + int64 nPrice = (fReceived ? order.GetCredit() : order.GetDebit()); + + string strHTML; + strHTML.reserve(4000); + strHTML += "\n" + "\n" + "\n" + "\n" + "\n"; + strHTML += "Time: " + HtmlEscape(DateTimeStr(order.nTimeReceived)) + "
\n"; + strHTML += "Price: " + HtmlEscape(FormatMoney(nPrice)) + "
\n"; + strHTML += "Status: " + HtmlEscape(FormatTxStatus(order)) + "
\n"; + + strHTML += "\n"; + for (int i = 0; i < order.vOrderForm.size(); i++) + { + strHTML += " "; + strHTML += "\n"; + } + strHTML += "
" + HtmlEscape(order.vOrderForm[i].first) + ":" + HtmlEscape(order.vOrderForm[i].second) + "
\n"; + + strHTML += "\n\n"; + + // Shrink capacity to fit + // (strings are ref counted, so it may live on in SetPage) + string(strHTML.begin(), strHTML.end()).swap(strHTML); + + m_htmlWin->SetPage(strHTML); +} + +void CViewOrderDialog::OnButtonOK(wxCommandEvent& event) +{ + Destroy(); +} + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CEditReviewDialog +// + +CEditReviewDialog::CEditReviewDialog(wxWindow* parent) : CEditReviewDialogBase(parent) +{ +} + +void CEditReviewDialog::OnButtonSubmit(wxCommandEvent& event) +{ + if (m_choiceStars->GetSelection() == -1) + { + wxMessageBox("Please select a rating "); + return; + } + + CReview review; + GetReview(review); + + // Sign the review + review.vchPubKeyFrom = keyUser.GetPubKey(); + if (!keyUser.Sign(review.GetSigHash(), review.vchSig)) + { + wxMessageBox("Unable to digitally sign the review "); + return; + } + + // Broadcast + if (!review.AcceptReview()) + { + wxMessageBox("Save failed "); + return; + } + RelayMessage(CInv(MSG_REVIEW, review.GetHash()), review); + + Destroy(); +} + +void CEditReviewDialog::OnButtonCancel(wxCommandEvent& event) +{ + Destroy(); +} + +void CEditReviewDialog::GetReview(CReview& review) +{ + review.mapValue["time"] = i64tostr(GetAdjustedTime()); + review.mapValue["stars"] = itostr(m_choiceStars->GetSelection()+1); + review.mapValue["review"] = m_textCtrlReview->GetValue(); +} + + + + + + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CMyApp +// + +// Define a new application +class CMyApp: public wxApp +{ + public: + CMyApp(){}; + ~CMyApp(){}; + bool OnInit(); + bool OnInit2(); + int OnExit(); + + // 2nd-level exception handling: we get all the exceptions occurring in any + // event handler here + virtual bool OnExceptionInMainLoop(); + + // 3rd, and final, level exception handling: whenever an unhandled + // exception is caught, this function is called + virtual void OnUnhandledException(); + + // and now for something different: this function is called in case of a + // crash (e.g. dereferencing null pointer, division by 0, ...) + virtual void OnFatalException(); +}; + +IMPLEMENT_APP(CMyApp) + +bool CMyApp::OnInit() +{ + try + { + return OnInit2(); + } + catch (std::exception& e) { + PrintException(&e, "OnInit()"); + } catch (...) { + PrintException(NULL, "OnInit()"); + } + return false; +} + +map ParseParameters(int argc, char* argv[]) +{ + map mapArgs; + for (int i = 0; i < argc; i++) + { + char psz[10000]; + strcpy(psz, argv[i]); + char* pszValue = ""; + if (strchr(psz, '=')) + { + pszValue = strchr(psz, '='); + *pszValue++ = '\0'; + } + strlwr(psz); + if (psz[0] == '-') + psz[0] = '/'; + mapArgs[psz] = pszValue; + } + return mapArgs; +} + +bool CMyApp::OnInit2() +{ +#ifdef _MSC_VER + // Turn off microsoft heap dump noise for now + _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE); + _CrtSetReportFile(_CRT_WARN, CreateFile("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); +#endif +#ifdef __WXDEBUG__ + // Disable malfunctioning wxWidgets debug assertion + g_isPainting = 10000; +#endif + + //// debug print + printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); + printf("Bitcoin version %d, Windows version %08x\n", VERSION, GetVersion()); + + // + // Limit to single instance per user + // Required to protect the database files if we're going to keep deleting log.* + // + wxString strMutexName = wxString("Bitcoin.") + getenv("HOMEPATH"); + for (int i = 0; i < strMutexName.size(); i++) + if (!isalnum(strMutexName[i])) + strMutexName[i] = '.'; + wxSingleInstanceChecker* psingleinstancechecker = new wxSingleInstanceChecker(strMutexName); + if (psingleinstancechecker->IsAnotherRunning()) + { + printf("Existing instance found\n"); + unsigned int nStart = GetTime(); + loop + { + // Show the previous instance and exit + HWND hwndPrev = FindWindow("wxWindowClassNR", "Bitcoin"); + if (hwndPrev) + { + if (IsIconic(hwndPrev)) + ShowWindow(hwndPrev, SW_RESTORE); + SetForegroundWindow(hwndPrev); + return false; + } + + if (GetTime() > nStart + 60) + return false; + + // Resume this instance if the other exits + delete psingleinstancechecker; + Sleep(1000); + psingleinstancechecker = new wxSingleInstanceChecker(strMutexName); + if (!psingleinstancechecker->IsAnotherRunning()) + break; + } + } + + // + // Parameters + // + wxImage::AddHandler(new wxPNGHandler); + map mapArgs = ParseParameters(argc, argv); + + if (mapArgs.count("/datadir")) + strSetDataDir = mapArgs["/datadir"]; + + if (mapArgs.count("/proxy")) + addrProxy = CAddress(mapArgs["/proxy"].c_str()); + + if (mapArgs.count("/debug")) + fDebug = true; + + if (mapArgs.count("/dropmessages")) + { + nDropMessagesTest = atoi(mapArgs["/dropmessages"]); + if (nDropMessagesTest == 0) + nDropMessagesTest = 20; + } + + if (mapArgs.count("/loadblockindextest")) + { + CTxDB txdb("r"); + txdb.LoadBlockIndex(); + PrintBlockTree(); + ExitProcess(0); + } + + // + // Load data files + // + string strErrors; + int64 nStart, nEnd; + + printf("Loading addresses...\n"); + QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + if (!LoadAddresses()) + strErrors += "Error loading addr.dat \n"; + QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); + printf(" addresses %20I64d\n", nEnd - nStart); + + printf("Loading block index...\n"); + QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + if (!LoadBlockIndex()) + strErrors += "Error loading blkindex.dat \n"; + QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); + printf(" block index %20I64d\n", nEnd - nStart); + + printf("Loading wallet...\n"); + QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + if (!LoadWallet()) + strErrors += "Error loading wallet.dat \n"; + QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); + printf(" wallet %20I64d\n", nEnd - nStart); + + printf("Done loading\n"); + + //// debug print + printf("mapBlockIndex.size() = %d\n", mapBlockIndex.size()); + printf("nBestHeight = %d\n", nBestHeight); + printf("mapKeys.size() = %d\n", mapKeys.size()); + printf("mapPubKeys.size() = %d\n", mapPubKeys.size()); + printf("mapWallet.size() = %d\n", mapWallet.size()); + printf("mapAddressBook.size() = %d\n", mapAddressBook.size()); + + if (!strErrors.empty()) + { + wxMessageBox(strErrors, "Bitcoin"); + OnExit(); + return false; + } + + // Add wallet transactions that aren't already in a block to mapTransactions + ReacceptWalletTransactions(); + + // + // Parameters + // + if (mapArgs.count("/printblockindex") || mapArgs.count("/printblocktree")) + { + PrintBlockTree(); + OnExit(); + return false; + } + + if (mapArgs.count("/gen")) + { + if (mapArgs["/gen"].empty()) + fGenerateBitcoins = true; + else + fGenerateBitcoins = atoi(mapArgs["/gen"].c_str()); + } + + // + // Create the main frame window + // + { + pframeMain = new CMainFrame(NULL); + pframeMain->Show(); + + if (!CheckDiskSpace()) + { + OnExit(); + return false; + } + + if (!StartNode(strErrors)) + wxMessageBox(strErrors, "Bitcoin"); + + if (fGenerateBitcoins) + if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) + printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); + + // + // Tests + // + if (argc >= 2 && stricmp(argv[1], "/send") == 0) + { + int64 nValue = 1; + if (argc >= 3) + ParseMoney(argv[2], nValue); + + string strAddress; + if (argc >= 4) + strAddress = argv[3]; + CAddress addr(strAddress.c_str()); + + CWalletTx wtx; + wtx.mapValue["to"] = strAddress; + wtx.mapValue["from"] = addrLocalHost.ToString(); + wtx.mapValue["message"] = "command line send"; + + // Send to IP address + CSendingDialog* pdialog = new CSendingDialog(pframeMain, addr, nValue, wtx); + if (!pdialog->ShowModal()) + return false; + } + + if (mapArgs.count("/randsendtest")) + { + if (!mapArgs["/randsendtest"].empty()) + _beginthread(ThreadRandSendTest, 0, new string(mapArgs["/randsendtest"])); + else + fRandSendTest = true; + fDebug = true; + } + } + + return true; +} + +int CMyApp::OnExit() +{ + Shutdown(NULL); + return wxApp::OnExit(); +} + +bool CMyApp::OnExceptionInMainLoop() +{ + try + { + throw; + } + catch (std::exception& e) + { + PrintException(&e, "CMyApp::OnExceptionInMainLoop()"); + wxLogWarning(_T("Exception %s %s"), typeid(e).name(), e.what()); + Sleep(1000); + throw; + } + catch (...) + { + PrintException(NULL, "CMyApp::OnExceptionInMainLoop()"); + wxLogWarning(_T("Unknown exception")); + Sleep(1000); + throw; + } + + return true; +} + +void CMyApp::OnUnhandledException() +{ + // this shows how we may let some exception propagate uncaught + try + { + throw; + } + catch (std::exception& e) + { + PrintException(&e, "CMyApp::OnUnhandledException()"); + wxLogWarning(_T("Exception %s %s"), typeid(e).name(), e.what()); + Sleep(1000); + throw; + } + catch (...) + { + PrintException(NULL, "CMyApp::OnUnhandledException()"); + wxLogWarning(_T("Unknown exception")); + Sleep(1000); + throw; + } +} + +void CMyApp::OnFatalException() +{ + wxMessageBox("Program has crashed and will terminate. ", "Bitcoin", wxOK | wxICON_ERROR); +} + + + +void MainFrameRepaint() +{ + if (pframeMain) + { + printf("MainFrameRepaint()\n"); + wxPaintEvent event; + pframeMain->Refresh(); + pframeMain->AddPendingEvent(event); + } +} + + + + + + + +// randsendtest to bitcoin address +void ThreadRandSendTest(void* parg) +{ + string strAddress = *(string*)parg; + uint160 hash160; + if (!AddressToHash160(strAddress, hash160)) + { + wxMessageBox(strprintf("ThreadRandSendTest: Bitcoin address '%s' not valid ", strAddress.c_str())); + return; + } + + loop + { + Sleep(GetRand(30) * 1000 + 100); + + // Message + CWalletTx wtx; + wtx.mapValue["to"] = strAddress; + wtx.mapValue["from"] = addrLocalHost.ToString(); + static int nRep; + wtx.mapValue["message"] = strprintf("randsendtest %d\n", ++nRep); + + // Value + int64 nValue = (GetRand(9) + 1) * 100 * CENT; + if (GetBalance() < nValue) + { + wxMessageBox("Out of money "); + return; + } + nValue += (nRep % 100) * CENT; + + // Send to bitcoin address + CScript scriptPubKey; + scriptPubKey << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; + + if (!SendMoney(scriptPubKey, nValue, wtx)) + return; + } +} + + +// randsendtest to any connected node +void RandSend() +{ + CWalletTx wtx; + + while (vNodes.empty()) + Sleep(1000); + CAddress addr; + CRITICAL_BLOCK(cs_vNodes) + addr = vNodes[GetRand(vNodes.size())]->addr; + + // Message + wtx.mapValue["to"] = addr.ToString(); + wtx.mapValue["from"] = addrLocalHost.ToString(); + static int nRep; + wtx.mapValue["message"] = strprintf("randsendtest %d\n", ++nRep); + + // Value + int64 nValue = (GetRand(999) + 1) * CENT; + if (GetBalance() < nValue) + { + wxMessageBox("Out of money "); + return; + } + + // Send to IP address + CSendingDialog* pdialog = new CSendingDialog(pframeMain, addr, nValue, wtx); + if (!pdialog->Show()) + wxMessageBox("ShowModal Failed "); +} diff --git a/ui.h b/ui.h new file mode 100644 index 00000000..163554a5 --- /dev/null +++ b/ui.h @@ -0,0 +1,420 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + + + + +DECLARE_EVENT_TYPE(wxEVT_CROSSTHREADCALL, -1) +DECLARE_EVENT_TYPE(wxEVT_REPLY1, -1) +DECLARE_EVENT_TYPE(wxEVT_REPLY2, -1) +DECLARE_EVENT_TYPE(wxEVT_REPLY3, -1) +DECLARE_EVENT_TYPE(wxEVT_TABLEADDED, -1) +DECLARE_EVENT_TYPE(wxEVT_TABLEUPDATED, -1) +DECLARE_EVENT_TYPE(wxEVT_TABLEDELETED, -1) + +enum +{ + UICALL_ADDORDER = 1, + UICALL_UPDATEORDER, +}; + + + +extern void HandleCtrlA(wxKeyEvent& event); +extern string DateTimeStr(int64 nTime); +extern string FormatTxStatus(const CWalletTx& wtx); +extern void CrossThreadCall(int nID, void* pdata); +extern void MainFrameRepaint(); +extern void Shutdown(void* parg); + + + + + + +class CMainFrame : public CMainFrameBase +{ +protected: + // Event handlers + void OnClose(wxCloseEvent& event); + void OnMouseEvents(wxMouseEvent& event); + void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } + void OnIdle(wxIdleEvent& event); + void OnPaint(wxPaintEvent& event); + void OnPaintListCtrl(wxPaintEvent& event); + void OnMenuFileExit(wxCommandEvent& event); + void OnMenuOptionsGenerate(wxCommandEvent& event); + void OnMenuOptionsChangeYourAddress(wxCommandEvent& event); + void OnMenuOptionsOptions(wxCommandEvent& event); + void OnMenuHelpAbout(wxCommandEvent& event); + void OnButtonSend(wxCommandEvent& event); + void OnButtonAddressBook(wxCommandEvent& event); + void OnSetFocusAddress(wxFocusEvent& event); + void OnMouseEventsAddress(wxMouseEvent& event); + void OnButtonCopy(wxCommandEvent& event); + void OnButtonChange(wxCommandEvent& event); + void OnListColBeginDrag(wxListEvent& event); + void OnListItemActivatedAllTransactions(wxListEvent& event); + void OnListItemActivatedProductsSent(wxListEvent& event); + void OnListItemActivatedOrdersSent(wxListEvent& event); + void OnListItemActivatedOrdersReceived(wxListEvent& event); + +public: + /** Constructor */ + CMainFrame(wxWindow* parent); + ~CMainFrame(); + + // Custom + bool fRefreshListCtrl; + bool fRefreshListCtrlRunning; + bool fOnSetFocusAddress; + CBlockIndex* pindexBestLast; + set setUnmaturedDisplayed; + + void OnCrossThreadCall(wxCommandEvent& event); + void InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5); + void InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); + void RefreshListCtrl(); + void RefreshStatus(); +}; + + + + +class CTxDetailsDialog : public CTxDetailsDialogBase +{ +protected: + // Event handlers + void OnButtonOK(wxCommandEvent& event); + +public: + /** Constructor */ + CTxDetailsDialog(wxWindow* parent, CWalletTx wtx); + + // State + CWalletTx wtx; +}; + + + +class COptionsDialog : public COptionsDialogBase +{ +protected: + // Event handlers + void OnKillFocusTransactionFee(wxFocusEvent& event); + void OnButtonOK(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + +public: + /** Constructor */ + COptionsDialog(wxWindow* parent); +}; + + + +class CAboutDialog : public CAboutDialogBase +{ +protected: + // Event handlers + void OnButtonOK(wxCommandEvent& event); + +public: + /** Constructor */ + CAboutDialog(wxWindow* parent); +}; + + + +class CSendDialog : public CSendDialogBase +{ +protected: + // Event handlers + void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } + void OnTextAddress(wxCommandEvent& event); + void OnKillFocusAmount(wxFocusEvent& event); + void OnButtonAddressBook(wxCommandEvent& event); + void OnButtonPaste(wxCommandEvent& event); + void OnButtonSend(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + +public: + /** Constructor */ + CSendDialog(wxWindow* parent, const wxString& strAddress=""); +}; + + + +class CSendingDialog : public CSendingDialogBase +{ +public: + // Event handlers + void OnClose(wxCloseEvent& event); + void OnButtonOK(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + void OnPaint(wxPaintEvent& event); + +public: + /** Constructor */ + CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 nPriceIn, const CWalletTx& wtxIn); + ~CSendingDialog(); + + // State + CAddress addr; + int64 nPrice; + CWalletTx wtx; + wxDateTime start; + string strStatus; + bool fCanCancel; + bool fAbort; + bool fSuccess; + bool fUIDone; + bool fWorkDone; + + void Close(); + void Repaint(); + bool Status(); + bool Status(const string& str); + bool Error(const string& str); + void StartTransfer(); + void OnReply2(CDataStream& vRecv); + void OnReply3(CDataStream& vRecv); +}; + +void SendingDialogStartTransfer(void* parg); +void SendingDialogOnReply2(void* parg, CDataStream& vRecv); +void SendingDialogOnReply3(void* parg, CDataStream& vRecv); + + + +class CYourAddressDialog : public CYourAddressDialogBase +{ +protected: + // Event handlers + void OnListEndLabelEdit(wxListEvent& event); + void OnListItemSelected(wxListEvent& event); + void OnListItemActivated(wxListEvent& event); + void OnButtonRename(wxCommandEvent& event); + void OnButtonNew(wxCommandEvent& event); + void OnButtonCopy(wxCommandEvent& event); + void OnButtonOK(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + void OnClose(wxCloseEvent& event); + +public: + /** Constructor */ + CYourAddressDialog(wxWindow* parent); + CYourAddressDialog(wxWindow* parent, const string& strInitSelected); + + // Custom + wxString GetAddress(); +}; + + + +class CAddressBookDialog : public CAddressBookDialogBase +{ +protected: + // Event handlers + void OnListEndLabelEdit(wxListEvent& event); + void OnListItemSelected(wxListEvent& event); + void OnListItemActivated(wxListEvent& event); + void OnButtonEdit(wxCommandEvent& event); + void OnButtonDelete(wxCommandEvent& event); + void OnButtonNew(wxCommandEvent& event); + void OnButtonCopy(wxCommandEvent& event); + void OnButtonOK(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + void OnClose(wxCloseEvent& event); + +public: + /** Constructor */ + CAddressBookDialog(wxWindow* parent, const wxString& strInitSelected, bool fSendingIn); + + // Custom + bool fSending; + wxString GetAddress(); + bool CheckIfMine(const string& strAddress, const string& strTitle); +}; + + + +class CProductsDialog : public CProductsDialogBase +{ +protected: + // Event handlers + void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } + void OnCombobox(wxCommandEvent& event); + void OnButtonSearch(wxCommandEvent& event); + void OnListItemActivated(wxListEvent& event); + +public: + /** Constructor */ + CProductsDialog(wxWindow* parent); + + // Custom + vector m_vProduct; +}; + + + +class CEditProductDialog : public CEditProductDialogBase +{ +protected: + // Event handlers + void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } + void OnButtonDel0(wxCommandEvent& event); + void OnButtonDel1(wxCommandEvent& event); + void OnButtonDel2(wxCommandEvent& event); + void OnButtonDel3(wxCommandEvent& event); + void OnButtonDel4(wxCommandEvent& event); + void OnButtonDel5(wxCommandEvent& event); + void OnButtonDel6(wxCommandEvent& event); + void OnButtonDel7(wxCommandEvent& event); + void OnButtonDel8(wxCommandEvent& event); + void OnButtonDel9(wxCommandEvent& event); + void OnButtonDel10(wxCommandEvent& event); + void OnButtonDel11(wxCommandEvent& event); + void OnButtonDel12(wxCommandEvent& event); + void OnButtonDel13(wxCommandEvent& event); + void OnButtonDel14(wxCommandEvent& event); + void OnButtonDel15(wxCommandEvent& event); + void OnButtonDel16(wxCommandEvent& event); + void OnButtonDel17(wxCommandEvent& event); + void OnButtonDel18(wxCommandEvent& event); + void OnButtonDel19(wxCommandEvent& event); + void OnButtonAddField(wxCommandEvent& event); + void OnButtonSend(wxCommandEvent& event); + void OnButtonPreview(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + +public: + /** Constructor */ + CEditProductDialog(wxWindow* parent); + + // Custom + enum { FIELDS_MAX = 20 }; + wxTextCtrl* m_textCtrlLabel[FIELDS_MAX]; + wxTextCtrl* m_textCtrlField[FIELDS_MAX]; + wxButton* m_buttonDel[FIELDS_MAX]; + + void LayoutAll(); + void ShowLine(int i, bool fShow=true); + void OnButtonDel(wxCommandEvent& event, int n); + void SetProduct(const CProduct& productIn); + void GetProduct(CProduct& product); + +}; + + + +class CViewProductDialog : public CViewProductDialogBase +{ +protected: + // Event handlers + void OnButtonSubmitForm(wxCommandEvent& event); + void OnButtonCancelForm(wxCommandEvent& event); + void OnButtonBack(wxCommandEvent& event); + void OnButtonNext(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + +public: + /** Constructor */ + CViewProductDialog(wxWindow* parent, const CProduct& productIn); + ~CViewProductDialog(); + + // Custom + CProduct product; + enum { FIELDS_MAX = 20 }; + wxStaticText* m_staticTextLabel[FIELDS_MAX]; + wxTextCtrl* m_textCtrlField[FIELDS_MAX]; + wxChoice* m_choiceField[FIELDS_MAX]; + + void GetOrder(CWalletTx& order); + void UpdateProductDisplay(bool fDetails); + void OnReply1(wxCommandEvent& event); +}; + + + +class CViewOrderDialog : public CViewOrderDialogBase +{ +protected: + // Event handlers + void OnButtonOK(wxCommandEvent& event); + +public: + /** Constructor */ + CViewOrderDialog(wxWindow* parent, CWalletTx order, bool fReceived); + + // Custom + bool fReceived; +}; + + + +class CEditReviewDialog : public CEditReviewDialogBase +{ +protected: + // Event handlers + void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } + void OnButtonSubmit(wxCommandEvent& event); + void OnButtonCancel(wxCommandEvent& event); + +public: + /** Constructor */ + CEditReviewDialog(wxWindow* parent); + + // Custom + void GetReview(CReview& review); +}; + + + +class CGetTextFromUserDialog : public CGetTextFromUserDialogBase +{ +protected: + // Event handlers + void OnButtonOK(wxCommandEvent& event) { EndModal(true); } + void OnButtonCancel(wxCommandEvent& event) { EndModal(false); } + void OnClose(wxCloseEvent& event) { EndModal(false); } + + void OnKeyDown(wxKeyEvent& event) + { + if (event.GetKeyCode() == '\r' || event.GetKeyCode() == WXK_NUMPAD_ENTER) + EndModal(true); + else + HandleCtrlA(event); + } + +public: + /** Constructor */ + CGetTextFromUserDialog(wxWindow* parent, + const string& strCaption, + const string& strMessage1, + const string& strValue1="", + const string& strMessage2="", + const string& strValue2="") : CGetTextFromUserDialogBase(parent, wxID_ANY, strCaption) + { + m_staticTextMessage1->SetLabel(strMessage1); + m_textCtrl1->SetValue(strValue1); + if (!strMessage2.empty()) + { + m_staticTextMessage2->Show(true); + m_staticTextMessage2->SetLabel(strMessage2); + m_textCtrl2->Show(true); + m_textCtrl2->SetValue(strValue2); + SetSize(wxDefaultCoord, 180); + } + } + + // Custom + string GetValue() { return (string)m_textCtrl1->GetValue(); } + string GetValue1() { return (string)m_textCtrl1->GetValue(); } + string GetValue2() { return (string)m_textCtrl2->GetValue(); } +}; + + + + + diff --git a/ui.rc b/ui.rc new file mode 100644 index 00000000..95258c7d --- /dev/null +++ b/ui.rc @@ -0,0 +1,14 @@ +bitcoin ICON "rc/bitcoin.ico" + +#include "wx/msw/wx.rc" + +check ICON "rc/check.ico" +send16 BITMAP "rc/send16.bmp" +send16mask BITMAP "rc/send16mask.bmp" +send16masknoshadow BITMAP "rc/send16masknoshadow.bmp" +send20 BITMAP "rc/send20.bmp" +send20mask BITMAP "rc/send20mask.bmp" +addressbook16 BITMAP "rc/addressbook16.bmp" +addressbook16mask BITMAP "rc/addressbook16mask.bmp" +addressbook20 BITMAP "rc/addressbook20.bmp" +addressbook20mask BITMAP "rc/addressbook20mask.bmp" diff --git a/uibase.cpp b/uibase.cpp new file mode 100644 index 00000000..2972e9af --- /dev/null +++ b/uibase.cpp @@ -0,0 +1,1825 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +/////////////////////////////////////////////////////////////////////////// +// C++ code generated with wxFormBuilder (version Apr 16 2008) +// http://www.wxformbuilder.org/ +// +// PLEASE DO "NOT" EDIT THIS FILE! +/////////////////////////////////////////////////////////////////////////// + +#include "uibase.h" + +/////////////////////////////////////////////////////////////////////////// + +CMainFrameBase::CMainFrameBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE)); + + m_menubar = new wxMenuBar(0); + m_menubar->SetBackgroundColour(wxColour(240, 240, 240)); + + m_menuFile = new wxMenu(); + wxMenuItem* m_menuFileExit; + m_menuFileExit = new wxMenuItem(m_menuFile, wxID_ANY, wxString(wxT("E&xit")) , wxEmptyString, wxITEM_NORMAL); + m_menuFile->Append(m_menuFileExit); + + m_menubar->Append(m_menuFile, wxT("&File")); + + m_menuOptions = new wxMenu(); + wxMenuItem* m_menuOptionsGenerateBitcoins; + m_menuOptionsGenerateBitcoins = new wxMenuItem(m_menuOptions, wxID_OPTIONSGENERATEBITCOINS, wxString(wxT("&Generate Coins")) , wxEmptyString, wxITEM_CHECK); + m_menuOptions->Append(m_menuOptionsGenerateBitcoins); + + wxMenuItem* m_menuChangeYourAddress; + m_menuChangeYourAddress = new wxMenuItem(m_menuOptions, wxID_ANY, wxString(wxT("&Change Your Address...")) , wxEmptyString, wxITEM_NORMAL); + m_menuOptions->Append(m_menuChangeYourAddress); + + wxMenuItem* m_menuOptionsOptions; + m_menuOptionsOptions = new wxMenuItem(m_menuOptions, wxID_ANY, wxString(wxT("&Options...")) , wxEmptyString, wxITEM_NORMAL); + m_menuOptions->Append(m_menuOptionsOptions); + + m_menubar->Append(m_menuOptions, wxT("&Options")); + + m_menuHelp = new wxMenu(); + wxMenuItem* m_menuHelpAbout; + m_menuHelpAbout = new wxMenuItem(m_menuHelp, wxID_ANY, wxString(wxT("&About...")) , wxEmptyString, wxITEM_NORMAL); + m_menuHelp->Append(m_menuHelpAbout); + + m_menubar->Append(m_menuHelp, wxT("&Help")); + + this->SetMenuBar(m_menubar); + + m_toolBar = this->CreateToolBar(wxTB_FLAT|wxTB_HORZ_TEXT, wxID_ANY); + m_toolBar->SetToolBitmapSize(wxSize(20,20)); + m_toolBar->SetToolSeparation(1); + m_toolBar->SetFont(wxFont(wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString)); + + m_toolBar->AddTool(wxID_BUTTONSEND, wxT("&Send Coins"), wxBitmap(wxT("send20"), wxBITMAP_TYPE_RESOURCE), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + m_toolBar->AddTool(wxID_BUTTONRECEIVE, wxT("&Address Book"), wxBitmap(wxT("addressbook20"), wxBITMAP_TYPE_RESOURCE), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + m_toolBar->Realize(); + + m_statusBar = this->CreateStatusBar(1, wxST_SIZEGRIP, wxID_ANY); + m_statusBar->SetBackgroundColour(wxColour(240, 240, 240)); + + wxBoxSizer* bSizer2; + bSizer2 = new wxBoxSizer(wxVERTICAL); + + + bSizer2->Add(0, 2, 0, wxEXPAND, 5); + + wxBoxSizer* bSizer85; + bSizer85 = new wxBoxSizer(wxHORIZONTAL); + + m_staticText32 = new wxStaticText(this, wxID_ANY, wxT("Your Bitcoin Address:"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText32->Wrap(-1); + bSizer85->Add(m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxLEFT, 5); + + m_textCtrlAddress = new wxTextCtrl(this, wxID_TEXTCTRLADDRESS, wxEmptyString, wxDefaultPosition, wxSize(250,-1), wxTE_READONLY); + m_textCtrlAddress->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); + + bSizer85->Add(m_textCtrlAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_buttonCopy = new wxButton(this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); + bSizer85->Add(m_buttonCopy, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5); + + m_button91 = new wxButton(this, wxID_BUTTONCHANGE, wxT("C&hange..."), wxDefaultPosition, wxDefaultSize, 0); + m_button91->Hide(); + + bSizer85->Add(m_button91, 0, wxRIGHT, 5); + + + bSizer85->Add(0, 0, 0, wxEXPAND, 5); + + bSizer2->Add(bSizer85, 0, wxEXPAND|wxRIGHT|wxLEFT, 5); + + wxBoxSizer* bSizer3; + bSizer3 = new wxBoxSizer(wxHORIZONTAL); + + m_panel14 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); + wxBoxSizer* bSizer66; + bSizer66 = new wxBoxSizer(wxHORIZONTAL); + + m_staticText41 = new wxStaticText(m_panel14, wxID_ANY, wxT("Balance:"), wxDefaultPosition, wxSize(-1,15), 0); + m_staticText41->Wrap(-1); + bSizer66->Add(m_staticText41, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5); + + m_staticTextBalance = new wxStaticText(m_panel14, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(120,15), wxALIGN_RIGHT|wxST_NO_AUTORESIZE); + m_staticTextBalance->Wrap(-1); + m_staticTextBalance->SetFont(wxFont(8, 70, 90, 90, false, wxEmptyString)); + m_staticTextBalance->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); + + bSizer66->Add(m_staticTextBalance, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); + + m_panel14->SetSizer(bSizer66); + m_panel14->Layout(); + bSizer66->Fit(m_panel14); + bSizer3->Add(m_panel14, 1, wxEXPAND|wxALIGN_BOTTOM|wxALL, 5); + + + bSizer3->Add(0, 0, 0, wxEXPAND, 5); + + wxString m_choiceFilterChoices[] = { wxT(" All"), wxT(" Sent"), wxT(" Received"), wxT(" In Progress") }; + int m_choiceFilterNChoices = sizeof(m_choiceFilterChoices) / sizeof(wxString); + m_choiceFilter = new wxChoice(this, wxID_ANY, wxDefaultPosition, wxSize(110,-1), m_choiceFilterNChoices, m_choiceFilterChoices, 0); + m_choiceFilter->SetSelection(0); + m_choiceFilter->Hide(); + + bSizer3->Add(m_choiceFilter, 0, wxALIGN_BOTTOM|wxTOP|wxRIGHT|wxLEFT, 5); + + bSizer2->Add(bSizer3, 0, wxEXPAND, 5); + + m_notebook = new wxNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0); + m_panel7 = new wxPanel(m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); + wxBoxSizer* bSizer157; + bSizer157 = new wxBoxSizer(wxVERTICAL); + + m_listCtrl = new wxListCtrl(m_panel7, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxALWAYS_SHOW_SB); + bSizer157->Add(m_listCtrl, 1, wxEXPAND|wxALL, 5); + + m_panel7->SetSizer(bSizer157); + m_panel7->Layout(); + bSizer157->Fit(m_panel7); + m_notebook->AddPage(m_panel7, wxT("All Transactions"), false); + + bSizer2->Add(m_notebook, 1, wxEXPAND, 5); + + wxBoxSizer* bSizer_TabsForFutureUse; + bSizer_TabsForFutureUse = new wxBoxSizer(wxVERTICAL); + + m_panel9 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); + m_panel9->Hide(); + + wxBoxSizer* bSizer159; + bSizer159 = new wxBoxSizer(wxVERTICAL); + + m_listCtrlEscrows = new wxListCtrl(m_panel9, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); + bSizer159->Add(m_listCtrlEscrows, 1, wxALL|wxEXPAND, 5); + + m_panel9->SetSizer(bSizer159); + m_panel9->Layout(); + bSizer159->Fit(m_panel9); + bSizer_TabsForFutureUse->Add(m_panel9, 1, wxEXPAND | wxALL, 5); + + m_panel8 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); + m_panel8->Hide(); + + wxBoxSizer* bSizer158; + bSizer158 = new wxBoxSizer(wxVERTICAL); + + m_listCtrlOrdersSent = new wxListCtrl(m_panel8, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); + bSizer158->Add(m_listCtrlOrdersSent, 1, wxALL|wxEXPAND, 5); + + m_panel8->SetSizer(bSizer158); + m_panel8->Layout(); + bSizer158->Fit(m_panel8); + bSizer_TabsForFutureUse->Add(m_panel8, 1, wxEXPAND | wxALL, 5); + + m_panel10 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); + m_panel10->Hide(); + + wxBoxSizer* bSizer160; + bSizer160 = new wxBoxSizer(wxVERTICAL); + + m_listCtrlProductsSent = new wxListCtrl(m_panel10, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); + bSizer160->Add(m_listCtrlProductsSent, 1, wxALL|wxEXPAND, 5); + + m_panel10->SetSizer(bSizer160); + m_panel10->Layout(); + bSizer160->Fit(m_panel10); + bSizer_TabsForFutureUse->Add(m_panel10, 1, wxEXPAND | wxALL, 5); + + m_panel11 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); + m_panel11->Hide(); + + wxBoxSizer* bSizer161; + bSizer161 = new wxBoxSizer(wxVERTICAL); + + m_listCtrlOrdersReceived = new wxListCtrl(m_panel11, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); + bSizer161->Add(m_listCtrlOrdersReceived, 1, wxALL|wxEXPAND, 5); + + m_panel11->SetSizer(bSizer161); + m_panel11->Layout(); + bSizer161->Fit(m_panel11); + bSizer_TabsForFutureUse->Add(m_panel11, 1, wxEXPAND | wxALL, 5); + + bSizer2->Add(bSizer_TabsForFutureUse, 1, wxEXPAND, 5); + + this->SetSizer(bSizer2); + this->Layout(); + + // Connect Events + this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CMainFrameBase::OnClose)); + this->Connect(wxEVT_IDLE, wxIdleEventHandler(CMainFrameBase::OnIdle)); + this->Connect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Connect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaint)); + this->Connect(m_menuFileExit->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuFileExit)); + this->Connect(m_menuOptionsGenerateBitcoins->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsGenerate)); + this->Connect(m_menuChangeYourAddress->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsChangeYourAddress)); + this->Connect(m_menuOptionsOptions->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsOptions)); + this->Connect(m_menuHelpAbout->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuHelpAbout)); + this->Connect(wxID_BUTTONSEND, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonSend)); + this->Connect(wxID_BUTTONRECEIVE, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonAddressBook)); + m_textCtrlAddress->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CMainFrameBase::OnKeyDown), NULL, this); + m_textCtrlAddress->Connect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Connect(wxEVT_SET_FOCUS, wxFocusEventHandler(CMainFrameBase::OnSetFocusAddress), NULL, this); + m_buttonCopy->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonCopy), NULL, this); + m_button91->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonChange), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler(CMainFrameBase::OnListColBeginDrag), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedAllTransactions), NULL, this); + m_listCtrl->Connect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaintListCtrl), NULL, this); + m_listCtrlOrdersSent->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersSent), NULL, this); + m_listCtrlProductsSent->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedProductsSent), NULL, this); + m_listCtrlOrdersReceived->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersReceived), NULL, this); +} + +CMainFrameBase::~CMainFrameBase() +{ + // Disconnect Events + this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CMainFrameBase::OnClose)); + this->Disconnect(wxEVT_IDLE, wxIdleEventHandler(CMainFrameBase::OnIdle)); + this->Disconnect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); + this->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaint)); + this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuFileExit)); + this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsGenerate)); + this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsChangeYourAddress)); + this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsOptions)); + this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuHelpAbout)); + this->Disconnect(wxID_BUTTONSEND, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonSend)); + this->Disconnect(wxID_BUTTONRECEIVE, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonAddressBook)); + m_textCtrlAddress->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CMainFrameBase::OnKeyDown), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_SET_FOCUS, wxFocusEventHandler(CMainFrameBase::OnSetFocusAddress), NULL, this); + m_buttonCopy->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonCopy), NULL, this); + m_button91->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonChange), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler(CMainFrameBase::OnListColBeginDrag), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedAllTransactions), NULL, this); + m_listCtrl->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaintListCtrl), NULL, this); + m_listCtrlOrdersSent->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersSent), NULL, this); + m_listCtrlProductsSent->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedProductsSent), NULL, this); + m_listCtrlOrdersReceived->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersReceived), NULL, this); +} + +CTxDetailsDialogBase::CTxDetailsDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer64; + bSizer64 = new wxBoxSizer(wxVERTICAL); + + wxBoxSizer* bSizer66; + bSizer66 = new wxBoxSizer(wxVERTICAL); + + m_htmlWin = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO); + bSizer66->Add(m_htmlWin, 1, wxALL|wxEXPAND, 5); + + bSizer64->Add(bSizer66, 1, wxEXPAND, 5); + + wxBoxSizer* bSizer65; + bSizer65 = new wxBoxSizer(wxVERTICAL); + + m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(85,25), 0); + bSizer65->Add(m_buttonOK, 0, wxALL, 5); + + bSizer64->Add(bSizer65, 0, wxALIGN_RIGHT, 5); + + this->SetSizer(bSizer64); + this->Layout(); + + // Connect Events + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CTxDetailsDialogBase::OnButtonOK), NULL, this); +} + +CTxDetailsDialogBase::~CTxDetailsDialogBase() +{ + // Disconnect Events + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CTxDetailsDialogBase::OnButtonOK), NULL, this); +} + +COptionsDialogBase::COptionsDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer55; + bSizer55 = new wxBoxSizer(wxVERTICAL); + + wxBoxSizer* bSizer57; + bSizer57 = new wxBoxSizer(wxVERTICAL); + + + bSizer57->Add(0, 20, 0, wxEXPAND, 5); + + m_staticText32 = new wxStaticText(this, wxID_ANY, wxT("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0); + m_staticText32->Wrap(-1); + bSizer57->Add(m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); + + wxBoxSizer* bSizer56; + bSizer56 = new wxBoxSizer(wxHORIZONTAL); + + m_staticText31 = new wxStaticText(this, wxID_ANY, wxT("Transaction fee:"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText31->Wrap(-1); + bSizer56->Add(m_staticText31, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5); + + m_textCtrlTransactionFee = new wxTextCtrl(this, wxID_TRANSACTIONFEE, wxEmptyString, wxDefaultPosition, wxSize(70,-1), 0); + bSizer56->Add(m_textCtrlTransactionFee, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + bSizer57->Add(bSizer56, 0, wxEXPAND, 5); + + bSizer55->Add(bSizer57, 1, wxEXPAND|wxLEFT, 5); + + wxBoxSizer* bSizer58; + bSizer58 = new wxBoxSizer(wxHORIZONTAL); + + m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(85,25), 0); + bSizer58->Add(m_buttonOK, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer58->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer55->Add(bSizer58, 0, wxALIGN_RIGHT, 5); + + this->SetSizer(bSizer55); + this->Layout(); + + // Connect Events + m_textCtrlTransactionFee->Connect(wxEVT_KILL_FOCUS, wxFocusEventHandler(COptionsDialogBase::OnKillFocusTransactionFee), NULL, this); + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonCancel), NULL, this); +} + +COptionsDialogBase::~COptionsDialogBase() +{ + // Disconnect Events + m_textCtrlTransactionFee->Disconnect(wxEVT_KILL_FOCUS, wxFocusEventHandler(COptionsDialogBase::OnKillFocusTransactionFee), NULL, this); + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonCancel), NULL, this); +} + +CAboutDialogBase::CAboutDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer60; + bSizer60 = new wxBoxSizer(wxVERTICAL); + + wxBoxSizer* bSizer62; + bSizer62 = new wxBoxSizer(wxHORIZONTAL); + + + bSizer62->Add(60, 0, 0, wxEXPAND, 5); + + wxBoxSizer* bSizer63; + bSizer63 = new wxBoxSizer(wxVERTICAL); + + + bSizer63->Add(0, 50, 0, wxEXPAND, 5); + + wxBoxSizer* bSizer64; + bSizer64 = new wxBoxSizer(wxHORIZONTAL); + + m_staticText40 = new wxStaticText(this, wxID_ANY, wxT("Bitcoin "), wxDefaultPosition, wxDefaultSize, 0); + m_staticText40->Wrap(-1); + m_staticText40->SetFont(wxFont(10, 74, 90, 92, false, wxT("Tahoma"))); + + bSizer64->Add(m_staticText40, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxLEFT, 5); + + m_staticTextVersion = new wxStaticText(this, wxID_ANY, wxT("version"), wxDefaultPosition, wxDefaultSize, 0); + m_staticTextVersion->Wrap(-1); + m_staticTextVersion->SetFont(wxFont(10, 74, 90, 90, false, wxT("Tahoma"))); + + bSizer64->Add(m_staticTextVersion, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxRIGHT, 5); + + bSizer63->Add(bSizer64, 0, wxEXPAND, 5); + + + bSizer63->Add(0, 4, 0, wxEXPAND, 5); + + m_staticTextMain = new wxStaticText(this, wxID_ANY, wxT("Copyright © 2009 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0); + m_staticTextMain->Wrap(400); + bSizer63->Add(m_staticTextMain, 0, wxALL, 5); + + + bSizer63->Add(0, 0, 1, wxEXPAND, 5); + + bSizer62->Add(bSizer63, 1, wxEXPAND, 5); + + bSizer60->Add(bSizer62, 1, wxEXPAND, 5); + + wxBoxSizer* bSizer61; + bSizer61 = new wxBoxSizer(wxHORIZONTAL); + + + bSizer61->Add(0, 0, 1, wxEXPAND, 5); + + m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(85,25), 0); + bSizer61->Add(m_buttonOK, 0, wxALL, 5); + + bSizer60->Add(bSizer61, 0, wxALIGN_RIGHT|wxEXPAND, 5); + + this->SetSizer(bSizer60); + this->Layout(); + + // Connect Events + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAboutDialogBase::OnButtonOK), NULL, this); +} + +CAboutDialogBase::~CAboutDialogBase() +{ + // Disconnect Events + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAboutDialogBase::OnButtonOK), NULL, this); +} + +CSendDialogBase::CSendDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer21; + bSizer21 = new wxBoxSizer(wxVERTICAL); + + + bSizer21->Add(0, 5, 0, wxEXPAND, 5); + + wxFlexGridSizer* fgSizer1; + fgSizer1 = new wxFlexGridSizer(3, 2, 0, 0); + fgSizer1->AddGrowableCol(1); + fgSizer1->SetFlexibleDirection(wxBOTH); + fgSizer1->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); + + + fgSizer1->Add(0, 0, 0, wxEXPAND, 5); + + m_staticText14 = new wxStaticText(this, wxID_ANY, wxT("Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, \nor Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online."), wxDefaultPosition, wxDefaultSize, 0); + m_staticText14->Wrap(-1); + fgSizer1->Add(m_staticText14, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + wxBoxSizer* bSizer47; + bSizer47 = new wxBoxSizer(wxHORIZONTAL); + + bSizer47->SetMinSize(wxSize(70,-1)); + + bSizer47->Add(0, 0, 1, wxEXPAND, 5); + + m_bitmapCheckMark = new wxStaticBitmap(this, wxID_ANY, wxICON(check), wxDefaultPosition, wxSize(16,16), 0); + bSizer47->Add(m_bitmapCheckMark, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_staticText36 = new wxStaticText(this, wxID_ANY, wxT("Pay &To:"), wxDefaultPosition, wxSize(-1,-1), wxALIGN_RIGHT); + m_staticText36->Wrap(-1); + bSizer47->Add(m_staticText36, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5); + + fgSizer1->Add(bSizer47, 1, wxEXPAND|wxLEFT, 5); + + wxBoxSizer* bSizer19; + bSizer19 = new wxBoxSizer(wxHORIZONTAL); + + m_textCtrlAddress = new wxTextCtrl(this, wxID_TEXTCTRLPAYTO, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + bSizer19->Add(m_textCtrlAddress, 1, wxALIGN_CENTER_VERTICAL|wxALL, 5); + + m_buttonPaste = new wxButton(this, wxID_BUTTONPASTE, wxT("&Paste"), wxDefaultPosition, wxSize(-1,-1), wxBU_EXACTFIT); + bSizer19->Add(m_buttonPaste, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5); + + m_buttonAddress = new wxButton(this, wxID_BUTTONADDRESSBOOK, wxT(" Address &Book..."), wxDefaultPosition, wxDefaultSize, 0); + bSizer19->Add(m_buttonAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5); + + fgSizer1->Add(bSizer19, 1, wxEXPAND|wxRIGHT, 5); + + m_staticText19 = new wxStaticText(this, wxID_ANY, wxT("&Amount:"), wxDefaultPosition, wxSize(-1,-1), wxALIGN_RIGHT); + m_staticText19->Wrap(-1); + fgSizer1->Add(m_staticText19, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT|wxALIGN_RIGHT, 5); + + m_textCtrlAmount = new wxTextCtrl(this, wxID_TEXTCTRLAMOUNT, wxEmptyString, wxDefaultPosition, wxSize(145,-1), 0); + m_textCtrlAmount->SetMaxLength(20); + m_textCtrlAmount->SetFont(wxFont(wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString)); + + fgSizer1->Add(m_textCtrlAmount, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); + + m_staticText20 = new wxStaticText(this, wxID_ANY, wxT("T&ransfer:"), wxDefaultPosition, wxSize(-1,-1), wxALIGN_RIGHT); + m_staticText20->Wrap(-1); + m_staticText20->Hide(); + + fgSizer1->Add(m_staticText20, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); + + wxString m_choiceTransferTypeChoices[] = { wxT(" Standard") }; + int m_choiceTransferTypeNChoices = sizeof(m_choiceTransferTypeChoices) / sizeof(wxString); + m_choiceTransferType = new wxChoice(this, wxID_CHOICETRANSFERTYPE, wxDefaultPosition, wxDefaultSize, m_choiceTransferTypeNChoices, m_choiceTransferTypeChoices, 0); + m_choiceTransferType->SetSelection(0); + m_choiceTransferType->Hide(); + + fgSizer1->Add(m_choiceTransferType, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); + + + fgSizer1->Add(0, 3, 0, wxEXPAND, 5); + + + fgSizer1->Add(0, 0, 0, wxEXPAND, 5); + + bSizer21->Add(fgSizer1, 0, wxEXPAND|wxLEFT, 5); + + wxBoxSizer* bSizer672; + bSizer672 = new wxBoxSizer(wxHORIZONTAL); + + wxBoxSizer* bSizer681; + bSizer681 = new wxBoxSizer(wxVERTICAL); + + m_staticTextFrom = new wxStaticText(this, wxID_ANY, wxT("&From:"), wxDefaultPosition, wxDefaultSize, 0); + m_staticTextFrom->Wrap(-1); + bSizer681->Add(m_staticTextFrom, 0, wxBOTTOM|wxLEFT, 5); + + m_textCtrlFrom = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + bSizer681->Add(m_textCtrlFrom, 0, wxLEFT|wxEXPAND, 5); + + bSizer672->Add(bSizer681, 1, wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT, 5); + + bSizer21->Add(bSizer672, 0, wxEXPAND, 5); + + wxBoxSizer* bSizer67; + bSizer67 = new wxBoxSizer(wxHORIZONTAL); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer(wxVERTICAL); + + m_staticTextMessage = new wxStaticText(this, wxID_ANY, wxT("&Message:"), wxDefaultPosition, wxDefaultSize, 0); + m_staticTextMessage->Wrap(-1); + bSizer68->Add(m_staticTextMessage, 0, wxTOP|wxBOTTOM|wxLEFT, 5); + + m_textCtrlMessage = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); + bSizer68->Add(m_textCtrlMessage, 1, wxEXPAND|wxLEFT, 5); + + bSizer67->Add(bSizer68, 1, wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT, 5); + + bSizer21->Add(bSizer67, 1, wxEXPAND, 5); + + wxBoxSizer* bSizer23; + bSizer23 = new wxBoxSizer(wxHORIZONTAL); + + + bSizer23->Add(0, 0, 1, wxEXPAND, 5); + + m_buttonSend = new wxButton(this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonSend->SetFont(wxFont(wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString)); + m_buttonSend->SetMinSize(wxSize(85,25)); + + bSizer23->Add(m_buttonSend, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer23->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer21->Add(bSizer23, 0, wxEXPAND, 5); + + this->SetSizer(bSizer21); + this->Layout(); + + // Connect Events + m_textCtrlAddress->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_textCtrlAddress->Connect(wxEVT_COMMAND_TEXT_UPDATED, wxCommandEventHandler(CSendDialogBase::OnTextAddress), NULL, this); + m_buttonPaste->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonPaste), NULL, this); + m_buttonAddress->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonAddressBook), NULL, this); + m_textCtrlAmount->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_textCtrlAmount->Connect(wxEVT_KILL_FOCUS, wxFocusEventHandler(CSendDialogBase::OnKillFocusAmount), NULL, this); + m_textCtrlFrom->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_textCtrlMessage->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_buttonSend->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonSend), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonCancel), NULL, this); +} + +CSendDialogBase::~CSendDialogBase() +{ + // Disconnect Events + m_textCtrlAddress->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_textCtrlAddress->Disconnect(wxEVT_COMMAND_TEXT_UPDATED, wxCommandEventHandler(CSendDialogBase::OnTextAddress), NULL, this); + m_buttonPaste->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonPaste), NULL, this); + m_buttonAddress->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonAddressBook), NULL, this); + m_textCtrlAmount->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_textCtrlAmount->Disconnect(wxEVT_KILL_FOCUS, wxFocusEventHandler(CSendDialogBase::OnKillFocusAmount), NULL, this); + m_textCtrlFrom->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_textCtrlMessage->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); + m_buttonSend->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonSend), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonCancel), NULL, this); +} + +CSendingDialogBase::CSendingDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer(wxVERTICAL); + + m_staticTextSending = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,14), 0); + m_staticTextSending->Wrap(-1); + bSizer68->Add(m_staticTextSending, 0, wxALIGN_CENTER_VERTICAL|wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 8); + + m_textCtrlStatus = new wxTextCtrl(this, wxID_ANY, wxT("\n\nConnecting..."), wxDefaultPosition, wxDefaultSize, wxTE_CENTRE|wxTE_MULTILINE|wxTE_NO_VSCROLL|wxTE_READONLY|wxNO_BORDER); + m_textCtrlStatus->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE)); + + bSizer68->Add(m_textCtrlStatus, 1, wxEXPAND|wxRIGHT|wxLEFT, 10); + + wxBoxSizer* bSizer69; + bSizer69 = new wxBoxSizer(wxHORIZONTAL); + + + bSizer69->Add(0, 0, 1, wxEXPAND, 5); + + m_buttonOK = new wxButton(this, wxID_ANY, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonOK->Enable(false); + m_buttonOK->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonOK, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer68->Add(bSizer69, 0, wxEXPAND, 5); + + this->SetSizer(bSizer68); + this->Layout(); + + // Connect Events + this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CSendingDialogBase::OnClose)); + this->Connect(wxEVT_PAINT, wxPaintEventHandler(CSendingDialogBase::OnPaint)); + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonCancel), NULL, this); +} + +CSendingDialogBase::~CSendingDialogBase() +{ + // Disconnect Events + this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CSendingDialogBase::OnClose)); + this->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CSendingDialogBase::OnPaint)); + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonCancel), NULL, this); +} + +CYourAddressDialogBase::CYourAddressDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer(wxVERTICAL); + + + bSizer68->Add(0, 5, 0, wxEXPAND, 5); + + m_staticText45 = new wxStaticText(this, wxID_ANY, wxT("These are your Bitcoin addresses for receiving payments. You may want to give a different one to each sender so you can keep track of who is paying you. The highlighted address is displayed in the main window."), wxDefaultPosition, wxDefaultSize, 0); + m_staticText45->Wrap(590); + bSizer68->Add(m_staticText45, 0, wxALL, 5); + + m_listCtrl = new wxListCtrl(this, wxID_LISTCTRL, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING); + bSizer68->Add(m_listCtrl, 1, wxALL|wxEXPAND, 5); + + wxBoxSizer* bSizer69; + bSizer69 = new wxBoxSizer(wxHORIZONTAL); + + + bSizer69->Add(0, 0, 1, wxEXPAND, 5); + + m_buttonRename = new wxButton(this, wxID_BUTTONRENAME, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0); + m_buttonRename->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonRename, 0, wxALL, 5); + + m_buttonNew = new wxButton(this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonNew->SetMinSize(wxSize(110,25)); + + bSizer69->Add(m_buttonNew, 0, wxALL, 5); + + m_buttonCopy = new wxButton(this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonCopy->SetMinSize(wxSize(120,25)); + + bSizer69->Add(m_buttonCopy, 0, wxALL, 5); + + m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonOK->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonOK, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonCancel->Hide(); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer68->Add(bSizer69, 0, wxEXPAND, 5); + + this->SetSizer(bSizer68); + this->Layout(); + + // Connect Events + this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CYourAddressDialogBase::OnClose)); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CYourAddressDialogBase::OnListEndLabelEdit), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CYourAddressDialogBase::OnListItemActivated), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CYourAddressDialogBase::OnListItemSelected), NULL, this); + m_buttonRename->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonRename), NULL, this); + m_buttonNew->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonNew), NULL, this); + m_buttonCopy->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCopy), NULL, this); + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCancel), NULL, this); +} + +CYourAddressDialogBase::~CYourAddressDialogBase() +{ + // Disconnect Events + this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CYourAddressDialogBase::OnClose)); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CYourAddressDialogBase::OnListEndLabelEdit), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CYourAddressDialogBase::OnListItemActivated), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CYourAddressDialogBase::OnListItemSelected), NULL, this); + m_buttonRename->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonRename), NULL, this); + m_buttonNew->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonNew), NULL, this); + m_buttonCopy->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCopy), NULL, this); + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCancel), NULL, this); +} + +CAddressBookDialogBase::CAddressBookDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer(wxVERTICAL); + + + bSizer68->Add(0, 5, 0, wxEXPAND, 5); + + m_staticText55 = new wxStaticText(this, wxID_ANY, wxT("Bitcoin Address"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText55->Wrap(-1); + m_staticText55->Hide(); + + bSizer68->Add(m_staticText55, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + m_listCtrl = new wxListCtrl(this, wxID_LISTCTRL, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING); + bSizer68->Add(m_listCtrl, 1, wxALL|wxEXPAND, 5); + + wxBoxSizer* bSizer69; + bSizer69 = new wxBoxSizer(wxHORIZONTAL); + + + bSizer69->Add(0, 0, 1, wxEXPAND, 5); + + m_buttonEdit = new wxButton(this, wxID_BUTTONEDIT, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0); + m_buttonEdit->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonEdit, 0, wxALL, 5); + + m_buttonNew = new wxButton(this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxDefaultSize, 0); + m_buttonNew->SetMinSize(wxSize(110,25)); + + bSizer69->Add(m_buttonNew, 0, wxALL, 5); + + m_buttonDelete = new wxButton(this, wxID_BUTTONDELETE, wxT("&Delete"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonDelete->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonDelete, 0, wxALL, 5); + + m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonOK->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonOK, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer69->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer68->Add(bSizer69, 0, wxEXPAND, 5); + + this->SetSizer(bSizer68); + this->Layout(); + + // Connect Events + this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CAddressBookDialogBase::OnClose)); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CAddressBookDialogBase::OnListEndLabelEdit), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CAddressBookDialogBase::OnListItemActivated), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CAddressBookDialogBase::OnListItemSelected), NULL, this); + m_buttonEdit->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonEdit), NULL, this); + m_buttonNew->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonNew), NULL, this); + m_buttonDelete->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonDelete), NULL, this); + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonCancel), NULL, this); +} + +CAddressBookDialogBase::~CAddressBookDialogBase() +{ + // Disconnect Events + this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CAddressBookDialogBase::OnClose)); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CAddressBookDialogBase::OnListEndLabelEdit), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CAddressBookDialogBase::OnListItemActivated), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CAddressBookDialogBase::OnListItemSelected), NULL, this); + m_buttonEdit->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonEdit), NULL, this); + m_buttonNew->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonNew), NULL, this); + m_buttonDelete->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonDelete), NULL, this); + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonCancel), NULL, this); +} + +CProductsDialogBase::CProductsDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer22; + bSizer22 = new wxBoxSizer(wxVERTICAL); + + wxBoxSizer* bSizer23; + bSizer23 = new wxBoxSizer(wxHORIZONTAL); + + m_comboBoxCategory = new wxComboBox(this, wxID_ANY, wxT("(Any Category)"), wxDefaultPosition, wxSize(150,-1), 0, NULL, 0); + m_comboBoxCategory->Append(wxT("(Any Category)")); + bSizer23->Add(m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlSearch = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + bSizer23->Add(m_textCtrlSearch, 1, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonSearch = new wxButton(this, wxID_ANY, wxT("&Search"), wxDefaultPosition, wxDefaultSize, 0); + bSizer23->Add(m_buttonSearch, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + bSizer22->Add(bSizer23, 0, wxEXPAND|wxTOP|wxBOTTOM|wxRIGHT, 5); + + m_listCtrl = new wxListCtrl(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); + bSizer22->Add(m_listCtrl, 1, wxALL|wxEXPAND, 5); + + this->SetSizer(bSizer22); + this->Layout(); + + // Connect Events + m_comboBoxCategory->Connect(wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler(CProductsDialogBase::OnCombobox), NULL, this); + m_textCtrlSearch->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CProductsDialogBase::OnKeyDown), NULL, this); + m_buttonSearch->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CProductsDialogBase::OnButtonSearch), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CProductsDialogBase::OnListItemActivated), NULL, this); +} + +CProductsDialogBase::~CProductsDialogBase() +{ + // Disconnect Events + m_comboBoxCategory->Disconnect(wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler(CProductsDialogBase::OnCombobox), NULL, this); + m_textCtrlSearch->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CProductsDialogBase::OnKeyDown), NULL, this); + m_buttonSearch->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CProductsDialogBase::OnButtonSearch), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CProductsDialogBase::OnListItemActivated), NULL, this); +} + +CEditProductDialogBase::CEditProductDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); + + wxBoxSizer* bSizer20; + bSizer20 = new wxBoxSizer(wxVERTICAL); + + m_scrolledWindow = new wxScrolledWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL); + m_scrolledWindow->SetScrollRate(5, 5); + m_scrolledWindow->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); + + wxBoxSizer* bSizer21; + bSizer21 = new wxBoxSizer(wxVERTICAL); + + wxFlexGridSizer* fgSizer8; + fgSizer8 = new wxFlexGridSizer(0, 2, 0, 0); + fgSizer8->AddGrowableCol(1); + fgSizer8->SetFlexibleDirection(wxBOTH); + fgSizer8->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); + + m_staticText106 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Category"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); + m_staticText106->Wrap(-1); + fgSizer8->Add(m_staticText106, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); + + m_comboBoxCategory = new wxComboBox(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0, NULL, 0); + m_comboBoxCategory->SetMinSize(wxSize(180,-1)); + + fgSizer8->Add(m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_staticText108 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Title"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); + m_staticText108->Wrap(-1); + fgSizer8->Add(m_staticText108, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); + + m_textCtrlTitle = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + fgSizer8->Add(m_textCtrlTitle, 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5); + + m_staticText107 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Price"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); + m_staticText107->Wrap(-1); + fgSizer8->Add(m_staticText107, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); + + m_textCtrlPrice = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlPrice->SetMinSize(wxSize(105,-1)); + + fgSizer8->Add(m_textCtrlPrice, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + bSizer21->Add(fgSizer8, 0, wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 5); + + m_staticText22 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Page 1: Description"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText22->Wrap(-1); + bSizer21->Add(m_staticText22, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + m_textCtrlDescription = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); + m_textCtrlDescription->SetMinSize(wxSize(-1,170)); + + bSizer21->Add(m_textCtrlDescription, 0, wxALL|wxEXPAND, 5); + + m_staticText23 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Page 2: Order Form"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText23->Wrap(-1); + bSizer21->Add(m_staticText23, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + m_textCtrlInstructions = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); + m_textCtrlInstructions->SetMinSize(wxSize(-1,120)); + + bSizer21->Add(m_textCtrlInstructions, 0, wxEXPAND|wxALL, 5); + + fgSizer5 = new wxFlexGridSizer(0, 3, 0, 0); + fgSizer5->AddGrowableCol(1); + fgSizer5->SetFlexibleDirection(wxBOTH); + fgSizer5->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); + + m_staticText24 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Label"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText24->Wrap(-1); + fgSizer5->Add(m_staticText24, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5); + + m_staticText25 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Comma separated list of choices, or leave blank for text field"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText25->Wrap(-1); + fgSizer5->Add(m_staticText25, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5); + + + fgSizer5->Add(0, 0, 1, wxEXPAND, 5); + + m_textCtrlLabel0 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel0->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel0, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField0 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField0, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel0 = new wxButton(m_scrolledWindow, wxID_DEL0, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel0, 0, wxRIGHT|wxLEFT|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlLabel1 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel1->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel1, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField1 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField1, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel1 = new wxButton(m_scrolledWindow, wxID_DEL1, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel1, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel2 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel2->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel2, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField2 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField2, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel2 = new wxButton(m_scrolledWindow, wxID_DEL2, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel2, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel3 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel3->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel3, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField3 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField3, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel3 = new wxButton(m_scrolledWindow, wxID_DEL3, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel3, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel4 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel4->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel4, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField4 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField4, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel4 = new wxButton(m_scrolledWindow, wxID_DEL4, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel4, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel5 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel5->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel5, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField5 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField5, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel5 = new wxButton(m_scrolledWindow, wxID_DEL5, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel5, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel6 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel6->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel6, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField6 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField6, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel6 = new wxButton(m_scrolledWindow, wxID_DEL6, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel6, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel7 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel7->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel7, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField7 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField7, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel7 = new wxButton(m_scrolledWindow, wxID_DEL7, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel7, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel8 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel8->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel8, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField8 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField8, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel8 = new wxButton(m_scrolledWindow, wxID_DEL8, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel8, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel9 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel9->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel9, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField9 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField9, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel9 = new wxButton(m_scrolledWindow, wxID_DEL9, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel9, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel10 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel10->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel10, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField10 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField10, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel10 = new wxButton(m_scrolledWindow, wxID_DEL10, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel10, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel11 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel11->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel11, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField11 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField11, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel11 = new wxButton(m_scrolledWindow, wxID_DEL11, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel11, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel12 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel12->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel12, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField12 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField12, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel12 = new wxButton(m_scrolledWindow, wxID_DEL12, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel12, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel13 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel13->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel13, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField13 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField13, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel13 = new wxButton(m_scrolledWindow, wxID_DEL13, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel13, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel14 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel14->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel14, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField14 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField14, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel14 = new wxButton(m_scrolledWindow, wxID_DEL14, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel14, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel15 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel15->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel15, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField15 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField15, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel15 = new wxButton(m_scrolledWindow, wxID_DEL15, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel15, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel16 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel16->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel16, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField16 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField16, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel16 = new wxButton(m_scrolledWindow, wxID_DEL16, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel16, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel17 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel17->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel17, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField17 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField17, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel17 = new wxButton(m_scrolledWindow, wxID_DEL17, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel17, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel18 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel18->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel18, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField18 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField18, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel18 = new wxButton(m_scrolledWindow, wxID_DEL18, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel18, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + m_textCtrlLabel19 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_textCtrlLabel19->SetMinSize(wxSize(150,-1)); + + fgSizer5->Add(m_textCtrlLabel19, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); + + m_textCtrlField19 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); + fgSizer5->Add(m_textCtrlField19, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); + + m_buttonDel19 = new wxButton(m_scrolledWindow, wxID_DEL19, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); + fgSizer5->Add(m_buttonDel19, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); + + bSizer21->Add(fgSizer5, 0, wxEXPAND, 5); + + wxBoxSizer* bSizer25; + bSizer25 = new wxBoxSizer(wxHORIZONTAL); + + m_buttonAddField = new wxButton(m_scrolledWindow, wxID_ANY, wxT("&Add Field"), wxDefaultPosition, wxDefaultSize, 0); + bSizer25->Add(m_buttonAddField, 0, wxALL, 5); + + bSizer21->Add(bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5); + + m_scrolledWindow->SetSizer(bSizer21); + m_scrolledWindow->Layout(); + bSizer21->Fit(m_scrolledWindow); + bSizer20->Add(m_scrolledWindow, 1, wxEXPAND|wxALL, 5); + + wxBoxSizer* bSizer26; + bSizer26 = new wxBoxSizer(wxHORIZONTAL); + + m_buttonOK = new wxButton(this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonOK->SetMinSize(wxSize(85,25)); + + bSizer26->Add(m_buttonOK, 0, wxALL, 5); + + m_buttonPreview = new wxButton(this, wxID_BUTTONPREVIEW, wxT("&Preview"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonPreview->SetMinSize(wxSize(85,25)); + + bSizer26->Add(m_buttonPreview, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer26->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer20->Add(bSizer26, 0, wxALIGN_RIGHT, 5); + + this->SetSizer(bSizer20); + this->Layout(); + + // Connect Events + m_textCtrlTitle->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlPrice->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlDescription->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlInstructions->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlLabel0->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField0->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel0->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel0), NULL, this); + m_textCtrlLabel1->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField1->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel1->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel1), NULL, this); + m_textCtrlLabel2->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField2->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel2->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel2), NULL, this); + m_textCtrlLabel3->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField3->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel3->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel3), NULL, this); + m_textCtrlLabel4->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField4->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel4->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel4), NULL, this); + m_textCtrlLabel5->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField5->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel5->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel5), NULL, this); + m_textCtrlLabel6->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField6->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel6->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel6), NULL, this); + m_textCtrlLabel7->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField7->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel7->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel7), NULL, this); + m_textCtrlLabel8->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField8->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel8->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel8), NULL, this); + m_textCtrlLabel9->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField9->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel9->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel9), NULL, this); + m_textCtrlLabel10->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField10->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel10->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel10), NULL, this); + m_textCtrlLabel11->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField11->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel11->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel11), NULL, this); + m_textCtrlLabel12->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField12->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel12->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel12), NULL, this); + m_textCtrlLabel13->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField13->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel13->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel13), NULL, this); + m_textCtrlLabel14->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField14->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel14->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel14), NULL, this); + m_textCtrlLabel15->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField15->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel15->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel15), NULL, this); + m_textCtrlLabel16->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField16->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel16->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel16), NULL, this); + m_textCtrlLabel17->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField17->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel17->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel17), NULL, this); + m_textCtrlLabel18->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField18->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel18->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel18), NULL, this); + m_textCtrlLabel19->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField19->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel19->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel19), NULL, this); + m_buttonAddField->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonAddField), NULL, this); + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonSend), NULL, this); + m_buttonPreview->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonPreview), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonCancel), NULL, this); +} + +CEditProductDialogBase::~CEditProductDialogBase() +{ + // Disconnect Events + m_textCtrlTitle->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlPrice->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlDescription->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlInstructions->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlLabel0->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField0->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel0->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel0), NULL, this); + m_textCtrlLabel1->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField1->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel1->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel1), NULL, this); + m_textCtrlLabel2->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField2->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel2->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel2), NULL, this); + m_textCtrlLabel3->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField3->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel3->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel3), NULL, this); + m_textCtrlLabel4->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField4->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel4->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel4), NULL, this); + m_textCtrlLabel5->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField5->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel5->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel5), NULL, this); + m_textCtrlLabel6->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField6->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel6->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel6), NULL, this); + m_textCtrlLabel7->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField7->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel7->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel7), NULL, this); + m_textCtrlLabel8->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField8->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel8->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel8), NULL, this); + m_textCtrlLabel9->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField9->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel9->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel9), NULL, this); + m_textCtrlLabel10->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField10->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel10->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel10), NULL, this); + m_textCtrlLabel11->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField11->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel11->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel11), NULL, this); + m_textCtrlLabel12->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField12->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel12->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel12), NULL, this); + m_textCtrlLabel13->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField13->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel13->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel13), NULL, this); + m_textCtrlLabel14->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField14->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel14->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel14), NULL, this); + m_textCtrlLabel15->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField15->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel15->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel15), NULL, this); + m_textCtrlLabel16->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField16->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel16->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel16), NULL, this); + m_textCtrlLabel17->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField17->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel17->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel17), NULL, this); + m_textCtrlLabel18->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField18->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel18->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel18), NULL, this); + m_textCtrlLabel19->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_textCtrlField19->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); + m_buttonDel19->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel19), NULL, this); + m_buttonAddField->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonAddField), NULL, this); + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonSend), NULL, this); + m_buttonPreview->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonPreview), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonCancel), NULL, this); +} + +CViewProductDialogBase::CViewProductDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); + + wxBoxSizer* bSizer20; + bSizer20 = new wxBoxSizer(wxVERTICAL); + + wxBoxSizer* bSizer116; + bSizer116 = new wxBoxSizer(wxHORIZONTAL); + + m_htmlWinReviews = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO); + m_htmlWinReviews->Hide(); + + bSizer116->Add(m_htmlWinReviews, 1, wxALL|wxEXPAND, 5); + + m_scrolledWindow = new wxScrolledWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL); + m_scrolledWindow->SetScrollRate(5, 5); + m_scrolledWindow->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); + + wxBoxSizer* bSizer21; + bSizer21 = new wxBoxSizer(wxVERTICAL); + + m_richTextHeading = new wxRichTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,50), wxTE_READONLY|wxNO_BORDER); + bSizer21->Add(m_richTextHeading, 0, wxEXPAND, 5); + + m_staticTextInstructions = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Order Form instructions here\nmultiple lines\n1\n2\n3\n4\n5\n6"), wxDefaultPosition, wxDefaultSize, 0); + m_staticTextInstructions->Wrap(-1); + bSizer21->Add(m_staticTextInstructions, 0, wxALL|wxEXPAND, 5); + + wxBoxSizer* bSizer25; + bSizer25 = new wxBoxSizer(wxHORIZONTAL); + + m_buttonSubmitForm = new wxButton(m_scrolledWindow, wxID_BUTTONSAMPLE, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0); + bSizer25->Add(m_buttonSubmitForm, 0, wxALL, 5); + + m_buttonCancelForm = new wxButton(m_scrolledWindow, wxID_CANCEL2, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); + bSizer25->Add(m_buttonCancelForm, 0, wxALL, 5); + + bSizer21->Add(bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5); + + m_scrolledWindow->SetSizer(bSizer21); + m_scrolledWindow->Layout(); + bSizer21->Fit(m_scrolledWindow); + bSizer116->Add(m_scrolledWindow, 1, wxEXPAND|wxALL, 5); + + bSizer20->Add(bSizer116, 1, wxEXPAND, 5); + + wxBoxSizer* bSizer26; + bSizer26 = new wxBoxSizer(wxHORIZONTAL); + + m_buttonBack = new wxButton(this, wxID_BUTTONBACK, wxT("< &Back "), wxDefaultPosition, wxDefaultSize, 0); + m_buttonBack->Enable(false); + m_buttonBack->SetMinSize(wxSize(85,25)); + + bSizer26->Add(m_buttonBack, 0, wxALL, 5); + + m_buttonNext = new wxButton(this, wxID_BUTTONNEXT, wxT(" &Next >"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonNext->SetMinSize(wxSize(85,25)); + + bSizer26->Add(m_buttonNext, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer26->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer20->Add(bSizer26, 0, wxALIGN_RIGHT, 5); + + this->SetSizer(bSizer20); + this->Layout(); + + // Connect Events + m_buttonSubmitForm->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonSubmitForm), NULL, this); + m_buttonCancelForm->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancelForm), NULL, this); + m_buttonBack->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonBack), NULL, this); + m_buttonNext->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonNext), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancel), NULL, this); +} + +CViewProductDialogBase::~CViewProductDialogBase() +{ + // Disconnect Events + m_buttonSubmitForm->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonSubmitForm), NULL, this); + m_buttonCancelForm->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancelForm), NULL, this); + m_buttonBack->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonBack), NULL, this); + m_buttonNext->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonNext), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancel), NULL, this); +} + +CViewOrderDialogBase::CViewOrderDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); + + wxBoxSizer* bSizer20; + bSizer20 = new wxBoxSizer(wxVERTICAL); + + wxBoxSizer* bSizer116; + bSizer116 = new wxBoxSizer(wxHORIZONTAL); + + m_htmlWin = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO); + bSizer116->Add(m_htmlWin, 1, wxALL|wxEXPAND, 5); + + bSizer20->Add(bSizer116, 1, wxEXPAND, 5); + + wxBoxSizer* bSizer26; + bSizer26 = new wxBoxSizer(wxHORIZONTAL); + + m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonOK->SetMinSize(wxSize(85,25)); + + bSizer26->Add(m_buttonOK, 0, wxALL, 5); + + bSizer20->Add(bSizer26, 0, wxALIGN_RIGHT, 5); + + this->SetSizer(bSizer20); + this->Layout(); + + // Connect Events + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewOrderDialogBase::OnButtonOK), NULL, this); +} + +CViewOrderDialogBase::~CViewOrderDialogBase() +{ + // Disconnect Events + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewOrderDialogBase::OnButtonOK), NULL, this); +} + +CEditReviewDialogBase::CEditReviewDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); + + wxBoxSizer* bSizer112; + bSizer112 = new wxBoxSizer(wxVERTICAL); + + + bSizer112->Add(0, 3, 0, 0, 5); + + m_staticTextSeller = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_staticTextSeller->Wrap(-1); + bSizer112->Add(m_staticTextSeller, 0, wxALL|wxEXPAND, 5); + + + bSizer112->Add(0, 3, 0, 0, 5); + + m_staticText110 = new wxStaticText(this, wxID_ANY, wxT("Rating"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText110->Wrap(-1); + bSizer112->Add(m_staticText110, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + wxString m_choiceStarsChoices[] = { wxT(" 1 star"), wxT(" 2 stars"), wxT(" 3 stars"), wxT(" 4 stars"), wxT(" 5 stars") }; + int m_choiceStarsNChoices = sizeof(m_choiceStarsChoices) / sizeof(wxString); + m_choiceStars = new wxChoice(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, m_choiceStarsNChoices, m_choiceStarsChoices, 0); + m_choiceStars->SetSelection(0); + bSizer112->Add(m_choiceStars, 0, wxALL, 5); + + m_staticText43 = new wxStaticText(this, wxID_ANY, wxT("Review"), wxDefaultPosition, wxDefaultSize, 0); + m_staticText43->Wrap(-1); + bSizer112->Add(m_staticText43, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + m_textCtrlReview = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); + bSizer112->Add(m_textCtrlReview, 1, wxALL|wxEXPAND, 5); + + wxBoxSizer* bSizer113; + bSizer113 = new wxBoxSizer(wxHORIZONTAL); + + m_buttonSubmit = new wxButton(this, wxID_SUBMIT, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonSubmit->SetMinSize(wxSize(85,25)); + + bSizer113->Add(m_buttonSubmit, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer113->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer112->Add(bSizer113, 0, wxALIGN_RIGHT, 5); + + this->SetSizer(bSizer112); + this->Layout(); + + // Connect Events + m_textCtrlReview->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditReviewDialogBase::OnKeyDown), NULL, this); + m_buttonSubmit->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonSubmit), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonCancel), NULL, this); +} + +CEditReviewDialogBase::~CEditReviewDialogBase() +{ + // Disconnect Events + m_textCtrlReview->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditReviewDialogBase::OnKeyDown), NULL, this); + m_buttonSubmit->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonSubmit), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonCancel), NULL, this); +} + +CPokerLobbyDialogBase::CPokerLobbyDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE)); + + wxBoxSizer* bSizer156; + bSizer156 = new wxBoxSizer(wxHORIZONTAL); + + m_treeCtrl = new wxTreeCtrl(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT); + m_treeCtrl->SetMinSize(wxSize(130,-1)); + + bSizer156->Add(m_treeCtrl, 0, wxEXPAND|wxTOP|wxBOTTOM|wxLEFT, 5); + + wxBoxSizer* bSizer172; + bSizer172 = new wxBoxSizer(wxVERTICAL); + + m_listCtrl = new wxListCtrl(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); + bSizer172->Add(m_listCtrl, 1, wxEXPAND|wxALL, 5); + + m_buttonNewTable = new wxButton(this, wxID_OPENNEWTABLE, wxT("&Open New Table"), wxDefaultPosition, wxDefaultSize, 0); + bSizer172->Add(m_buttonNewTable, 0, wxALL, 5); + + bSizer156->Add(bSizer172, 1, wxEXPAND, 5); + + this->SetSizer(bSizer156); + this->Layout(); + + // Connect Events + m_treeCtrl->Connect(wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler(CPokerLobbyDialogBase::OnTreeSelChanged), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemActivated), NULL, this); + m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemSelected), NULL, this); + m_buttonNewTable->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerLobbyDialogBase::OnButtonNewTable), NULL, this); +} + +CPokerLobbyDialogBase::~CPokerLobbyDialogBase() +{ + // Disconnect Events + m_treeCtrl->Disconnect(wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler(CPokerLobbyDialogBase::OnTreeSelChanged), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemActivated), NULL, this); + m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemSelected), NULL, this); + m_buttonNewTable->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerLobbyDialogBase::OnButtonNewTable), NULL, this); +} + +CPokerDialogBase::CPokerDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer174; + bSizer174 = new wxBoxSizer(wxVERTICAL); + + m_checkSitOut = new wxCheckBox(this, wxID_ANY, wxT("Deal Me Out"), wxDefaultPosition, wxDefaultSize, 0); + + bSizer174->Add(m_checkSitOut, 0, wxALL, 5); + + m_buttonDealHand = new wxButton(this, wxID_DEALHAND, wxT("&Deal Hand"), wxDefaultPosition, wxSize(150,25), 0); + bSizer174->Add(m_buttonDealHand, 0, wxALL, 5); + + m_buttonFold = new wxButton(this, wxID_FOLD, wxT("&Fold"), wxDefaultPosition, wxSize(80,25), 0); + bSizer174->Add(m_buttonFold, 0, wxALL, 5); + + m_buttonCall = new wxButton(this, wxID_CALL, wxT("&Call"), wxDefaultPosition, wxSize(80,25), 0); + bSizer174->Add(m_buttonCall, 0, wxALL, 5); + + m_buttonRaise = new wxButton(this, wxID_RAISE, wxT("&Raise"), wxDefaultPosition, wxSize(80,25), 0); + bSizer174->Add(m_buttonRaise, 0, wxALL, 5); + + m_buttonLeaveTable = new wxButton(this, wxID_LEAVETABLE, wxT("&Leave Table"), wxDefaultPosition, wxSize(90,25), 0); + bSizer174->Add(m_buttonLeaveTable, 0, wxALL, 5); + + m_textDitchPlayer = new wxTextCtrl(this, wxID_DITCHPLAYER, wxEmptyString, wxDefaultPosition, wxSize(45,-1), wxTE_PROCESS_ENTER); + bSizer174->Add(m_textDitchPlayer, 0, wxALL, 5); + + m_checkPreFold = new wxCheckBox(this, wxID_ANY, wxT("FOLD"), wxDefaultPosition, wxSize(100,-1), 0); + + bSizer174->Add(m_checkPreFold, 0, wxALL, 5); + + m_checkPreCall = new wxCheckBox(this, wxID_ANY, wxT("CALL"), wxDefaultPosition, wxSize(100,-1), 0); + + bSizer174->Add(m_checkPreCall, 0, wxALL, 5); + + m_checkPreCallAny = new wxCheckBox(this, wxID_ANY, wxT("CALL ANY"), wxDefaultPosition, wxSize(100,-1), 0); + + bSizer174->Add(m_checkPreCallAny, 0, wxALL, 5); + + m_checkPreRaise = new wxCheckBox(this, wxID_ANY, wxT("RAISE"), wxDefaultPosition, wxSize(100,-1), 0); + + bSizer174->Add(m_checkPreRaise, 0, wxALL, 5); + + m_checkPreRaiseAny = new wxCheckBox(this, wxID_ANY, wxT("RAISE ANY"), wxDefaultPosition, wxSize(100,-1), 0); + + bSizer174->Add(m_checkPreRaiseAny, 0, wxALL, 5); + + this->SetSizer(bSizer174); + this->Layout(); + m_statusBar = this->CreateStatusBar(1, wxST_SIZEGRIP, wxID_ANY); + + // Connect Events + this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CPokerDialogBase::OnClose)); + this->Connect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_LEFT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_RIGHT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_MOTION, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Connect(wxEVT_PAINT, wxPaintEventHandler(CPokerDialogBase::OnPaint)); + this->Connect(wxEVT_SIZE, wxSizeEventHandler(CPokerDialogBase::OnSize)); + m_checkSitOut->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckSitOut), NULL, this); + m_buttonDealHand->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonDealHand), NULL, this); + m_buttonFold->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonFold), NULL, this); + m_buttonCall->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonCall), NULL, this); + m_buttonRaise->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonRaise), NULL, this); + m_buttonLeaveTable->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonLeaveTable), NULL, this); + m_textDitchPlayer->Connect(wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler(CPokerDialogBase::OnDitchPlayer), NULL, this); + m_checkPreFold->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreFold), NULL, this); + m_checkPreCall->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCall), NULL, this); + m_checkPreCallAny->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCallAny), NULL, this); + m_checkPreRaise->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaise), NULL, this); + m_checkPreRaiseAny->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaiseAny), NULL, this); +} + +CPokerDialogBase::~CPokerDialogBase() +{ + // Disconnect Events + this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CPokerDialogBase::OnClose)); + this->Disconnect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_LEFT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_RIGHT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_MOTION, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); + this->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CPokerDialogBase::OnPaint)); + this->Disconnect(wxEVT_SIZE, wxSizeEventHandler(CPokerDialogBase::OnSize)); + m_checkSitOut->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckSitOut), NULL, this); + m_buttonDealHand->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonDealHand), NULL, this); + m_buttonFold->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonFold), NULL, this); + m_buttonCall->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonCall), NULL, this); + m_buttonRaise->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonRaise), NULL, this); + m_buttonLeaveTable->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonLeaveTable), NULL, this); + m_textDitchPlayer->Disconnect(wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler(CPokerDialogBase::OnDitchPlayer), NULL, this); + m_checkPreFold->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreFold), NULL, this); + m_checkPreCall->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCall), NULL, this); + m_checkPreCallAny->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCallAny), NULL, this); + m_checkPreRaise->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaise), NULL, this); + m_checkPreRaiseAny->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaiseAny), NULL, this); +} + +CGetTextFromUserDialogBase::CGetTextFromUserDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +{ + this->SetSizeHints(wxDefaultSize, wxDefaultSize); + + wxBoxSizer* bSizer79; + bSizer79 = new wxBoxSizer(wxVERTICAL); + + wxBoxSizer* bSizer81; + bSizer81 = new wxBoxSizer(wxVERTICAL); + + + bSizer81->Add(0, 0, 1, wxEXPAND, 5); + + m_staticTextMessage1 = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_staticTextMessage1->Wrap(-1); + bSizer81->Add(m_staticTextMessage1, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + m_textCtrl1 = new wxTextCtrl(this, wxID_TEXTCTRL, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER); + bSizer81->Add(m_textCtrl1, 0, wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL, 5); + + m_staticTextMessage2 = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); + m_staticTextMessage2->Wrap(-1); + m_staticTextMessage2->Hide(); + + bSizer81->Add(m_staticTextMessage2, 0, wxTOP|wxRIGHT|wxLEFT, 5); + + m_textCtrl2 = new wxTextCtrl(this, wxID_TEXTCTRL, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER); + m_textCtrl2->Hide(); + + bSizer81->Add(m_textCtrl2, 0, wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL, 5); + + + bSizer81->Add(0, 0, 1, wxEXPAND, 5); + + bSizer79->Add(bSizer81, 1, wxEXPAND|wxALL, 10); + + wxBoxSizer* bSizer80; + bSizer80 = new wxBoxSizer(wxHORIZONTAL); + + + bSizer80->Add(0, 0, 1, wxEXPAND, 5); + + m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(-1,-1), 0); + m_buttonOK->SetMinSize(wxSize(85,25)); + + bSizer80->Add(m_buttonOK, 0, wxALL, 5); + + m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); + m_buttonCancel->SetMinSize(wxSize(85,25)); + + bSizer80->Add(m_buttonCancel, 0, wxALL, 5); + + bSizer79->Add(bSizer80, 0, wxEXPAND, 5); + + this->SetSizer(bSizer79); + this->Layout(); + + // Connect Events + this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CGetTextFromUserDialogBase::OnClose)); + m_textCtrl1->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); + m_textCtrl2->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); + m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonCancel), NULL, this); +} + +CGetTextFromUserDialogBase::~CGetTextFromUserDialogBase() +{ + // Disconnect Events + this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CGetTextFromUserDialogBase::OnClose)); + m_textCtrl1->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); + m_textCtrl2->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); + m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonOK), NULL, this); + m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonCancel), NULL, this); +} diff --git a/uibase.h b/uibase.h new file mode 100644 index 00000000..bfcd8ecc --- /dev/null +++ b/uibase.h @@ -0,0 +1,723 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +/////////////////////////////////////////////////////////////////////////// +// C++ code generated with wxFormBuilder (version Apr 16 2008) +// http://www.wxformbuilder.org/ +// +// PLEASE DO "NOT" EDIT THIS FILE! +/////////////////////////////////////////////////////////////////////////// + +#ifndef __uibase__ +#define __uibase__ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +/////////////////////////////////////////////////////////////////////////// + +#define wxID_MAINFRAME 1000 +#define wxID_OPTIONSGENERATEBITCOINS 1001 +#define wxID_BUTTONSEND 1002 +#define wxID_BUTTONRECEIVE 1003 +#define wxID_TEXTCTRLADDRESS 1004 +#define wxID_BUTTONCOPY 1005 +#define wxID_BUTTONCHANGE 1006 +#define wxID_TRANSACTIONFEE 1007 +#define wxID_TEXTCTRLPAYTO 1008 +#define wxID_BUTTONPASTE 1009 +#define wxID_BUTTONADDRESSBOOK 1010 +#define wxID_TEXTCTRLAMOUNT 1011 +#define wxID_CHOICETRANSFERTYPE 1012 +#define wxID_LISTCTRL 1013 +#define wxID_BUTTONRENAME 1014 +#define wxID_BUTTONNEW 1015 +#define wxID_BUTTONEDIT 1016 +#define wxID_BUTTONDELETE 1017 +#define wxID_DEL0 1018 +#define wxID_DEL1 1019 +#define wxID_DEL2 1020 +#define wxID_DEL3 1021 +#define wxID_DEL4 1022 +#define wxID_DEL5 1023 +#define wxID_DEL6 1024 +#define wxID_DEL7 1025 +#define wxID_DEL8 1026 +#define wxID_DEL9 1027 +#define wxID_DEL10 1028 +#define wxID_DEL11 1029 +#define wxID_DEL12 1030 +#define wxID_DEL13 1031 +#define wxID_DEL14 1032 +#define wxID_DEL15 1033 +#define wxID_DEL16 1034 +#define wxID_DEL17 1035 +#define wxID_DEL18 1036 +#define wxID_DEL19 1037 +#define wxID_BUTTONPREVIEW 1038 +#define wxID_BUTTONSAMPLE 1039 +#define wxID_CANCEL2 1040 +#define wxID_BUTTONBACK 1041 +#define wxID_BUTTONNEXT 1042 +#define wxID_SUBMIT 1043 +#define wxID_OPENNEWTABLE 1044 +#define wxID_DEALHAND 1045 +#define wxID_FOLD 1046 +#define wxID_CALL 1047 +#define wxID_RAISE 1048 +#define wxID_LEAVETABLE 1049 +#define wxID_DITCHPLAYER 1050 +#define wxID_TEXTCTRL 1051 + +/////////////////////////////////////////////////////////////////////////////// +/// Class CMainFrameBase +/////////////////////////////////////////////////////////////////////////////// +class CMainFrameBase : public wxFrame +{ +private: + +protected: + wxMenuBar* m_menubar; + wxMenu* m_menuFile; + wxMenu* m_menuHelp; + wxToolBar* m_toolBar; + wxStatusBar* m_statusBar; + + wxStaticText* m_staticText32; + wxTextCtrl* m_textCtrlAddress; + wxButton* m_buttonCopy; + wxButton* m_button91; + + wxPanel* m_panel14; + wxStaticText* m_staticText41; + wxStaticText* m_staticTextBalance; + + wxChoice* m_choiceFilter; + wxNotebook* m_notebook; + wxPanel* m_panel7; + wxPanel* m_panel9; + wxPanel* m_panel8; + wxPanel* m_panel10; + wxPanel* m_panel11; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose(wxCloseEvent& event){ event.Skip(); } + virtual void OnIdle(wxIdleEvent& event){ event.Skip(); } + virtual void OnMouseEvents(wxMouseEvent& event){ event.Skip(); } + virtual void OnPaint(wxPaintEvent& event){ event.Skip(); } + virtual void OnMenuFileExit(wxCommandEvent& event){ event.Skip(); } + virtual void OnMenuOptionsGenerate(wxCommandEvent& event){ event.Skip(); } + virtual void OnMenuOptionsChangeYourAddress(wxCommandEvent& event){ event.Skip(); } + virtual void OnMenuOptionsOptions(wxCommandEvent& event){ event.Skip(); } + virtual void OnMenuHelpAbout(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonSend(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonAddressBook(wxCommandEvent& event){ event.Skip(); } + virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } + virtual void OnMouseEventsAddress(wxMouseEvent& event){ event.Skip(); } + virtual void OnSetFocusAddress(wxFocusEvent& event){ event.Skip(); } + virtual void OnButtonCopy(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonChange(wxCommandEvent& event){ event.Skip(); } + virtual void OnListColBeginDrag(wxListEvent& event){ event.Skip(); } + virtual void OnListItemActivatedAllTransactions(wxListEvent& event){ event.Skip(); } + virtual void OnPaintListCtrl(wxPaintEvent& event){ event.Skip(); } + virtual void OnListItemActivatedOrdersSent(wxListEvent& event){ event.Skip(); } + virtual void OnListItemActivatedProductsSent(wxListEvent& event){ event.Skip(); } + virtual void OnListItemActivatedOrdersReceived(wxListEvent& event){ event.Skip(); } + + +public: + wxMenu* m_menuOptions; + wxListCtrl* m_listCtrl; + wxListCtrl* m_listCtrlEscrows; + wxListCtrl* m_listCtrlOrdersSent; + wxListCtrl* m_listCtrlProductsSent; + wxListCtrl* m_listCtrlOrdersReceived; + CMainFrameBase(wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(705,484), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); + ~CMainFrameBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CTxDetailsDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CTxDetailsDialogBase : public wxDialog +{ +private: + +protected: + wxHtmlWindow* m_htmlWin; + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + + +public: + CTxDetailsDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Transaction Details"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(620,450), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); + ~CTxDetailsDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class COptionsDialogBase +/////////////////////////////////////////////////////////////////////////////// +class COptionsDialogBase : public wxDialog +{ +private: + +protected: + + wxStaticText* m_staticText32; + wxStaticText* m_staticText31; + wxTextCtrl* m_textCtrlTransactionFee; + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnKillFocusTransactionFee(wxFocusEvent& event){ event.Skip(); } + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + COptionsDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Options"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(500,261), long style = wxDEFAULT_DIALOG_STYLE); + ~COptionsDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CAboutDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CAboutDialogBase : public wxDialog +{ +private: + +protected: + + + wxStaticText* m_staticText40; + + wxStaticText* m_staticTextMain; + + + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + + +public: + wxStaticText* m_staticTextVersion; + CAboutDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("About Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(507,298), long style = wxDEFAULT_DIALOG_STYLE); + ~CAboutDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CSendDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CSendDialogBase : public wxDialog +{ +private: + +protected: + + + wxStaticText* m_staticText14; + + wxStaticBitmap* m_bitmapCheckMark; + wxStaticText* m_staticText36; + wxTextCtrl* m_textCtrlAddress; + wxButton* m_buttonPaste; + wxButton* m_buttonAddress; + wxStaticText* m_staticText19; + wxTextCtrl* m_textCtrlAmount; + wxStaticText* m_staticText20; + wxChoice* m_choiceTransferType; + + + wxStaticText* m_staticTextFrom; + wxTextCtrl* m_textCtrlFrom; + wxStaticText* m_staticTextMessage; + wxTextCtrl* m_textCtrlMessage; + + wxButton* m_buttonSend; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } + virtual void OnTextAddress(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonPaste(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonAddressBook(wxCommandEvent& event){ event.Skip(); } + virtual void OnKillFocusAmount(wxFocusEvent& event){ event.Skip(); } + virtual void OnButtonSend(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + CSendDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Send Coins"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(675,312), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); + ~CSendDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CSendingDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CSendingDialogBase : public wxDialog +{ +private: + +protected: + wxStaticText* m_staticTextSending; + wxTextCtrl* m_textCtrlStatus; + + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose(wxCloseEvent& event){ event.Skip(); } + virtual void OnPaint(wxPaintEvent& event){ event.Skip(); } + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + CSendingDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Sending..."), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(442,151), long style = wxDEFAULT_DIALOG_STYLE); + ~CSendingDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CYourAddressDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CYourAddressDialogBase : public wxDialog +{ +private: + +protected: + + wxStaticText* m_staticText45; + wxListCtrl* m_listCtrl; + + wxButton* m_buttonRename; + wxButton* m_buttonNew; + wxButton* m_buttonCopy; + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose(wxCloseEvent& event){ event.Skip(); } + virtual void OnListEndLabelEdit(wxListEvent& event){ event.Skip(); } + virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } + virtual void OnListItemSelected(wxListEvent& event){ event.Skip(); } + virtual void OnButtonRename(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonNew(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCopy(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + CYourAddressDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Your Bitcoin Addresses"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(610,390), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); + ~CYourAddressDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CAddressBookDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CAddressBookDialogBase : public wxDialog +{ +private: + +protected: + + wxStaticText* m_staticText55; + wxListCtrl* m_listCtrl; + + wxButton* m_buttonEdit; + wxButton* m_buttonNew; + wxButton* m_buttonDelete; + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose(wxCloseEvent& event){ event.Skip(); } + virtual void OnListEndLabelEdit(wxListEvent& event){ event.Skip(); } + virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } + virtual void OnListItemSelected(wxListEvent& event){ event.Skip(); } + virtual void OnButtonEdit(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonNew(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDelete(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + wxButton* m_buttonCancel; + CAddressBookDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Address Book"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(610,390), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); + ~CAddressBookDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CProductsDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CProductsDialogBase : public wxDialog +{ +private: + +protected: + wxComboBox* m_comboBoxCategory; + wxTextCtrl* m_textCtrlSearch; + wxButton* m_buttonSearch; + wxListCtrl* m_listCtrl; + + // Virtual event handlers, overide them in your derived class + virtual void OnCombobox(wxCommandEvent& event){ event.Skip(); } + virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } + virtual void OnButtonSearch(wxCommandEvent& event){ event.Skip(); } + virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } + + +public: + CProductsDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Marketplace"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(708,535), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); + ~CProductsDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CEditProductDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CEditProductDialogBase : public wxFrame +{ +private: + +protected: + wxScrolledWindow* m_scrolledWindow; + wxStaticText* m_staticText106; + wxComboBox* m_comboBoxCategory; + wxStaticText* m_staticText108; + wxTextCtrl* m_textCtrlTitle; + wxStaticText* m_staticText107; + wxTextCtrl* m_textCtrlPrice; + wxStaticText* m_staticText22; + wxTextCtrl* m_textCtrlDescription; + wxStaticText* m_staticText23; + wxTextCtrl* m_textCtrlInstructions; + wxStaticText* m_staticText24; + wxStaticText* m_staticText25; + + wxTextCtrl* m_textCtrlLabel0; + wxTextCtrl* m_textCtrlField0; + wxButton* m_buttonDel0; + wxTextCtrl* m_textCtrlLabel1; + wxTextCtrl* m_textCtrlField1; + wxButton* m_buttonDel1; + wxTextCtrl* m_textCtrlLabel2; + wxTextCtrl* m_textCtrlField2; + wxButton* m_buttonDel2; + wxTextCtrl* m_textCtrlLabel3; + wxTextCtrl* m_textCtrlField3; + wxButton* m_buttonDel3; + wxTextCtrl* m_textCtrlLabel4; + wxTextCtrl* m_textCtrlField4; + wxButton* m_buttonDel4; + wxTextCtrl* m_textCtrlLabel5; + wxTextCtrl* m_textCtrlField5; + wxButton* m_buttonDel5; + wxTextCtrl* m_textCtrlLabel6; + wxTextCtrl* m_textCtrlField6; + wxButton* m_buttonDel6; + wxTextCtrl* m_textCtrlLabel7; + wxTextCtrl* m_textCtrlField7; + wxButton* m_buttonDel7; + wxTextCtrl* m_textCtrlLabel8; + wxTextCtrl* m_textCtrlField8; + wxButton* m_buttonDel8; + wxTextCtrl* m_textCtrlLabel9; + wxTextCtrl* m_textCtrlField9; + wxButton* m_buttonDel9; + wxTextCtrl* m_textCtrlLabel10; + wxTextCtrl* m_textCtrlField10; + wxButton* m_buttonDel10; + wxTextCtrl* m_textCtrlLabel11; + wxTextCtrl* m_textCtrlField11; + wxButton* m_buttonDel11; + wxTextCtrl* m_textCtrlLabel12; + wxTextCtrl* m_textCtrlField12; + wxButton* m_buttonDel12; + wxTextCtrl* m_textCtrlLabel13; + wxTextCtrl* m_textCtrlField13; + wxButton* m_buttonDel13; + wxTextCtrl* m_textCtrlLabel14; + wxTextCtrl* m_textCtrlField14; + wxButton* m_buttonDel14; + wxTextCtrl* m_textCtrlLabel15; + wxTextCtrl* m_textCtrlField15; + wxButton* m_buttonDel15; + wxTextCtrl* m_textCtrlLabel16; + wxTextCtrl* m_textCtrlField16; + wxButton* m_buttonDel16; + wxTextCtrl* m_textCtrlLabel17; + wxTextCtrl* m_textCtrlField17; + wxButton* m_buttonDel17; + wxTextCtrl* m_textCtrlLabel18; + wxTextCtrl* m_textCtrlField18; + wxButton* m_buttonDel18; + wxTextCtrl* m_textCtrlLabel19; + wxTextCtrl* m_textCtrlField19; + wxButton* m_buttonDel19; + wxButton* m_buttonAddField; + wxButton* m_buttonOK; + wxButton* m_buttonPreview; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } + virtual void OnButtonDel0(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel1(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel2(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel3(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel4(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel5(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel6(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel7(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel8(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel9(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel10(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel11(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel12(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel13(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel14(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel15(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel16(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel17(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel18(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDel19(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonAddField(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonSend(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonPreview(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + wxFlexGridSizer* fgSizer5; + CEditProductDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Edit Product"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(660,640), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); + ~CEditProductDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CViewProductDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CViewProductDialogBase : public wxFrame +{ +private: + +protected: + wxHtmlWindow* m_htmlWinReviews; + wxScrolledWindow* m_scrolledWindow; + wxRichTextCtrl* m_richTextHeading; + wxStaticText* m_staticTextInstructions; + wxButton* m_buttonSubmitForm; + wxButton* m_buttonCancelForm; + wxButton* m_buttonBack; + wxButton* m_buttonNext; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonSubmitForm(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancelForm(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonBack(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonNext(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + CViewProductDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Order Form"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(630,520), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); + ~CViewProductDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CViewOrderDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CViewOrderDialogBase : public wxFrame +{ +private: + +protected: + wxHtmlWindow* m_htmlWin; + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + + +public: + CViewOrderDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("View Order"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(630,520), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); + ~CViewOrderDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CEditReviewDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CEditReviewDialogBase : public wxFrame +{ +private: + +protected: + + wxStaticText* m_staticTextSeller; + + wxStaticText* m_staticText110; + wxChoice* m_choiceStars; + wxStaticText* m_staticText43; + wxTextCtrl* m_textCtrlReview; + wxButton* m_buttonSubmit; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } + virtual void OnButtonSubmit(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + CEditReviewDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Enter Review"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(630,440), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); + ~CEditReviewDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CPokerLobbyDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CPokerLobbyDialogBase : public wxFrame +{ +private: + +protected: + wxTreeCtrl* m_treeCtrl; + wxListCtrl* m_listCtrl; + wxButton* m_buttonNewTable; + + // Virtual event handlers, overide them in your derived class + virtual void OnTreeSelChanged(wxTreeEvent& event){ event.Skip(); } + virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } + virtual void OnListItemSelected(wxListEvent& event){ event.Skip(); } + virtual void OnButtonNewTable(wxCommandEvent& event){ event.Skip(); } + + +public: + CPokerLobbyDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker Lobby"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(586,457), long style = wxDEFAULT_FRAME_STYLE|wxTAB_TRAVERSAL); + ~CPokerLobbyDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CPokerDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CPokerDialogBase : public wxFrame +{ +private: + +protected: + wxButton* m_buttonDealHand; + wxButton* m_buttonFold; + wxButton* m_buttonCall; + wxButton* m_buttonRaise; + wxButton* m_buttonLeaveTable; + wxTextCtrl* m_textDitchPlayer; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose(wxCloseEvent& event){ event.Skip(); } + virtual void OnMouseEvents(wxMouseEvent& event){ event.Skip(); } + virtual void OnPaint(wxPaintEvent& event){ event.Skip(); } + virtual void OnSize(wxSizeEvent& event){ event.Skip(); } + virtual void OnCheckSitOut(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonDealHand(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonFold(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCall(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonRaise(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonLeaveTable(wxCommandEvent& event){ event.Skip(); } + virtual void OnDitchPlayer(wxCommandEvent& event){ event.Skip(); } + virtual void OnCheckPreFold(wxCommandEvent& event){ event.Skip(); } + virtual void OnCheckPreCall(wxCommandEvent& event){ event.Skip(); } + virtual void OnCheckPreCallAny(wxCommandEvent& event){ event.Skip(); } + virtual void OnCheckPreRaise(wxCommandEvent& event){ event.Skip(); } + virtual void OnCheckPreRaiseAny(wxCommandEvent& event){ event.Skip(); } + + +public: + wxCheckBox* m_checkSitOut; + wxCheckBox* m_checkPreFold; + wxCheckBox* m_checkPreCall; + wxCheckBox* m_checkPreCallAny; + wxCheckBox* m_checkPreRaise; + wxCheckBox* m_checkPreRaiseAny; + wxStatusBar* m_statusBar; + CPokerDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(806,550), long style = wxDEFAULT_FRAME_STYLE|wxFRAME_NO_TASKBAR|wxFULL_REPAINT_ON_RESIZE|wxTAB_TRAVERSAL); + ~CPokerDialogBase(); + +}; + +/////////////////////////////////////////////////////////////////////////////// +/// Class CGetTextFromUserDialogBase +/////////////////////////////////////////////////////////////////////////////// +class CGetTextFromUserDialogBase : public wxDialog +{ +private: + +protected: + + wxStaticText* m_staticTextMessage1; + wxTextCtrl* m_textCtrl1; + wxStaticText* m_staticTextMessage2; + wxTextCtrl* m_textCtrl2; + + + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose(wxCloseEvent& event){ event.Skip(); } + virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } + virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } + virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } + + +public: + CGetTextFromUserDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxEmptyString, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(403,138), long style = wxDEFAULT_DIALOG_STYLE); + ~CGetTextFromUserDialogBase(); + +}; + +#endif //__uibase__ diff --git a/uint256.h b/uint256.h new file mode 100644 index 00000000..9a0c7704 --- /dev/null +++ b/uint256.h @@ -0,0 +1,750 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include +#include +#if defined(_MSC_VER) || defined(__BORLANDC__) +typedef __int64 int64; +typedef unsigned __int64 uint64; +#else +typedef long long int64; +typedef unsigned long long uint64; +#endif +#if defined(_MSC_VER) && _MSC_VER < 1300 +#define for if (false) ; else for +#endif + + +inline int Testuint256AdHoc(vector vArg); + + + +// We have to keep a separate base class without constructors +// so the compiler will let us use it in a union +template +class base_uint +{ +protected: + enum { WIDTH=BITS/32 }; + unsigned int pn[WIDTH]; +public: + + bool operator!() const + { + for (int i = 0; i < WIDTH; i++) + if (pn[i] != 0) + return false; + return true; + } + + const base_uint operator~() const + { + base_uint ret; + for (int i = 0; i < WIDTH; i++) + ret.pn[i] = ~pn[i]; + return ret; + } + + const base_uint operator-() const + { + base_uint ret; + for (int i = 0; i < WIDTH; i++) + ret.pn[i] = ~pn[i]; + ret++; + return ret; + } + + + base_uint& operator=(uint64 b) + { + pn[0] = (unsigned int)b; + pn[1] = (unsigned int)(b >> 32); + for (int i = 2; i < WIDTH; i++) + pn[i] = 0; + return *this; + } + + base_uint& operator^=(const base_uint& b) + { + for (int i = 0; i < WIDTH; i++) + pn[i] ^= b.pn[i]; + return *this; + } + + base_uint& operator&=(const base_uint& b) + { + for (int i = 0; i < WIDTH; i++) + pn[i] &= b.pn[i]; + return *this; + } + + base_uint& operator|=(const base_uint& b) + { + for (int i = 0; i < WIDTH; i++) + pn[i] |= b.pn[i]; + return *this; + } + + base_uint& operator^=(uint64 b) + { + pn[0] ^= (unsigned int)b; + pn[1] ^= (unsigned int)(b >> 32); + return *this; + } + + base_uint& operator&=(uint64 b) + { + pn[0] &= (unsigned int)b; + pn[1] &= (unsigned int)(b >> 32); + return *this; + } + + base_uint& operator|=(uint64 b) + { + pn[0] |= (unsigned int)b; + pn[1] |= (unsigned int)(b >> 32); + return *this; + } + + base_uint& operator<<=(unsigned int shift) + { + base_uint a(*this); + for (int i = 0; i < WIDTH; i++) + pn[i] = 0; + int k = shift / 32; + shift = shift % 32; + for (int i = 0; i < WIDTH; i++) + { + if (i+k+1 < WIDTH && shift != 0) + pn[i+k+1] |= (a.pn[i] >> (32-shift)); + if (i+k < WIDTH) + pn[i+k] |= (a.pn[i] << shift); + } + return *this; + } + + base_uint& operator>>=(unsigned int shift) + { + base_uint a(*this); + for (int i = 0; i < WIDTH; i++) + pn[i] = 0; + int k = shift / 32; + shift = shift % 32; + for (int i = 0; i < WIDTH; i++) + { + if (i-k-1 >= 0 && shift != 0) + pn[i-k-1] |= (a.pn[i] << (32-shift)); + if (i-k >= 0) + pn[i-k] |= (a.pn[i] >> shift); + } + return *this; + } + + base_uint& operator+=(const base_uint& b) + { + uint64 carry = 0; + for (int i = 0; i < WIDTH; i++) + { + uint64 n = carry + pn[i] + b.pn[i]; + pn[i] = n & 0xffffffff; + carry = n >> 32; + } + return *this; + } + + base_uint& operator-=(const base_uint& b) + { + *this += -b; + return *this; + } + + base_uint& operator+=(uint64 b64) + { + base_uint b; + b = b64; + *this += b; + return *this; + } + + base_uint& operator-=(uint64 b64) + { + base_uint b; + b = b64; + *this += -b; + return *this; + } + + + base_uint& operator++() + { + // prefix operator + int i = 0; + while (++pn[i] == 0 && i < WIDTH-1) + i++; + return *this; + } + + const base_uint operator++(int) + { + // postfix operator + const base_uint ret = *this; + ++(*this); + return ret; + } + + base_uint& operator--() + { + // prefix operator + int i = 0; + while (--pn[i] == -1 && i < WIDTH-1) + i++; + return *this; + } + + const base_uint operator--(int) + { + // postfix operator + const base_uint ret = *this; + --(*this); + return ret; + } + + + friend inline bool operator<(const base_uint& a, const base_uint& b) + { + for (int i = base_uint::WIDTH-1; i >= 0; i--) + { + if (a.pn[i] < b.pn[i]) + return true; + else if (a.pn[i] > b.pn[i]) + return false; + } + return false; + } + + friend inline bool operator<=(const base_uint& a, const base_uint& b) + { + for (int i = base_uint::WIDTH-1; i >= 0; i--) + { + if (a.pn[i] < b.pn[i]) + return true; + else if (a.pn[i] > b.pn[i]) + return false; + } + return true; + } + + friend inline bool operator>(const base_uint& a, const base_uint& b) + { + for (int i = base_uint::WIDTH-1; i >= 0; i--) + { + if (a.pn[i] > b.pn[i]) + return true; + else if (a.pn[i] < b.pn[i]) + return false; + } + return false; + } + + friend inline bool operator>=(const base_uint& a, const base_uint& b) + { + for (int i = base_uint::WIDTH-1; i >= 0; i--) + { + if (a.pn[i] > b.pn[i]) + return true; + else if (a.pn[i] < b.pn[i]) + return false; + } + return true; + } + + friend inline bool operator==(const base_uint& a, const base_uint& b) + { + for (int i = 0; i < base_uint::WIDTH; i++) + if (a.pn[i] != b.pn[i]) + return false; + return true; + } + + friend inline bool operator==(const base_uint& a, uint64 b) + { + if (a.pn[0] != (unsigned int)b) + return false; + if (a.pn[1] != (unsigned int)(b >> 32)) + return false; + for (int i = 2; i < base_uint::WIDTH; i++) + if (a.pn[i] != 0) + return false; + return true; + } + + friend inline bool operator!=(const base_uint& a, const base_uint& b) + { + return (!(a == b)); + } + + friend inline bool operator!=(const base_uint& a, uint64 b) + { + return (!(a == b)); + } + + + + std::string GetHex() const + { + char psz[sizeof(pn)*2 + 1]; + for (int i = 0; i < sizeof(pn); i++) + sprintf(psz + i*2, "%02x", ((unsigned char*)pn)[sizeof(pn) - i - 1]); + return string(psz, psz + sizeof(pn)*2); + } + + void SetHex(const std::string& str) + { + for (int i = 0; i < WIDTH; i++) + pn[i] = 0; + + // skip 0x + const char* psz = str.c_str(); + while (isspace(*psz)) + psz++; + if (psz[0] == '0' && tolower(psz[1]) == 'x') + psz += 2; + while (isspace(*psz)) + psz++; + + // hex string to uint + static char phexdigit[256] = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,1,2,3,4,5,6,7,8,9,0,0,0,0,0,0, 0,0xa,0xb,0xc,0xd,0xe,0xf,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0xa,0xb,0xc,0xd,0xe,0xf,0,0,0,0,0,0,0,0,0 }; + const char* pbegin = psz; + while (phexdigit[*psz] || *psz == '0') + psz++; + psz--; + unsigned char* p1 = (unsigned char*)pn; + unsigned char* pend = p1 + WIDTH * 4; + while (psz >= pbegin && p1 < pend) + { + *p1 = phexdigit[(unsigned char)*psz--]; + if (psz >= pbegin) + { + *p1 |= (phexdigit[(unsigned char)*psz--] << 4); + p1++; + } + } + } + + std::string ToString() const + { + return (GetHex()); + } + + unsigned char* begin() + { + return (unsigned char*)&pn[0]; + } + + unsigned char* end() + { + return (unsigned char*)&pn[WIDTH]; + } + + unsigned int size() + { + return sizeof(pn); + } + + + unsigned int GetSerializeSize(int nType=0, int nVersion=VERSION) const + { + return sizeof(pn); + } + + template + void Serialize(Stream& s, int nType=0, int nVersion=VERSION) const + { + s.write((char*)pn, sizeof(pn)); + } + + template + void Unserialize(Stream& s, int nType=0, int nVersion=VERSION) + { + s.read((char*)pn, sizeof(pn)); + } + + + friend class uint160; + friend class uint256; + friend inline int Testuint256AdHoc(vector vArg); +}; + +typedef base_uint<160> base_uint160; +typedef base_uint<256> base_uint256; + + + +// +// uint160 and uint256 could be implemented as templates, but to keep +// compile errors and debugging cleaner, they're copy and pasted. +// It's safe to search and replace 160 with 256 and vice versa. +// + + + +////////////////////////////////////////////////////////////////////////////// +// +// uint160 +// + +class uint160 : public base_uint160 +{ +public: + typedef base_uint160 basetype; + + uint160() + { + } + + uint160(const basetype& b) + { + for (int i = 0; i < WIDTH; i++) + pn[i] = b.pn[i]; + } + + uint160& operator=(const basetype& b) + { + for (int i = 0; i < WIDTH; i++) + pn[i] = b.pn[i]; + return *this; + } + + uint160(uint64 b) + { + pn[0] = (unsigned int)b; + pn[1] = (unsigned int)(b >> 32); + for (int i = 2; i < WIDTH; i++) + pn[i] = 0; + } + + uint160& operator=(uint64 b) + { + pn[0] = (unsigned int)b; + pn[1] = (unsigned int)(b >> 32); + for (int i = 2; i < WIDTH; i++) + pn[i] = 0; + return *this; + } + + explicit uint160(const std::string& str) + { + SetHex(str); + } + + explicit uint160(const std::vector& vch) + { + if (vch.size() == sizeof(pn)) + memcpy(pn, &vch[0], sizeof(pn)); + else + *this = 0; + } +}; + +inline bool operator==(const uint160& a, uint64 b) { return (base_uint160)a == b; } +inline bool operator!=(const uint160& a, uint64 b) { return (base_uint160)a != b; } +inline const uint160 operator<<(const base_uint160& a, unsigned int shift) { return uint160(a) <<= shift; } +inline const uint160 operator>>(const base_uint160& a, unsigned int shift) { return uint160(a) >>= shift; } +inline const uint160 operator<<(const uint160& a, unsigned int shift) { return uint160(a) <<= shift; } +inline const uint160 operator>>(const uint160& a, unsigned int shift) { return uint160(a) >>= shift; } + +inline const uint160 operator^(const base_uint160& a, const base_uint160& b) { return uint160(a) ^= b; } +inline const uint160 operator&(const base_uint160& a, const base_uint160& b) { return uint160(a) &= b; } +inline const uint160 operator|(const base_uint160& a, const base_uint160& b) { return uint160(a) |= b; } +inline const uint160 operator+(const base_uint160& a, const base_uint160& b) { return uint160(a) += b; } +inline const uint160 operator-(const base_uint160& a, const base_uint160& b) { return uint160(a) -= b; } + +inline bool operator<(const base_uint160& a, const uint160& b) { return (base_uint160)a < (base_uint160)b; } +inline bool operator<=(const base_uint160& a, const uint160& b) { return (base_uint160)a <= (base_uint160)b; } +inline bool operator>(const base_uint160& a, const uint160& b) { return (base_uint160)a > (base_uint160)b; } +inline bool operator>=(const base_uint160& a, const uint160& b) { return (base_uint160)a >= (base_uint160)b; } +inline bool operator==(const base_uint160& a, const uint160& b) { return (base_uint160)a == (base_uint160)b; } +inline bool operator!=(const base_uint160& a, const uint160& b) { return (base_uint160)a != (base_uint160)b; } +inline const uint160 operator^(const base_uint160& a, const uint160& b) { return (base_uint160)a ^ (base_uint160)b; } +inline const uint160 operator&(const base_uint160& a, const uint160& b) { return (base_uint160)a & (base_uint160)b; } +inline const uint160 operator|(const base_uint160& a, const uint160& b) { return (base_uint160)a | (base_uint160)b; } +inline const uint160 operator+(const base_uint160& a, const uint160& b) { return (base_uint160)a + (base_uint160)b; } +inline const uint160 operator-(const base_uint160& a, const uint160& b) { return (base_uint160)a - (base_uint160)b; } + +inline bool operator<(const uint160& a, const base_uint160& b) { return (base_uint160)a < (base_uint160)b; } +inline bool operator<=(const uint160& a, const base_uint160& b) { return (base_uint160)a <= (base_uint160)b; } +inline bool operator>(const uint160& a, const base_uint160& b) { return (base_uint160)a > (base_uint160)b; } +inline bool operator>=(const uint160& a, const base_uint160& b) { return (base_uint160)a >= (base_uint160)b; } +inline bool operator==(const uint160& a, const base_uint160& b) { return (base_uint160)a == (base_uint160)b; } +inline bool operator!=(const uint160& a, const base_uint160& b) { return (base_uint160)a != (base_uint160)b; } +inline const uint160 operator^(const uint160& a, const base_uint160& b) { return (base_uint160)a ^ (base_uint160)b; } +inline const uint160 operator&(const uint160& a, const base_uint160& b) { return (base_uint160)a & (base_uint160)b; } +inline const uint160 operator|(const uint160& a, const base_uint160& b) { return (base_uint160)a | (base_uint160)b; } +inline const uint160 operator+(const uint160& a, const base_uint160& b) { return (base_uint160)a + (base_uint160)b; } +inline const uint160 operator-(const uint160& a, const base_uint160& b) { return (base_uint160)a - (base_uint160)b; } + +inline bool operator<(const uint160& a, const uint160& b) { return (base_uint160)a < (base_uint160)b; } +inline bool operator<=(const uint160& a, const uint160& b) { return (base_uint160)a <= (base_uint160)b; } +inline bool operator>(const uint160& a, const uint160& b) { return (base_uint160)a > (base_uint160)b; } +inline bool operator>=(const uint160& a, const uint160& b) { return (base_uint160)a >= (base_uint160)b; } +inline bool operator==(const uint160& a, const uint160& b) { return (base_uint160)a == (base_uint160)b; } +inline bool operator!=(const uint160& a, const uint160& b) { return (base_uint160)a != (base_uint160)b; } +inline const uint160 operator^(const uint160& a, const uint160& b) { return (base_uint160)a ^ (base_uint160)b; } +inline const uint160 operator&(const uint160& a, const uint160& b) { return (base_uint160)a & (base_uint160)b; } +inline const uint160 operator|(const uint160& a, const uint160& b) { return (base_uint160)a | (base_uint160)b; } +inline const uint160 operator+(const uint160& a, const uint160& b) { return (base_uint160)a + (base_uint160)b; } +inline const uint160 operator-(const uint160& a, const uint160& b) { return (base_uint160)a - (base_uint160)b; } + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// uint256 +// + +class uint256 : public base_uint256 +{ +public: + typedef base_uint256 basetype; + + uint256() + { + } + + uint256(const basetype& b) + { + for (int i = 0; i < WIDTH; i++) + pn[i] = b.pn[i]; + } + + uint256& operator=(const basetype& b) + { + for (int i = 0; i < WIDTH; i++) + pn[i] = b.pn[i]; + return *this; + } + + uint256(uint64 b) + { + pn[0] = (unsigned int)b; + pn[1] = (unsigned int)(b >> 32); + for (int i = 2; i < WIDTH; i++) + pn[i] = 0; + } + + uint256& operator=(uint64 b) + { + pn[0] = (unsigned int)b; + pn[1] = (unsigned int)(b >> 32); + for (int i = 2; i < WIDTH; i++) + pn[i] = 0; + return *this; + } + + explicit uint256(const std::string& str) + { + SetHex(str); + } + + explicit uint256(const std::vector& vch) + { + if (vch.size() == sizeof(pn)) + memcpy(pn, &vch[0], sizeof(pn)); + else + *this = 0; + } +}; + +inline bool operator==(const uint256& a, uint64 b) { return (base_uint256)a == b; } +inline bool operator!=(const uint256& a, uint64 b) { return (base_uint256)a != b; } +inline const uint256 operator<<(const base_uint256& a, unsigned int shift) { return uint256(a) <<= shift; } +inline const uint256 operator>>(const base_uint256& a, unsigned int shift) { return uint256(a) >>= shift; } +inline const uint256 operator<<(const uint256& a, unsigned int shift) { return uint256(a) <<= shift; } +inline const uint256 operator>>(const uint256& a, unsigned int shift) { return uint256(a) >>= shift; } + +inline const uint256 operator^(const base_uint256& a, const base_uint256& b) { return uint256(a) ^= b; } +inline const uint256 operator&(const base_uint256& a, const base_uint256& b) { return uint256(a) &= b; } +inline const uint256 operator|(const base_uint256& a, const base_uint256& b) { return uint256(a) |= b; } +inline const uint256 operator+(const base_uint256& a, const base_uint256& b) { return uint256(a) += b; } +inline const uint256 operator-(const base_uint256& a, const base_uint256& b) { return uint256(a) -= b; } + +inline bool operator<(const base_uint256& a, const uint256& b) { return (base_uint256)a < (base_uint256)b; } +inline bool operator<=(const base_uint256& a, const uint256& b) { return (base_uint256)a <= (base_uint256)b; } +inline bool operator>(const base_uint256& a, const uint256& b) { return (base_uint256)a > (base_uint256)b; } +inline bool operator>=(const base_uint256& a, const uint256& b) { return (base_uint256)a >= (base_uint256)b; } +inline bool operator==(const base_uint256& a, const uint256& b) { return (base_uint256)a == (base_uint256)b; } +inline bool operator!=(const base_uint256& a, const uint256& b) { return (base_uint256)a != (base_uint256)b; } +inline const uint256 operator^(const base_uint256& a, const uint256& b) { return (base_uint256)a ^ (base_uint256)b; } +inline const uint256 operator&(const base_uint256& a, const uint256& b) { return (base_uint256)a & (base_uint256)b; } +inline const uint256 operator|(const base_uint256& a, const uint256& b) { return (base_uint256)a | (base_uint256)b; } +inline const uint256 operator+(const base_uint256& a, const uint256& b) { return (base_uint256)a + (base_uint256)b; } +inline const uint256 operator-(const base_uint256& a, const uint256& b) { return (base_uint256)a - (base_uint256)b; } + +inline bool operator<(const uint256& a, const base_uint256& b) { return (base_uint256)a < (base_uint256)b; } +inline bool operator<=(const uint256& a, const base_uint256& b) { return (base_uint256)a <= (base_uint256)b; } +inline bool operator>(const uint256& a, const base_uint256& b) { return (base_uint256)a > (base_uint256)b; } +inline bool operator>=(const uint256& a, const base_uint256& b) { return (base_uint256)a >= (base_uint256)b; } +inline bool operator==(const uint256& a, const base_uint256& b) { return (base_uint256)a == (base_uint256)b; } +inline bool operator!=(const uint256& a, const base_uint256& b) { return (base_uint256)a != (base_uint256)b; } +inline const uint256 operator^(const uint256& a, const base_uint256& b) { return (base_uint256)a ^ (base_uint256)b; } +inline const uint256 operator&(const uint256& a, const base_uint256& b) { return (base_uint256)a & (base_uint256)b; } +inline const uint256 operator|(const uint256& a, const base_uint256& b) { return (base_uint256)a | (base_uint256)b; } +inline const uint256 operator+(const uint256& a, const base_uint256& b) { return (base_uint256)a + (base_uint256)b; } +inline const uint256 operator-(const uint256& a, const base_uint256& b) { return (base_uint256)a - (base_uint256)b; } + +inline bool operator<(const uint256& a, const uint256& b) { return (base_uint256)a < (base_uint256)b; } +inline bool operator<=(const uint256& a, const uint256& b) { return (base_uint256)a <= (base_uint256)b; } +inline bool operator>(const uint256& a, const uint256& b) { return (base_uint256)a > (base_uint256)b; } +inline bool operator>=(const uint256& a, const uint256& b) { return (base_uint256)a >= (base_uint256)b; } +inline bool operator==(const uint256& a, const uint256& b) { return (base_uint256)a == (base_uint256)b; } +inline bool operator!=(const uint256& a, const uint256& b) { return (base_uint256)a != (base_uint256)b; } +inline const uint256 operator^(const uint256& a, const uint256& b) { return (base_uint256)a ^ (base_uint256)b; } +inline const uint256 operator&(const uint256& a, const uint256& b) { return (base_uint256)a & (base_uint256)b; } +inline const uint256 operator|(const uint256& a, const uint256& b) { return (base_uint256)a | (base_uint256)b; } +inline const uint256 operator+(const uint256& a, const uint256& b) { return (base_uint256)a + (base_uint256)b; } +inline const uint256 operator-(const uint256& a, const uint256& b) { return (base_uint256)a - (base_uint256)b; } + + + + + + + + + + + + +inline int Testuint256AdHoc(vector vArg) +{ + uint256 g(0); + + + printf("%s\n", g.ToString().c_str()); + g--; printf("g--\n"); + printf("%s\n", g.ToString().c_str()); + g--; printf("g--\n"); + printf("%s\n", g.ToString().c_str()); + g++; printf("g++\n"); + printf("%s\n", g.ToString().c_str()); + g++; printf("g++\n"); + printf("%s\n", g.ToString().c_str()); + g++; printf("g++\n"); + printf("%s\n", g.ToString().c_str()); + g++; printf("g++\n"); + printf("%s\n", g.ToString().c_str()); + + + + uint256 a(7); + printf("a=7\n"); + printf("%s\n", a.ToString().c_str()); + + uint256 b; + printf("b undefined\n"); + printf("%s\n", b.ToString().c_str()); + int c = 3; + + a = c; + a.pn[3] = 15; + printf("%s\n", a.ToString().c_str()); + uint256 k(c); + + a = 5; + a.pn[3] = 15; + printf("%s\n", a.ToString().c_str()); + b = 1; + b <<= 52; + + a |= b; + + a ^= 0x500; + + printf("a %s\n", a.ToString().c_str()); + + a = a | b | (uint256)0x1000; + + + printf("a %s\n", a.ToString().c_str()); + printf("b %s\n", b.ToString().c_str()); + + a = 0xfffffffe; + a.pn[4] = 9; + + printf("%s\n", a.ToString().c_str()); + a++; + printf("%s\n", a.ToString().c_str()); + a++; + printf("%s\n", a.ToString().c_str()); + a++; + printf("%s\n", a.ToString().c_str()); + a++; + printf("%s\n", a.ToString().c_str()); + + a--; + printf("%s\n", a.ToString().c_str()); + a--; + printf("%s\n", a.ToString().c_str()); + a--; + printf("%s\n", a.ToString().c_str()); + uint256 d = a--; + printf("%s\n", d.ToString().c_str()); + printf("%s\n", a.ToString().c_str()); + a--; + printf("%s\n", a.ToString().c_str()); + a--; + printf("%s\n", a.ToString().c_str()); + + d = a; + + printf("%s\n", d.ToString().c_str()); + for (int i = uint256::WIDTH-1; i >= 0; i--) printf("%08x", d.pn[i]); printf("\n"); + + uint256 neg = d; + neg = ~neg; + printf("%s\n", neg.ToString().c_str()); + + + uint256 e = uint256("0xABCDEF123abcdef12345678909832180000011111111"); + printf("\n"); + printf("%s\n", e.ToString().c_str()); + + + printf("\n"); + uint256 x1 = uint256("0xABCDEF123abcdef12345678909832180000011111111"); + uint256 x2; + printf("%s\n", x1.ToString().c_str()); + for (int i = 0; i < 270; i += 4) + { + x2 = x1 << i; + printf("%s\n", x2.ToString().c_str()); + } + + printf("\n"); + printf("%s\n", x1.ToString().c_str()); + for (int i = 0; i < 270; i += 4) + { + x2 = x1; + x2 >>= i; + printf("%s\n", x2.ToString().c_str()); + } + + + for (int i = 0; i < 100; i++) + { + uint256 k = (~uint256(0) >> i); + printf("%s\n", k.ToString().c_str()); + } + + for (int i = 0; i < 100; i++) + { + uint256 k = (~uint256(0) << i); + printf("%s\n", k.ToString().c_str()); + } + + return (0); +} diff --git a/uiproject.fbp b/uiproject.fbp new file mode 100644 index 00000000..49b1fefa --- /dev/null +++ b/uiproject.fbp @@ -0,0 +1,11860 @@ + + + + + + C++ + 1 + UTF-8 + connect + uibase + 1000 + none + 0 + + + . + + 1 + 0 + 0 + + wxSYS_COLOUR_BTNFACE + + + 1 + + + + 0 + wxID_MAINFRAME + + + CMainFrameBase + + 705,484 + wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER + + Bitcoin + + + + wxTAB_TRAVERSAL + 1 + + + + OnClose + + + + + OnIdle + + + + + + + + + + + + OnMouseEvents + + OnPaint + + + + + + + + 240,240,240 + + 1 + + + 0 + wxID_ANY + MyMenuBar + + + m_menubar + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + &File + m_menuFile + protected + + + 0 + 1 + + wxID_ANY + wxITEM_NORMAL + E&xit + m_menuFileExit + none + + + OnMenuFileExit + + + + + &Options + m_menuOptions + public + + + 0 + 1 + + wxID_OPTIONSGENERATEBITCOINS + wxITEM_CHECK + &Generate Coins + m_menuOptionsGenerateBitcoins + none + + + OnMenuOptionsGenerate + + + + + 0 + 1 + + wxID_ANY + wxITEM_NORMAL + &Change Your Address... + m_menuChangeYourAddress + none + + + OnMenuOptionsChangeYourAddress + + + + + 0 + 1 + + wxID_ANY + wxITEM_NORMAL + &Options... + m_menuOptionsOptions + none + + + OnMenuOptionsOptions + + + + + &Help + m_menuHelp + protected + + + 0 + 1 + + wxID_ANY + wxITEM_NORMAL + &About... + m_menuHelpAbout + none + + + OnMenuHelpAbout + + + + + + + 20,20 + + 1 + + ,90,90,-1,70,0 + 0 + wxID_ANY + + + + m_toolBar + 1 + protected + + 1 + -1,-1 + wxTB_FLAT|wxTB_HORZ_TEXT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + send20; Load From Resource + wxID_BUTTONSEND + wxITEM_NORMAL + &Send Coins + m_tool1 + + + + OnButtonSend + + + + + + addressbook20; Load From Resource + wxID_BUTTONRECEIVE + wxITEM_NORMAL + &Address Book + m_tool2 + + + + OnButtonAddressBook + + + + + + + 240,240,240 + + 1 + + 1 + + 0 + wxID_ANY + + + m_statusBar + protected + + + wxST_SIZEGRIP + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer2 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 2 + protected + 0 + + + + 5 + wxEXPAND|wxRIGHT|wxLEFT + 0 + + + bSizer85 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Your Bitcoin Address: + + + m_staticText32 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + wxSYS_COLOUR_MENU + + 1 + + + 0 + wxID_TEXTCTRLADDRESS + -1,-1 + 0 + -1,-1 + m_textCtrlAddress + protected + + 250,-1 + wxTE_READONLY + + + + + + + + + + OnKeyDown + + + + + + + + + + + OnMouseEventsAddress + + + + + + OnSetFocusAddress + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONCOPY + &Copy to Clipboard + + + m_buttonCopy + protected + + + wxBU_EXACTFIT + + + + + + OnButtonCopy + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxRIGHT + 0 + + + + 0 + 1 + + + 1 + wxID_BUTTONCHANGE + C&hange... + + + m_button91 + protected + + + + + + + + + OnButtonChange + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + 0 + protected + 0 + + + + + + 5 + wxEXPAND + 0 + + + bSizer3 + wxHORIZONTAL + none + + 5 + wxEXPAND|wxALIGN_BOTTOM|wxALL + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_panel14 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer66 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Balance: + + + m_staticText41 + protected + + -1,15 + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + wxSYS_COLOUR_WINDOW + + 1 + + ,90,90,8,70,0 + 0 + wxID_ANY + + + + m_staticTextBalance + protected + + 120,15 + wxALIGN_RIGHT|wxST_NO_AUTORESIZE + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + 0 + protected + 0 + + + + 5 + wxALIGN_BOTTOM|wxTOP|wxRIGHT|wxLEFT + 0 + + + " All" " Sent" " Received" " In Progress" + + 1 + + + 1 + wxID_ANY + + + m_choiceFilter + protected + + 0 + 110,-1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + + + + 1 + + + 0 + wxID_ANY + + + m_notebook + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + All Transactions + 0 + + + + 1 + + + 0 + wxID_ANY + + + m_panel7 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer157 + wxVERTICAL + none + + 5 + wxEXPAND|wxALL + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrl + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING + + + + + wxALWAYS_SHOW_SB + + + + + + + + + + + + + + + OnListColBeginDrag + + + + + + + + + OnListItemActivatedAllTransactions + + + + + + + + + + + + + OnPaintListCtrl + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + + bSizer_TabsForFutureUse + wxVERTICAL + none + + 5 + wxEXPAND | wxALL + 1 + + + + 1 + + + 1 + wxID_ANY + + + m_panel9 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer159 + wxVERTICAL + none + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlEscrows + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND | wxALL + 1 + + + + 1 + + + 1 + wxID_ANY + + + m_panel8 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer158 + wxVERTICAL + none + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlOrdersSent + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListItemActivatedOrdersSent + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND | wxALL + 1 + + + + 1 + + + 1 + wxID_ANY + + + m_panel10 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer160 + wxVERTICAL + none + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlProductsSent + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListItemActivatedProductsSent + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND | wxALL + 1 + + + + 1 + + + 1 + wxID_ANY + + + m_panel11 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer161 + wxVERTICAL + none + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlOrdersReceived + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListItemActivatedOrdersReceived + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CTxDetailsDialogBase + + 620,450 + wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER + + Transaction Details + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer64 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + bSizer66 + wxVERTICAL + none + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_htmlWin + protected + + + wxHW_SCROLLBAR_AUTO + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_RIGHT + 0 + + + bSizer65 + wxVERTICAL + none + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OK + OK + + + m_buttonOK + protected + + 85,25 + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + COptionsDialogBase + + 500,261 + wxDEFAULT_DIALOG_STYLE + + Options + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer55 + wxVERTICAL + none + + 5 + wxEXPAND|wxLEFT + 1 + + + bSizer57 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 20 + protected + 0 + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + Optional transaction fee you give to the nodes that process your transactions. + + + m_staticText32 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + bSizer56 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Transaction fee: + + + m_staticText31 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_TRANSACTIONFEE + + 0 + + m_textCtrlTransactionFee + protected + + 70,-1 + + + + + + + + + + + + + OnKillFocusTransactionFee + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_RIGHT + 0 + + + bSizer58 + wxHORIZONTAL + none + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OK + OK + + + m_buttonOK + protected + + 85,25 + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + -1,-1 + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CAboutDialogBase + + 507,298 + wxDEFAULT_DIALOG_STYLE + + About Bitcoin + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer60 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + bSizer62 + wxHORIZONTAL + none + + 5 + wxEXPAND + 0 + + 0 + protected + 60 + + + + 5 + wxEXPAND + 1 + + + bSizer63 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 50 + protected + 0 + + + + 5 + wxEXPAND + 0 + + + bSizer64 + wxHORIZONTAL + none + + 5 + wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + Tahoma,90,92,10,74,0 + 0 + wxID_ANY + Bitcoin + + + m_staticText40 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxRIGHT + 0 + + + + 1 + + Tahoma,90,90,10,74,0 + 0 + wxID_ANY + version + + + m_staticTextVersion + public + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + 4 + protected + 0 + + + + 5 + wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + Copyright © 2009 Satoshi Nakamoto. This is experimental software. Do not rely on it for actual financial transactions. Distributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php. This product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com). + + + m_staticTextMain + protected + + + + + + + + + 400 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + + + + + 5 + wxALIGN_RIGHT|wxEXPAND + 0 + + + bSizer61 + wxHORIZONTAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OK + OK + + + m_buttonOK + protected + + 85,25 + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CSendDialogBase + + 675,312 + wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER + + Send Coins + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer21 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 5 + protected + 0 + + + + 5 + wxEXPAND|wxLEFT + 0 + + 2 + wxBOTH + 1 + + 0 + + fgSizer1 + wxFLEX_GROWMODE_SPECIFIED + none + 3 + 0 + + 5 + wxEXPAND + 0 + + 0 + protected + 0 + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, or Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online. + + + m_staticText14 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND|wxLEFT + 1 + + 70,-1 + bSizer47 + wxHORIZONTAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + check; Load From Icon Resource [-1; -1] + + 1 + + + 0 + wxID_ANY + + + m_bitmapCheckMark + protected + + 16,16 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Pay &To: + + + m_staticText36 + protected + + -1,-1 + wxALIGN_RIGHT + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND|wxRIGHT + 1 + + + bSizer19 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 1 + + + + 1 + + + 0 + wxID_TEXTCTRLPAYTO + + 0 + + m_textCtrlAddress + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + OnTextAddress + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONPASTE + &Paste + + + m_buttonPaste + protected + + -1,-1 + wxBU_EXACTFIT + + + + + + OnButtonPaste + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONADDRESSBOOK + Address &Book... + + + m_buttonAddress + protected + + + + + + + + + OnButtonAddressBook + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT|wxALIGN_RIGHT + 0 + + + + 1 + + + 0 + wxID_ANY + &Amount: + + + m_staticText19 + protected + + -1,-1 + wxALIGN_RIGHT + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + + 1 + + ,90,90,-1,70,0 + 0 + wxID_TEXTCTRLAMOUNT + + 20 + + m_textCtrlAmount + protected + + 145,-1 + + + + + + + + + + + OnKeyDown + + OnKillFocusAmount + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 1 + wxID_ANY + T&ransfer: + + + m_staticText20 + protected + + -1,-1 + wxALIGN_RIGHT + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + " Standard" + + 1 + + + 1 + wxID_CHOICETRANSFERTYPE + + + m_choiceTransferType + protected + + 0 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + 3 + protected + 0 + + + + 5 + wxEXPAND + 0 + + 0 + protected + 0 + + + + + + 5 + wxEXPAND + 0 + + + bSizer672 + wxHORIZONTAL + none + + 5 + wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT + 1 + + + bSizer681 + wxVERTICAL + none + + 5 + wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + &From: + + + m_staticTextFrom + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxLEFT|wxEXPAND + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + + m_textCtrlFrom + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + + bSizer67 + wxHORIZONTAL + none + + 5 + wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT + 1 + + + bSizer68 + wxVERTICAL + none + + 5 + wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + &Message: + + + m_staticTextMessage + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND|wxLEFT + 1 + + + + 1 + + + 0 + wxID_ANY + + 0 + + m_textCtrlMessage + protected + + + wxTE_MULTILINE + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + bSizer23 + wxHORIZONTAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL + 0 + + + + 0 + 1 + + ,90,90,-1,70,0 + 0 + wxID_BUTTONSEND + &Send + + 85,25 + m_buttonSend + protected + + -1,-1 + + + + + + + OnButtonSend + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + -1,-1 + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CSendingDialogBase + + 442,151 + wxDEFAULT_DIALOG_STYLE + + Sending... + + + + + + + + OnClose + + + + + + + + + + + + + + + + + + + + OnPaint + + + + + + + + + bSizer68 + wxVERTICAL + none + + 8 + wxALIGN_CENTER_VERTICAL|wxEXPAND|wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + + + + m_staticTextSending + protected + + -1,14 + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 10 + wxEXPAND|wxRIGHT|wxLEFT + 1 + + wxSYS_COLOUR_BTNFACE + + 1 + + + 0 + wxID_ANY + + 0 + + m_textCtrlStatus + protected + + + wxTE_CENTRE|wxTE_MULTILINE|wxTE_NO_VSCROLL|wxTE_READONLY + + + Connecting... + + + wxNO_BORDER + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + bSizer69 + wxHORIZONTAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL + 0 + + + + 0 + 0 + + + 0 + wxID_ANY + OK + + 85,25 + m_buttonOK + protected + + + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + -1,-1 + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CYourAddressDialogBase + + 610,390 + wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER + + Your Bitcoin Addresses + + + + + + + + OnClose + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer68 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 5 + protected + 0 + + + + 5 + wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + These are your Bitcoin addresses for receiving payments. You may want to give a different one to each sender so you can keep track of who is paying you. The highlighted address is displayed in the main window. + + + m_staticText45 + protected + + + + + + + + + 590 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_LISTCTRL + + + m_listCtrl + protected + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListEndLabelEdit + + OnListItemActivated + + + + + OnListItemSelected + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + bSizer69 + wxHORIZONTAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONRENAME + &Edit... + + 85,25 + m_buttonRename + protected + + + + + + + + + OnButtonRename + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONNEW + &New Address... + + 110,25 + m_buttonNew + protected + + -1,-1 + + + + + + + OnButtonNew + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONCOPY + &Copy to Clipboard + + 120,25 + m_buttonCopy + protected + + -1,-1 + + + + + + + OnButtonCopy + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OK + OK + + 85,25 + m_buttonOK + protected + + + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 1 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + -1,-1 + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CAddressBookDialogBase + + 610,390 + wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER + + Address Book + + + + + + + + OnClose + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer68 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 5 + protected + 0 + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 1 + wxID_ANY + Bitcoin Address + + + m_staticText55 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_LISTCTRL + + + m_listCtrl + protected + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListEndLabelEdit + + OnListItemActivated + + + + + OnListItemSelected + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + bSizer69 + wxHORIZONTAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONEDIT + &Edit... + + 85,25 + m_buttonEdit + protected + + + + + + + + + OnButtonEdit + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONNEW + &New Address... + + 110,25 + m_buttonNew + protected + + + + + + + + + OnButtonNew + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONDELETE + &Delete + + 85,25 + m_buttonDelete + protected + + + + + + + + + OnButtonDelete + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OK + OK + + 85,25 + m_buttonOK + protected + + -1,-1 + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + public + + -1,-1 + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CProductsDialogBase + + 708,535 + wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER + + Marketplace + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer22 + wxVERTICAL + none + + 5 + wxEXPAND|wxTOP|wxBOTTOM|wxRIGHT + 0 + + + bSizer23 + wxHORIZONTAL + none + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + "(Any Category)" + + 1 + + + 0 + wxID_ANY + + + m_comboBoxCategory + protected + + 150,-1 + + + + (Any Category) + + + + + OnCombobox + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 1 + + + + 1 + + + 0 + wxID_ANY + + 0 + + m_textCtrlSearch + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 0 + 1 + + + 0 + wxID_ANY + &Search + + + m_buttonSearch + protected + + + + + + + + + OnButtonSearch + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrl + protected + + + wxLC_NO_SORT_HEADER|wxLC_REPORT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListItemActivated + + + + + + + + + + + + + + + + + + + + + + + + + wxSYS_COLOUR_MENU + + + 1 + + + + 0 + wxID_ANY + + + CEditProductDialogBase + + 660,640 + wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER + + Edit Product + + + + wxTAB_TRAVERSAL + 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer20 + wxVERTICAL + none + + 5 + wxEXPAND|wxALL + 1 + + wxSYS_COLOUR_WINDOW + + 1 + + + 0 + wxID_ANY + + + m_scrolledWindow + protected + + 5 + 5 + + + + + + wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer21 + wxVERTICAL + none + + 5 + wxEXPAND|wxTOP|wxRIGHT|wxLEFT + 0 + + 2 + wxBOTH + 1 + + 0 + + fgSizer8 + wxFLEX_GROWMODE_SPECIFIED + none + 0 + 0 + + 5 + wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Category + + + m_staticText106 + protected + + + wxALIGN_RIGHT + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + + 1 + + + 0 + wxID_ANY + + 180,-1 + m_comboBoxCategory + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Title + + + m_staticText108 + protected + + + wxALIGN_RIGHT + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + 0 + + m_textCtrlTitle + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Price + + + m_staticText107 + protected + + + wxALIGN_RIGHT + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 105,-1 + m_textCtrlPrice + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Page 1: Description + + + m_staticText22 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + -1,170 + m_textCtrlDescription + protected + + + wxTE_MULTILINE + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Page 2: Order Form + + + m_staticText23 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND|wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + -1,120 + m_textCtrlInstructions + protected + + + wxTE_MULTILINE + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + 3 + wxBOTH + 1 + + 0 + + fgSizer5 + wxFLEX_GROWMODE_SPECIFIED + public + 0 + 0 + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Label + + + m_staticText24 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Comma separated list of choices, or leave blank for text field + + + m_staticText25 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel0 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField0 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxRIGHT|wxLEFT|wxALIGN_CENTER_VERTICAL + 0 + + + + 0 + 1 + + + 0 + wxID_DEL0 + Delete + + -1,-1 + m_buttonDel0 + protected + + 60,20 + + + + + + + OnButtonDel0 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel1 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField1 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL1 + Delete + + -1,-1 + m_buttonDel1 + protected + + 60,20 + + + + + + + OnButtonDel1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel2 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField2 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL2 + Delete + + -1,-1 + m_buttonDel2 + protected + + 60,20 + + + + + + + OnButtonDel2 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel3 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField3 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL3 + Delete + + -1,-1 + m_buttonDel3 + protected + + 60,20 + + + + + + + OnButtonDel3 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel4 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField4 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL4 + Delete + + -1,-1 + m_buttonDel4 + protected + + 60,20 + + + + + + + OnButtonDel4 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel5 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField5 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL5 + Delete + + -1,-1 + m_buttonDel5 + protected + + 60,20 + + + + + + + OnButtonDel5 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel6 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField6 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL6 + Delete + + -1,-1 + m_buttonDel6 + protected + + 60,20 + + + + + + + OnButtonDel6 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel7 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField7 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL7 + Delete + + -1,-1 + m_buttonDel7 + protected + + 60,20 + + + + + + + OnButtonDel7 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel8 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField8 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL8 + Delete + + -1,-1 + m_buttonDel8 + protected + + 60,20 + + + + + + + OnButtonDel8 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel9 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField9 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL9 + Delete + + -1,-1 + m_buttonDel9 + protected + + 60,20 + + + + + + + OnButtonDel9 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel10 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField10 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL10 + Delete + + -1,-1 + m_buttonDel10 + protected + + 60,20 + + + + + + + OnButtonDel10 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel11 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField11 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL11 + Delete + + -1,-1 + m_buttonDel11 + protected + + 60,20 + + + + + + + OnButtonDel11 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel12 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField12 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL12 + Delete + + -1,-1 + m_buttonDel12 + protected + + 60,20 + + + + + + + OnButtonDel12 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel13 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField13 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL13 + Delete + + -1,-1 + m_buttonDel13 + protected + + 60,20 + + + + + + + OnButtonDel13 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel14 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField14 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL14 + Delete + + -1,-1 + m_buttonDel14 + protected + + 60,20 + + + + + + + OnButtonDel14 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel15 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField15 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL15 + Delete + + -1,-1 + m_buttonDel15 + protected + + 60,20 + + + + + + + OnButtonDel15 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel16 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField16 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL16 + Delete + + -1,-1 + m_buttonDel16 + protected + + 60,20 + + + + + + + OnButtonDel16 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel17 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField17 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL17 + Delete + + -1,-1 + m_buttonDel17 + protected + + 60,20 + + + + + + + OnButtonDel17 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel18 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField18 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL18 + Delete + + -1,-1 + m_buttonDel18 + protected + + 60,20 + + + + + + + OnButtonDel18 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + + 0 + 150,-1 + m_textCtrlLabel19 + protected + + + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + -1,-1 + 0 + -1,-1 + m_textCtrlField19 + protected + + -1,-1 + + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT + 0 + + + + 0 + 1 + + + 0 + wxID_DEL19 + Delete + + -1,-1 + m_buttonDel19 + protected + + 60,20 + + + + + + + OnButtonDel19 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_HORIZONTAL + 0 + + + bSizer25 + wxHORIZONTAL + none + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_ANY + &Add Field + + + m_buttonAddField + protected + + + + + + + + + OnButtonAddField + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_RIGHT + 0 + + + bSizer26 + wxHORIZONTAL + none + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONSEND + &Send + + 85,25 + m_buttonOK + protected + + + + + + + + + OnButtonSend + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONPREVIEW + &Preview + + 85,25 + m_buttonPreview + protected + + + + + + + + + OnButtonPreview + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + wxSYS_COLOUR_MENU + + + 1 + + + + 0 + wxID_ANY + + + CViewProductDialogBase + + 630,520 + wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER + + Order Form + + + + wxTAB_TRAVERSAL + 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer20 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + bSizer116 + wxHORIZONTAL + none + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 1 + wxID_ANY + + + m_htmlWinReviews + protected + + + wxHW_SCROLLBAR_AUTO + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND|wxALL + 1 + + wxSYS_COLOUR_WINDOW + + 1 + + + 0 + wxID_ANY + + + m_scrolledWindow + protected + + 5 + 5 + + + + + + wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer21 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + + + 1 + + + 0 + wxID_ANY + + -1,-1 + m_richTextHeading + protected + + -1,50 + wxTE_READONLY + + + + + wxNO_BORDER + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND + 0 + + + + 1 + + + 0 + wxID_ANY + Order Form instructions here multiple lines 1 2 3 4 5 6 + + + m_staticTextInstructions + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_HORIZONTAL + 0 + + + bSizer25 + wxHORIZONTAL + none + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONSAMPLE + &Submit + + -1,-1 + m_buttonSubmitForm + protected + + + + + + + + + OnButtonSubmitForm + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL2 + Cancel + + + m_buttonCancelForm + protected + + + + + + + + + OnButtonCancelForm + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_RIGHT + 0 + + + bSizer26 + wxHORIZONTAL + none + + 5 + wxALL + 0 + + + + 0 + 0 + + + 0 + wxID_BUTTONBACK + < &Back + + 85,25 + m_buttonBack + protected + + + + + + + + + OnButtonBack + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONNEXT + &Next > + + 85,25 + m_buttonNext + protected + + + + + + + + + OnButtonNext + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + wxSYS_COLOUR_MENU + + + 1 + + + + 0 + wxID_ANY + + + CViewOrderDialogBase + + 630,520 + wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER + + View Order + + + + wxTAB_TRAVERSAL + 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer20 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + bSizer116 + wxHORIZONTAL + none + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_htmlWin + protected + + + wxHW_SCROLLBAR_AUTO + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_RIGHT + 0 + + + bSizer26 + wxHORIZONTAL + none + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OK + OK + + 85,25 + m_buttonOK + protected + + + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + wxSYS_COLOUR_MENU + + + 1 + + + + 0 + wxID_ANY + + + CEditReviewDialogBase + + 630,440 + wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER + + Enter Review + + + + wxTAB_TRAVERSAL + 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer112 + wxVERTICAL + none + + 5 + + 0 + + 3 + protected + 0 + + + + 5 + wxALL|wxEXPAND + 0 + + + + 1 + + + 0 + wxID_ANY + + + + m_staticTextSeller + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + + 0 + + 3 + protected + 0 + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Rating + + + m_staticText110 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + " 1 star" " 2 stars" " 3 stars" " 4 stars" " 5 stars" + + 1 + + + 0 + wxID_ANY + + + m_choiceStars + protected + + 0 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Review + + + m_staticText43 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + 0 + + m_textCtrlReview + protected + + + wxTE_MULTILINE + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_RIGHT + 0 + + + bSizer113 + wxHORIZONTAL + none + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_SUBMIT + &Submit + + 85,25 + m_buttonSubmit + protected + + + + + + + + + OnButtonSubmit + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + wxSYS_COLOUR_BTNFACE + + + 1 + + + + 0 + wxID_ANY + + + CPokerLobbyDialogBase + + 586,457 + wxDEFAULT_FRAME_STYLE + + Poker Lobby + + + + wxTAB_TRAVERSAL + 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer156 + wxHORIZONTAL + none + + 5 + wxEXPAND|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + + 130,-1 + m_treeCtrl + protected + + + wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + OnTreeSelChanged + + + + + + + + 5 + wxEXPAND + 1 + + + bSizer172 + wxVERTICAL + none + + 5 + wxEXPAND|wxALL + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrl + protected + + + wxLC_NO_SORT_HEADER|wxLC_REPORT + + + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListItemActivated + + + + + OnListItemSelected + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OPENNEWTABLE + &Open New Table + + + m_buttonNewTable + protected + + + + + + + + + OnButtonNewTable + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CPokerDialogBase + + 806,550 + wxDEFAULT_FRAME_STYLE|wxFRAME_NO_TASKBAR + + Poker + + + + wxFULL_REPAINT_ON_RESIZE|wxTAB_TRAVERSAL + 1 + + + + OnClose + + + + + + + + + + + + + + + + + OnMouseEvents + + OnPaint + + + + + OnSize + + + + bSizer174 + wxVERTICAL + none + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + Deal Me Out + + + m_checkSitOut + public + + + + + + + + + + OnCheckSitOut + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_DEALHAND + &Deal Hand + + + m_buttonDealHand + protected + + 150,25 + + + + + + + OnButtonDealHand + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_FOLD + &Fold + + + m_buttonFold + protected + + 80,25 + + + + + + + OnButtonFold + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CALL + &Call + + + m_buttonCall + protected + + 80,25 + + + + + + + OnButtonCall + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_RAISE + &Raise + + + m_buttonRaise + protected + + 80,25 + + + + + + + OnButtonRaise + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_LEAVETABLE + &Leave Table + + + m_buttonLeaveTable + protected + + 90,25 + + + + + + + OnButtonLeaveTable + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 1 + + + 0 + wxID_DITCHPLAYER + + 0 + + m_textDitchPlayer + protected + + 45,-1 + wxTE_PROCESS_ENTER + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + OnDitchPlayer + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + FOLD + + + m_checkPreFold + public + + 100,-1 + + + + + + + + OnCheckPreFold + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + CALL + + + m_checkPreCall + public + + 100,-1 + + + + + + + + OnCheckPreCall + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + CALL ANY + + + m_checkPreCallAny + public + + 100,-1 + + + + + + + + OnCheckPreCallAny + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + RAISE + + + m_checkPreRaise + public + + 100,-1 + + + + + + + + OnCheckPreRaise + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + RAISE ANY + + + m_checkPreRaiseAny + public + + 100,-1 + + + + + + + + OnCheckPreRaiseAny + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + 1 + + 0 + wxID_ANY + + + m_statusBar + public + + + wxST_SIZEGRIP + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CGetTextFromUserDialogBase + + 403,138 + wxDEFAULT_DIALOG_STYLE + + + + + + + + + + OnClose + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer79 + wxVERTICAL + none + + 10 + wxEXPAND|wxALL + 1 + + + bSizer81 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + + + + m_staticTextMessage1 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL + 0 + + + + 1 + + + 0 + wxID_TEXTCTRL + + 0 + + m_textCtrl1 + protected + + + wxTE_PROCESS_ENTER + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 1 + wxID_ANY + + + + m_staticTextMessage2 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL + 0 + + + + 1 + + + 1 + wxID_TEXTCTRL + + 0 + + m_textCtrl2 + protected + + + wxTE_PROCESS_ENTER + + + + + + + + + + OnKeyDown + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + + + 5 + wxEXPAND + 0 + + + bSizer80 + wxHORIZONTAL + none + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_OK + OK + + 85,25 + m_buttonOK + protected + + -1,-1 + + + + + + + OnButtonOK + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 0 + 1 + + + 0 + wxID_CANCEL + Cancel + + 85,25 + m_buttonCancel + protected + + + + + + + + + OnButtonCancel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/util.cpp b/util.cpp new file mode 100644 index 00000000..9c0ab142 --- /dev/null +++ b/util.cpp @@ -0,0 +1,383 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" + + + +bool fDebug = false; + + + + +// Init openssl library multithreading support +static HANDLE* lock_cs; + +void win32_locking_callback(int mode, int type, const char* file, int line) +{ + if (mode & CRYPTO_LOCK) + WaitForSingleObject(lock_cs[type], INFINITE); + else + ReleaseMutex(lock_cs[type]); +} + +// Init +class CInit +{ +public: + CInit() + { + // Init openssl library multithreading support + lock_cs = (HANDLE*)OPENSSL_malloc(CRYPTO_num_locks() * sizeof(HANDLE)); + for (int i = 0; i < CRYPTO_num_locks(); i++) + lock_cs[i] = CreateMutex(NULL,FALSE,NULL); + CRYPTO_set_locking_callback(win32_locking_callback); + + // Seed random number generator with screen scrape and other hardware sources + RAND_screen(); + + // Seed random number generator with perfmon data + RandAddSeed(true); + } + ~CInit() + { + // Shutdown openssl library multithreading support + CRYPTO_set_locking_callback(NULL); + for (int i =0 ; i < CRYPTO_num_locks(); i++) + CloseHandle(lock_cs[i]); + OPENSSL_free(lock_cs); + } +} +instance_of_cinit; + + + + +void RandAddSeed(bool fPerfmon) +{ + // Seed with CPU performance counter + LARGE_INTEGER PerformanceCount; + QueryPerformanceCounter(&PerformanceCount); + RAND_add(&PerformanceCount, sizeof(PerformanceCount), 1.5); + memset(&PerformanceCount, 0, sizeof(PerformanceCount)); + + static int64 nLastPerfmon; + if (fPerfmon || GetTime() > nLastPerfmon + 5 * 60) + { + nLastPerfmon = GetTime(); + + // Seed with the entire set of perfmon data + unsigned char pdata[250000]; + memset(pdata, 0, sizeof(pdata)); + unsigned long nSize = sizeof(pdata); + long ret = RegQueryValueEx(HKEY_PERFORMANCE_DATA, "Global", NULL, NULL, pdata, &nSize); + RegCloseKey(HKEY_PERFORMANCE_DATA); + if (ret == ERROR_SUCCESS) + { + uint256 hash; + SHA256(pdata, nSize, (unsigned char*)&hash); + RAND_add(&hash, sizeof(hash), min(nSize/500.0, (double)sizeof(hash))); + hash = 0; + memset(pdata, 0, nSize); + + time_t nTime; + time(&nTime); + struct tm* ptmTime = gmtime(&nTime); + char pszTime[200]; + strftime(pszTime, sizeof(pszTime), "%x %H:%M:%S", ptmTime); + printf("%s RandAddSeed() got %d bytes of performance data\n", pszTime, nSize); + } + } +} + + + + + + + + + + +// Safer snprintf +// - prints up to limit-1 characters +// - output string is always null terminated even if limit reached +// - return value is the number of characters actually printed +int my_snprintf(char* buffer, size_t limit, const char* format, ...) +{ + if (limit == 0) + return 0; + va_list arg_ptr; + va_start(arg_ptr, format); + int ret = _vsnprintf(buffer, limit, format, arg_ptr); + va_end(arg_ptr); + if (ret < 0 || ret >= limit) + { + ret = limit - 1; + buffer[limit-1] = 0; + } + return ret; +} + + +string strprintf(const char* format, ...) +{ + char buffer[50000]; + char* p = buffer; + int limit = sizeof(buffer); + int ret; + loop + { + va_list arg_ptr; + va_start(arg_ptr, format); + ret = _vsnprintf(p, limit, format, arg_ptr); + va_end(arg_ptr); + if (ret >= 0 && ret < limit) + break; + if (p != buffer) + delete p; + limit *= 2; + p = new char[limit]; + if (p == NULL) + throw std::bad_alloc(); + } +#ifdef _MSC_VER + // msvc optimisation + if (p == buffer) + return string(p, p+ret); +#endif + string str(p, p+ret); + if (p != buffer) + delete p; + return str; +} + + +bool error(const char* format, ...) +{ + char buffer[50000]; + int limit = sizeof(buffer); + va_list arg_ptr; + va_start(arg_ptr, format); + int ret = _vsnprintf(buffer, limit, format, arg_ptr); + va_end(arg_ptr); + if (ret < 0 || ret >= limit) + { + ret = limit - 1; + buffer[limit-1] = 0; + } + printf("ERROR: %s\n", buffer); + return false; +} + + +void PrintException(std::exception* pex, const char* pszThread) +{ + char pszModule[260]; + pszModule[0] = '\0'; + GetModuleFileName(NULL, pszModule, sizeof(pszModule)); + _strlwr(pszModule); + char pszMessage[1000]; + if (pex) + snprintf(pszMessage, sizeof(pszMessage), + "EXCEPTION: %s \n%s \n%s in %s \n", typeid(*pex).name(), pex->what(), pszModule, pszThread); + else + snprintf(pszMessage, sizeof(pszMessage), + "UNKNOWN EXCEPTION \n%s in %s \n", pszModule, pszThread); + printf("\n\n************************\n%s", pszMessage); + if (wxTheApp) + wxMessageBox(pszMessage, "Error", wxOK | wxICON_ERROR); + throw; + //DebugBreak(); +} + + +void ParseString(const string& str, char c, vector& v) +{ + unsigned int i1 = 0; + unsigned int i2; + do + { + i2 = str.find(c, i1); + v.push_back(str.substr(i1, i2-i1)); + i1 = i2+1; + } + while (i2 != str.npos); +} + + +string FormatMoney(int64 n, bool fPlus) +{ + n /= CENT; + string str = strprintf("%I64d.%02I64d", (n > 0 ? n : -n)/100, (n > 0 ? n : -n)%100); + for (int i = 6; i < str.size(); i += 4) + if (isdigit(str[str.size() - i - 1])) + str.insert(str.size() - i, 1, ','); + if (n < 0) + str.insert((unsigned int)0, 1, '-'); + else if (fPlus && n > 0) + str.insert((unsigned int)0, 1, '+'); + return str; +} + +bool ParseMoney(const char* pszIn, int64& nRet) +{ + string strWhole; + int64 nCents = 0; + const char* p = pszIn; + while (isspace(*p)) + p++; + for (; *p; p++) + { + if (*p == ',' && p > pszIn && isdigit(p[-1]) && isdigit(p[1]) && isdigit(p[2]) && isdigit(p[3]) && !isdigit(p[4])) + continue; + if (*p == '.') + { + p++; + if (isdigit(*p)) + { + nCents = 10 * (*p++ - '0'); + if (isdigit(*p)) + nCents += (*p++ - '0'); + } + break; + } + if (isspace(*p)) + break; + if (!isdigit(*p)) + return false; + strWhole.insert(strWhole.end(), *p); + } + for (; *p; p++) + if (!isspace(*p)) + return false; + if (strWhole.size() > 14) + return false; + if (nCents < 0 || nCents > 99) + return false; + int64 nWhole = atoi64(strWhole); + int64 nPreValue = nWhole * 100 + nCents; + int64 nValue = nPreValue * CENT; + if (nValue / CENT != nPreValue) + return false; + if (nValue / COIN != nWhole) + return false; + nRet = nValue; + return true; +} + + + + + + + + + + +bool FileExists(const char* psz) +{ +#ifdef WIN32 + return GetFileAttributes(psz) != -1; +#else + return access(psz, 0) != -1; +#endif +} + +int GetFilesize(FILE* file) +{ + int nSavePos = ftell(file); + int nFilesize = -1; + if (fseek(file, 0, SEEK_END) == 0) + nFilesize = ftell(file); + fseek(file, nSavePos, SEEK_SET); + return nFilesize; +} + + + + + + + + +uint64 GetRand(uint64 nMax) +{ + if (nMax == 0) + return 0; + + // The range of the random source must be a multiple of the modulus + // to give every possible output value an equal possibility + uint64 nRange = (_UI64_MAX / nMax) * nMax; + uint64 nRand = 0; + do + RAND_bytes((unsigned char*)&nRand, sizeof(nRand)); + while (nRand >= nRange); + return (nRand % nMax); +} + + + + + + + + + + +// +// "Never go to sea with two chronometers; take one or three." +// Our three chronometers are: +// - System clock +// - Median of other server's clocks +// - NTP servers +// +// note: NTP isn't implemented yet, so until then we just use the median +// of other nodes clocks to correct ours. +// + +int64 GetTime() +{ + return time(NULL); +} + +static int64 nTimeOffset = 0; + +int64 GetAdjustedTime() +{ + return GetTime() + nTimeOffset; +} + +void AddTimeData(unsigned int ip, int64 nTime) +{ + int64 nOffsetSample = nTime - GetTime(); + + // Ignore duplicates + static set setKnown; + if (!setKnown.insert(ip).second) + return; + + // Add data + static vector vTimeOffsets; + if (vTimeOffsets.empty()) + vTimeOffsets.push_back(0); + vTimeOffsets.push_back(nOffsetSample); + printf("Added time data, samples %d, ip %08x, offset %+I64d (%+I64d minutes)\n", vTimeOffsets.size(), ip, vTimeOffsets.back(), vTimeOffsets.back()/60); + if (vTimeOffsets.size() >= 5 && vTimeOffsets.size() % 2 == 1) + { + sort(vTimeOffsets.begin(), vTimeOffsets.end()); + int64 nMedian = vTimeOffsets[vTimeOffsets.size()/2]; + nTimeOffset = nMedian; + if ((nMedian > 0 ? nMedian : -nMedian) > 5 * 60) + { + // Only let other nodes change our clock so far before we + // go to the NTP servers + /// todo: Get time from NTP servers, then set a flag + /// to make sure it doesn't get changed again + } + foreach(int64 n, vTimeOffsets) + printf("%+I64d ", n); + printf("| nTimeOffset = %+I64d (%+I64d minutes)\n", nTimeOffset, nTimeOffset/60); + } +} diff --git a/util.h b/util.h new file mode 100644 index 00000000..97617ab4 --- /dev/null +++ b/util.h @@ -0,0 +1,399 @@ +// Copyright (c) 2009 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + + +#if defined(_MSC_VER) || defined(__BORLANDC__) +typedef __int64 int64; +typedef unsigned __int64 uint64; +#else +typedef long long int64; +typedef unsigned long long uint64; +#endif +#if defined(_MSC_VER) && _MSC_VER < 1300 +#define for if (false) ; else for +#endif + +#ifndef _MSC_VER +#define __forceinline inline +#endif + +#define foreach BOOST_FOREACH +#define loop for (;;) +#define BEGIN(a) ((char*)&(a)) +#define END(a) ((char*)&((&(a))[1])) +#define UBEGIN(a) ((unsigned char*)&(a)) +#define UEND(a) ((unsigned char*)&((&(a))[1])) +#define ARRAYLEN(array) (sizeof(array)/sizeof((array)[0])) + +#ifdef _WINDOWS +#define printf OutputDebugStringF +#endif + +#ifdef snprintf +#undef snprintf +#endif +#define snprintf my_snprintf + +#ifndef PRId64 +#if defined(_MSC_VER) || defined(__BORLANDC__) || defined(__MSVCRT__) +#define PRId64 "I64d" +#define PRIu64 "I64u" +#define PRIx64 "I64x" +#else +#define PRId64 "lld" +#define PRIu64 "llu" +#define PRIx64 "llx" +#endif +#endif + +// This is needed because the foreach macro can't get over the comma in pair +#define PAIRTYPE(t1, t2) pair + +// Used to bypass the rule against non-const reference to temporary +// where it makes sense with wrappers such as CFlatData or CTxDB +template +inline T& REF(const T& val) +{ + return (T&)val; +} + + + + + + + + + +extern bool fDebug; + +void RandAddSeed(bool fPerfmon=false); +int my_snprintf(char* buffer, size_t limit, const char* format, ...); +string strprintf(const char* format, ...); +bool error(const char* format, ...); +void PrintException(std::exception* pex, const char* pszThread); +void ParseString(const string& str, char c, vector& v); +string FormatMoney(int64 n, bool fPlus=false); +bool ParseMoney(const char* pszIn, int64& nRet); +bool FileExists(const char* psz); +int GetFilesize(FILE* file); +uint64 GetRand(uint64 nMax); +int64 GetTime(); +int64 GetAdjustedTime(); +void AddTimeData(unsigned int ip, int64 nTime); + + + + + + + + + + + + +// Wrapper to automatically initialize critical section +// Could use wxCriticalSection for portability, but it doesn't support TryEnterCriticalSection +class CCriticalSection +{ +protected: + CRITICAL_SECTION cs; +public: + char* pszFile; + int nLine; + explicit CCriticalSection() { InitializeCriticalSection(&cs); } + ~CCriticalSection() { DeleteCriticalSection(&cs); } + void Enter() { EnterCriticalSection(&cs); } + void Leave() { LeaveCriticalSection(&cs); } + bool TryEnter() { return TryEnterCriticalSection(&cs); } + CRITICAL_SECTION* operator&() { return &cs; } +}; + +// Automatically leave critical section when leaving block, needed for exception safety +class CCriticalBlock +{ +protected: + CRITICAL_SECTION* pcs; +public: + CCriticalBlock(CRITICAL_SECTION& csIn) { pcs = &csIn; EnterCriticalSection(pcs); } + CCriticalBlock(CCriticalSection& csIn) { pcs = &csIn; EnterCriticalSection(pcs); } + ~CCriticalBlock() { LeaveCriticalSection(pcs); } +}; + +// WARNING: This will catch continue and break! +// break is caught with an assertion, but there's no way to detect continue. +// I'd rather be careful than suffer the other more error prone syntax. +// The compiler will optimise away all this loop junk. +#define CRITICAL_BLOCK(cs) \ + for (bool fcriticalblockonce=true; fcriticalblockonce; assert(("break caught by CRITICAL_BLOCK!", !fcriticalblockonce)), fcriticalblockonce=false) \ + for (CCriticalBlock criticalblock(cs); fcriticalblockonce && (cs.pszFile=__FILE__, cs.nLine=__LINE__, true); fcriticalblockonce=false, cs.pszFile=NULL, cs.nLine=0) + +class CTryCriticalBlock +{ +protected: + CRITICAL_SECTION* pcs; +public: + CTryCriticalBlock(CRITICAL_SECTION& csIn) { pcs = (TryEnterCriticalSection(&csIn) ? &csIn : NULL); } + CTryCriticalBlock(CCriticalSection& csIn) { pcs = (TryEnterCriticalSection(&csIn) ? &csIn : NULL); } + ~CTryCriticalBlock() { if (pcs) LeaveCriticalSection(pcs); } + bool Entered() { return pcs != NULL; } +}; + +#define TRY_CRITICAL_BLOCK(cs) \ + for (bool fcriticalblockonce=true; fcriticalblockonce; assert(("break caught by TRY_CRITICAL_BLOCK!", !fcriticalblockonce)), fcriticalblockonce=false) \ + for (CTryCriticalBlock criticalblock(cs); fcriticalblockonce && (fcriticalblockonce = criticalblock.Entered()) && (cs.pszFile=__FILE__, cs.nLine=__LINE__, true); fcriticalblockonce=false, cs.pszFile=NULL, cs.nLine=0) + + + + + + + + + + + + +inline string i64tostr(int64 n) +{ + return strprintf("%"PRId64, n); +} + +inline string itostr(int n) +{ + return strprintf("%d", n); +} + +inline int64 atoi64(const char* psz) +{ +#ifdef _MSC_VER + return _atoi64(psz); +#else + return strtoll(psz, NULL, 10); +#endif +} + +inline int64 atoi64(const string& str) +{ +#ifdef _MSC_VER + return _atoi64(str.c_str()); +#else + return strtoll(str.c_str(), NULL, 10); +#endif +} + +inline int atoi(const string& str) +{ + return atoi(str.c_str()); +} + +inline int roundint(double d) +{ + return (int)(d > 0 ? d + 0.5 : d - 0.5); +} + +template +string HexStr(const T itbegin, const T itend, bool fSpaces=true) +{ + const unsigned char* pbegin = (const unsigned char*)&itbegin[0]; + const unsigned char* pend = pbegin + (itend - itbegin) * sizeof(itbegin[0]); + string str; + for (const unsigned char* p = pbegin; p != pend; p++) + str += strprintf((fSpaces && p != pend-1 ? "%02x " : "%02x"), *p); + return str; +} + +template +string HexNumStr(const T itbegin, const T itend, bool f0x=true) +{ + const unsigned char* pbegin = (const unsigned char*)&itbegin[0]; + const unsigned char* pend = pbegin + (itend - itbegin) * sizeof(itbegin[0]); + string str = (f0x ? "0x" : ""); + for (const unsigned char* p = pend-1; p >= pbegin; p--) + str += strprintf("%02X", *p); + return str; +} + +template +void PrintHex(const T pbegin, const T pend, const char* pszFormat="%s", bool fSpaces=true) +{ + printf(pszFormat, HexStr(pbegin, pend, fSpaces).c_str()); +} + + + + + + + + +inline int OutputDebugStringF(const char* pszFormat, ...) +{ +#ifdef __WXDEBUG__ + // log file + FILE* fileout = fopen("debug.log", "a"); + if (fileout) + { + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + vfprintf(fileout, pszFormat, arg_ptr); + va_end(arg_ptr); + fclose(fileout); + } + + // accumulate a line at a time + static CCriticalSection cs_OutputDebugStringF; + CRITICAL_BLOCK(cs_OutputDebugStringF) + { + static char pszBuffer[50000]; + static char* pend; + if (pend == NULL) + pend = pszBuffer; + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + int limit = END(pszBuffer) - pend - 2; + int ret = _vsnprintf(pend, limit, pszFormat, arg_ptr); + va_end(arg_ptr); + if (ret < 0 || ret >= limit) + { + pend = END(pszBuffer) - 2; + *pend++ = '\n'; + } + else + pend += ret; + *pend = '\0'; + char* p1 = pszBuffer; + char* p2; + while (p2 = strchr(p1, '\n')) + { + p2++; + char c = *p2; + *p2 = '\0'; + OutputDebugString(p1); + *p2 = c; + p1 = p2; + } + if (p1 != pszBuffer) + memmove(pszBuffer, p1, pend - p1 + 1); + pend -= (p1 - pszBuffer); + return ret; + } +#endif + + if (!wxTheApp) + { + // print to console + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + vprintf(pszFormat, arg_ptr); + va_end(arg_ptr); + } + return 0; +} + + + + + + + + + +inline void heapchk() +{ + if (_heapchk() != _HEAPOK) + DebugBreak(); +} + +// Randomize the stack to help protect against buffer overrun exploits +#define IMPLEMENT_RANDOMIZE_STACK(ThreadFn) \ + { \ + static char nLoops; \ + if (nLoops <= 0) \ + nLoops = GetRand(50) + 1; \ + if (nLoops-- > 1) \ + { \ + ThreadFn; \ + return; \ + } \ + } + +#define CATCH_PRINT_EXCEPTION(pszFn) \ + catch (std::exception& e) { \ + PrintException(&e, (pszFn)); \ + } catch (...) { \ + PrintException(NULL, (pszFn)); \ + } + + + + + + + + + +template +inline uint256 Hash(const T1 pbegin, const T1 pend) +{ + uint256 hash1; + SHA256((unsigned char*)&pbegin[0], (pend - pbegin) * sizeof(pbegin[0]), (unsigned char*)&hash1); + uint256 hash2; + SHA256((unsigned char*)&hash1, sizeof(hash1), (unsigned char*)&hash2); + return hash2; +} + +template +inline uint256 Hash(const T1 p1begin, const T1 p1end, + const T2 p2begin, const T2 p2end) +{ + uint256 hash1; + SHA256_CTX ctx; + SHA256_Init(&ctx); + SHA256_Update(&ctx, (unsigned char*)&p1begin[0], (p1end - p1begin) * sizeof(p1begin[0])); + SHA256_Update(&ctx, (unsigned char*)&p2begin[0], (p2end - p2begin) * sizeof(p2begin[0])); + SHA256_Final((unsigned char*)&hash1, &ctx); + uint256 hash2; + SHA256((unsigned char*)&hash1, sizeof(hash1), (unsigned char*)&hash2); + return hash2; +} + +template +inline uint256 Hash(const T1 p1begin, const T1 p1end, + const T2 p2begin, const T2 p2end, + const T3 p3begin, const T3 p3end) +{ + uint256 hash1; + SHA256_CTX ctx; + SHA256_Init(&ctx); + SHA256_Update(&ctx, (unsigned char*)&p1begin[0], (p1end - p1begin) * sizeof(p1begin[0])); + SHA256_Update(&ctx, (unsigned char*)&p2begin[0], (p2end - p2begin) * sizeof(p2begin[0])); + SHA256_Update(&ctx, (unsigned char*)&p3begin[0], (p3end - p3begin) * sizeof(p3begin[0])); + SHA256_Final((unsigned char*)&hash1, &ctx); + uint256 hash2; + SHA256((unsigned char*)&hash1, sizeof(hash1), (unsigned char*)&hash2); + return hash2; +} + +template +uint256 SerializeHash(const T& obj, int nType=SER_GETHASH, int nVersion=VERSION) +{ + // Most of the time is spent allocating and deallocating CDataStream's + // buffer. If this ever needs to be optimized further, make a CStaticStream + // class with its buffer on the stack. + CDataStream ss(nType, nVersion); + ss.reserve(10000); + ss << obj; + return Hash(ss.begin(), ss.end()); +} + +inline uint160 Hash160(const vector& vch) +{ + uint256 hash1; + SHA256(&vch[0], vch.size(), (unsigned char*)&hash1); + uint160 hash2; + RIPEMD160((unsigned char*)&hash1, sizeof(hash1), (unsigned char*)&hash2); + return hash2; +} From 465e1d99f25cc7f45efcacfcbb1f95c1c1f9a4ee Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sun, 30 Aug 2009 03:50:53 +0000 Subject: [PATCH 002/133] Added changelog.txt git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@2 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- changelog.txt | 2 ++ 1 file changed, 2 insertions(+) create mode 100644 changelog.txt diff --git a/changelog.txt b/changelog.txt new file mode 100644 index 00000000..19120c3e --- /dev/null +++ b/changelog.txt @@ -0,0 +1,2 @@ +Changes after 0.1.5: +-------------------- From 8dca7864f793072701f810e4c5ea12a6e1087085 Mon Sep 17 00:00:00 2001 From: sirius-m Date: Wed, 16 Sep 2009 13:26:04 +0000 Subject: [PATCH 003/133] No dll's here git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@8 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- libeay32.dll | Bin 1306630 -> 0 bytes mingwm10.dll | Bin 11673 -> 0 bytes 2 files changed, 0 insertions(+), 0 deletions(-) delete mode 100644 libeay32.dll delete mode 100644 mingwm10.dll diff --git a/libeay32.dll b/libeay32.dll deleted file mode 100644 index 3bc745c4c0f4751cab5195d9fbdf19ea42113016..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1306630 zcmeFadw5jU)jvLy49O%6oKZ&%8ey!XCK55xv?dT~ATb~y%!N#lOYnkp91#)DC~C;X znbDjaC#2q~wzc)rTC25|S`dqY01>r{cti1uRnK&!QfVOsWqzNv_BoTu1m5@kJ@51T zJ>NgRJP(<3_GRt0*IIk+wbx#IpUdWTnX*kLlLh~iNt0T3e>-DN;M$eW2}Uhnur1Oy&6|(}|MFRAn+QM!%kAM8rHNrG^Dr$Z@(llqJNI&M^ z{Uc3U*D-VcSBc~0YgWmCFzrJ0hY+H^raB=&Ux99qUwox}C0@Hez#r=Shxn}^8Vvla zHkrD|uMh>-p8}$k4mX(!@aw>D+CP-59)H!!m7D~cn%pDOat9t;3EnjV9@7Z4GED=m zN(Ps&CV1D1a!faJ5O$5;Qg4jrJl#%1uG)|NZy>BL(V}FMQ#z zE$Yc8_ZL@2cga6iPq*VqIT(BK96}?UuvEUaVzina?UoNDcZzbHxH1uw&o(Izm$v)7 zd{fyjg+cUMcw}qqkT!Sz!XCi)qXD}bvW1VD<*AD7vZ$BXm0Fioy~L_mT-xu>$~P%J z;jbxQx_3)icXW?*Z+CP{XlnFe$fk~9#CC-0%JwL`W7qV}Z(O*=WdQ<7=SIW+Cf3`d}+@B>^h5Ie?ym0@p^`}ur?)nP+Ea?hwwe8xB!3tfpHsfQ=uEF5Ivz#;{Cu}%Dn=Af+q#4y~y`aK0Q(X zhS<8ZiC*j1w04=&Iso$Qy|QP3ULX+O)B4vy?6#@0%Lz^%RF}G3^2+en)}Top>CwiN zn@pE4QhL-nO7L5}NkTVhH~090X07`)$lF$Xbf3}-IlRO~3;?X4DQHJ_Y=O(=)2dFT zG6g<@w(~+jn+sSz0ZLuqDo_@hls-X)`HhPf_5hixU?8xZ`f1Z zy3@JQvR%<`C0n-l)>x_Ay)~?hl@G#OE0mq{dwOeKMFFH!9Ua=_;L6|#?X~aBkp1vh zXX2eM8V_jC3Owx5J+h1FFx;bEJH}*EYpm3>?$B5Tnc=P0`SD#_%qHTeb$$<3G{zh1 z_&^u84dqj1pFmk%-k82i)rczsbkyIwTIq*qR_mIh%yGGaX~hJjNs%`x=Q2}thw`cv z*-9@FS37>ncA%?sT%~e$%$B9hbj<|Hv3p%K$$4W(?%GGIc?X@zH(qu*!ud`78o6 zu3qzk@CI3==-UFmPusYeSrp#76#OgeUf81xqD~*TL4KRowkgF-DRKhDoS0+LekPDV z9z&zMl((cv9*V2$1Pw>UKR=pG>dl~orJ#x1t=76`x*L4jj55)KZg~#py*^HMxzWRE z#owwctCwP|=yxD+LNtleMrXF=Ak_crppeBIMm4&b)Mxc z6KLD^IsL8dktJ4^;ZL)sv7`f@)$7r|6oU?85l8Z&-Sc}4de!UtV-EDk!LlS~q+E51 z6$58fSvOP#(^iax*h|3Dq)xY}r?!1Ytwr~Me%8O@$s`v+@mL|~`#P*r^tVU}A8u(L z27xh|VkH-*jW?qqD{@TBsw6ukrKU4S#aBkR&xk#70lhjz#2CELw0 zfAZ3VT^ECn>_a=-%`+~Y9^KV)AaPe$+u>GJYd(=teF+Rr0^hhFzfyDx{)Ug3H&8F6 z&T_o9R+&wqwZ*SQJIJ)oFJqnnp9R=!ONvCC&mfU8ckXiyRv&QS0UlSEbZM>P`3x*$r_M9 z(gDYppf4Hn*E-3xYKESV=TL4qX_m9RYO3F7WRW3rkbv4*p54KK9hJF}L%4O;0!QZ6KHT}}o38qkiYH>kQzedk2 z+q2BuEvJ~K#{Y?7q?Wj#AheFGL%YHL70F9ghYS56dFsd^{?8Kpk8eg{>OgdMZ0RI( z{5FAYAR8Dl^@EcGTR>K5+&Kq%1cN$-=qGq7TZz<_V`Urq=hOZT3NroT4+G}Fd@cB2 zh_a>8B)Czo^FLScel_v$G7|<8fk8)Vy)h^ecT9E_`C~I(MV{CkSCLoSh9wIWNTmCP zC8)6~FWlsEWIeZ1UXEIU=?1#q!2JT-C;nq#>i_{QDJ*9hScbP23bl#)#Q&4i`M=lT z|5Zcz?+=t2t$|mCb$%_lo;i#DY=2=gnwKI0RD;A}`qcr2idCOAx+~;nsl>FOrHlnd zY|3hMs!-S9g3BO*jd)9XOwOM&4eH(o(YJftnBAl^AzQUBrx_Xa3?7iCI7Z?F* zG4YEAC`bnk0%v%lMQ^=jqCP@?2khK|1qS}XVcIi)F>v3a)L7-w-|<*4>sH=7@YMEP z6WFNU2AvZsCNf`{syI1#6GUS2znRG4t&aI+-HUYlltx;#dzG(jm*@}7cw4*0F=ca~ z6{hA*`^$Drzhz`%RLb55{0@tw-BLMu`Yi{P;RFoyG?^BWh3?i)7k!Ig0+Jrutq+#O zuLtA{>KlBpmnF;gko_n7VVbq|(~y#9!>nD8ddbjab!7Okg`Zn~sNQ7dk|i*s4x19E z4W1v%x;K9gj4qgx*Higs%fBE!S%rb>(bhe$b4K=q zGi2o5W^M3m=Z!WPmW{Q_O6&FDRfQ_Wl(ycu}C{V&EV zv2lEUI4lZFQus zf?)jX1y~PhRd5+#jH)wi9@S^}QcvC5ontW$XSxcrXV_4UhN3c(TT}Wn+Y|b40f3ca zXg6Uf(>i4r5MD*r5{ORbZefA=r5`Ae3Fv&dA5*;*^P9Bk!tg&!!$-4X)00@uI;1|ZZfVm>^XAX8!#!Cr;|VVq=~8Dk%7l-Om-E6$ol^9Ayp@re z=Awp3KB&xBcm_f?v~*bw%29n`&ZDLjd6|kf?hGHvlkOym5koxm!?xtwwQEO9cdVuA zGhGGI9RwH!(1WX}2VYwofSb#!o#N2NK|Y`Oolg+l+6mtiX~8UcG*W%Q0PC%i@AO0| zdL^V3Q$XoN%r~+fLV;=}ZY^Jo42-%0j80C6@ldZ#fsGAt#PVy#P-)(i)~s8*tINf$~vJ1i5>*`n~vtpKvT z-7>re6fYu+>@#%HNjP@UqHcM+?bIm%UKGE8=4qN{Pd^>bi!JD8{EP3CN$qv3*@TG) zLMcUKC=-8;CSP_u46N^XXa}C9<@fFAOK>rUQxrYNe9DtGQMSqDYS8xQVAVG*XX8Dt zOYc@*K9qJ{NSze^H`TzNdP@UgyM%&B2-E=S&u5I`6iq^N4g^q$p8{~)W9=v51!(C zMRIVZAQsu#@z;?STM8kBAt{7Sc9r-2m=|Htd16Hld&zF7 zzCw4<64(KD*K3;^nG|Aii>2*Nrl83ebo;f(8KOPhH(Tj#86QnHU+rt~*K3`kJX|04 z$^$$D__d!s$6aC=&~FQS+=2fX*vA^okk10IcFiBa=IzAfgFMa3ccCHh#NqL4m7=Z0 zT7NJv@H+5{Uri%>9s7II{TFJDEWJ;{-!pO2yo%mZ2MD!nQ_iDX(CN9-D5Pf+w9J~#KazA`$7?iy5G^Q3P_E`q&36ulhXp(!b^o548o#xhp)and}@HZ(n zcKHTxgWs)PUBK%>IErqw^)fef*2FB6*mnW>b}3nz>#>6%5f+8fTrY%X?jVVw0x9t;86tkUW!>?a zqDi0D2AK!BkyI&?yxKC6M1m#k*M%Y>@iV66Z2c`dFrSe9L<`nZH3! zJx*E+Io%kIPM_8T8bUbIS3tI$=VW<2AO}t}y6F^Pn6-q@4UFK!GWR$8TJb7^@D5$*`-dJEuwqYpM)Rz+Ky~^E*fF#gbycW z|B|ku3LoLg#qPU3NWFK zP)aakO`+k@etf3dtZ+&@l!DcoNozYy*}PyTbbf4ux` zxc_|lsc`=U`H67<#qzJh{Xdo;#ySAH zQu%Jn`{AS6$_E<=P>UT)m^Ao{uzDHYo)z9~nNlgW?BAYcQnA*S+P_8F&3##A`#|_^ zB^RwLTcl1>B0BA*$dh;>vPqGj;Ysj7igeO*YQG|#A3mKQGl(B`f*&_cBBE9jQOk*_ z=Mz!SC8C~1M8$r?cp3aVUmit7J%NadHPVmeT=t_a={gqw)B*nOOYu*cY5{Fo2Q7sD zs(crAEGAN>w_u8kMrC()JO{t2RRe?%D{+Xfg{wB7F}=A^HG}G*lZoP3cXwwgX3k@_ zl^Gi^2LJw-^@Tn!#9*t&cq&y(;YE~TzD527GM9CCshl%|siF!<1)O=0+ZGl5i3 z9lZ;LEjqL-yw`bFH`;sQds zqGrLI*oBMOCv9RjHg3ppApIQwP zwn%CFi6BjNbVo~H4>ScO1C}Lxi|q{gN2-TR`>_Wn+svWSW!=4#t&^zX40|!UYt>9$ zTkx_AdlIljd$@8lCKGe>=|a7XV)bHY^^$lZH6JtJ9}{3Pe^7JsGl?~+{{0_@kI-KQ zY_VVor;b-A#%4I`Jux^(Vl@SBpLXgIGAdxGKuguwZW)6e&MNdT%$q5^!8T}pdYj!Y z=cCNUD3kc0OC6_Z#eM2X^|r#|&2X2Flr7O6tIi4cpCKEzy@13-U)Y0^K38^raa@^E zkoXvIh)IdNQ}iW!{TI*xLOM>{D1pDm&2(Tj{?_ z)t)TY&m**{pRj@(WT@-%Cg@{>{w!jLl#`a$Qv08QRppq~9@UT8M|mey?+)B-z=hl@R^F?^q;TiP9dIwGKlOhiQM|GjaJ&213 z;e}+wRf_xo{Skar=33N+SZXmFqkHt_n6h16ZI#;pE*NlyU;rlZ(Be!M{K^0d!37Hz z3l?~Y1@{jJ3z}((v)wWg=-8A-o3h{Fg6?n;`-9Ant>6dRkRQ(E$VY?P2t9lpj(qtc za|FuGssP|Y`6NUg&_WmG{hIYr?mdeiYwi4IzJ*VTW?bwmVHi449$P?kz<-rA^%+EP{h9<7zA-K7vekL;h)fMq7ua-oOvY!WjWiOQB!84^C0gq1GKZ zTcpA@=F?h`iUpJ}P>e)>-~!6BXvH59hPCtI`o{bMBvjMDdW=|1X^I_9!i?wqYD=np z@or5mqr~`dpnpZ^TxnoA+Njs=xQrUM(~g4FN&Blr0(jR{tEXPuv5yooXErZaC#v(U z7@^j(J&6m+j9Y20Ji5LEP~>6jRs2w_felud?N2E9fs=JyNC$aq_^n(~9%1$hTSna!m=D;tw^bbVovR%_chA4qC z!>#ma+aG2xBzfwg`M1O4e={hACAPzh3I7V!thyautqWD>&I@Aky8X;7lVWiNO<0P% zJU+knjxEbnwg=N%!l8Fe$FC-Qentt>fCvO-CWIXRBL`8*byF+Qh1Pe5tHsx%y+jGs8cN}_98AXK&S`QQiJxR z6nZCTj7P?AY}pQf$IL>9N5}6sTMb5i8W%g%R-Gv^tlg`AqGRaY7h74VL`x~Ecj(NojT1bb)ExRX>B?oIS)H7c!cn#);Ar6 z`Qb!qEA4%lk%hMMniJBd%2@eJ(k72NW?2z?c0ajMj+V9L#Gai4n_t>=OBV97W0sd< zB~xNQr|h~G2 z#(rNzSuqEiyF4%UqL1D>F3*qE?5xU@?zoFb09n9@Dp}tLF@k%8MzcG1$I@K*VK6dW z-iCVUigNNxl&U3vEkp!eec*c(f z$N-r$_%r;N{9%8H;Lpj&=TENS&v4?;XwVVNc_`NOv?=CjWo}K*iWT0y`OBHOjtL!{`!#dU;L(jbkLs95g(MPr z(xw~pb%w?)&x32jW9yiQ9hci;zMZZ8c{jCS>>q<=%qwup<@1AM*mO|KpC)Gc1+$u5 z3mUYqyZ<|8eGF##4KCHXs@&T8x*=Q|!CVR%Tv~cOF8PQ{gY0!+PBy?4OStb8^uaEl z7w#*N3kIza-(g`q$ljyzS#cJ|gbB+4i-`pl z?hHd1w%58&apHwXtQ0AN-wQ}oVD_<6DkYPEtHTd%BY8!8*uT6AkB7F=Vo{3R&(CC6 zNRe0gnT$j!(v0jw6kB4FmmJ#WM);f9~Lj zz+wjL9RQol?d1nna7N`o1_dYSJc%yFGw-iZcY#2uUpxO}3`95w+yPQyfz@2S|F6_j zCs{V+W;wJC$iW|4iD&2!UvO05elEAGMyH6>c_;G35UciRx55Pq8iy+^%66%9ESS^Q z%mE4OpGG}cJ*+N8iyIhHJGTf+LF&91kkp*A9pOGxVj`Q_mzyhlUdRUua@KPbyjV_j zSWaMl=?-k>+S3PWC!wp=9OeDW-7UR`wz0@XW2DU?2)g~?6%D`4Ri%NdTTzURIQZ4o z8?EXXUxQ};I}IO3>j{y|lXipFp6dCU^N|s6#~*_~+c9+VX^_yWHohGr&kb{F0bGUG z&8=_He!ftbEkyfK)yo(#RsxTGs5kZkkw7Ug>P3SuYn}$!_=%{x zq|1=+16*bjmtm9-pv-`b(E^yHX`P??fS;u;+5#^BWB)o!23c2rGlIjRopSY8IM*VsDDHGeo}L z{CQJS6))fn(%TTK$6sTif-#8z#v4H!6^>e2icH4i#Sh|-h!FM&?q0%3lmQ8)(4}pV zxNiJuI7(=&S0I?vjkr<|^ufOkJl6aMx$#ZpOTmt?wI297)kc>u(8#ccO|sCSh=)+% z(C!zOBQd^^rQV9}{HD(U6KSGGf};Z~xY{$-sqS2j4D_X3EmIrq2mu!wU5adhoIs;X zQHvmiw?Ugb0PZ{nXE0|`3f!7k`hl*5+zfk=WTrv8=^%20q!k1~?qY15P_>v3f@t1` zX3S5SP4PYGgkI{o35{w$03or$PZtW(B z1wfS0{(K2-sg+=#yNm%j{IuQ>_2yAjmJV1S@6`U~yeNZGqm;X}>FI5g=Voww>frW-{bh%S zAO!)+fHv5d!2WGEt)uMN8Bw-Lk++Z+TW(ekb&P`LAYU5U6$&W_ijTlS#dOf-!0(3Q zmAj?rohXaY*jTeUbS2=DMNQl8^_u%`1UYBShaI!b;rQg%!=rCLtD`Qf>SFoHur|4B z^v$QVHfLWfKaBk#d1Lx`uZ!P0h@b9YR^UCz05d{+{bI%<+?N%ybc{-DL9>AaXx|`s z2n6qVl?fid7I2X7);9c<5_qqIP{rCm?ce(S0G{SACWw>-o)@NvXgu~#4grD3&@S=^ z1nm84k;UtK#J6+4*7KoA9Wq`VqJAv}eoXqAC^R6y>kEH01RqMu#K)0_&)|IR(oB3P zU%T-5_;7v3=b5!b@R>FgA4;<5UylsMhmtbsvp5YO&L5!9 z+DRGop?od;!Ex!s^%ruYLah zaq;8&j9=qW{E99bBHxshiBEYNKAb;5uWK{$p?uAEe0);%wb$<-La#YP@u8$ld3x;4 z0s0Ki*WRznpbzD1J-d%fAFj{znKTri)}i=NQYL*IY4~vd0DUga#E0^=3y+Tv*Jpg5 zxo-%4o*jx0C1v6h**QR;!TH*Ti5c{veC^Tqj!Pe|&-jcPiqD5b@u8$leE$8$06v^Q zK%e4Fd?;T#>G=4h>T8|%3aDx0-(E39o+&9)eioqCpS(oYzd-#vsrj||0!k}~mG_qPH1aQ*;&;uRVAP`>t-J^>zsKB@ZJ ztfBb4JQ$z#{~r8K`kIPl%9A4vzrp$1rJ49qzINg9@#FeTuV;QbgkH|dA@WU0ne>Xh zHb5`VAE4KUi!$g%`P!rUgm?^kaec;T%usx$4aJ9&GV%HMs{{B9&ew`F@u7U}q~qhm z^%Kt99>exuVkY=PTe6 zZE3(mD;}C=)d6JlZvnEkAO8FMQNI--XOmhHZB|)%WPT$K_ZtpvsAW-3UO{DEM;R%) z0_%W8YQGKMZ}MZ;#8z4=kGA}4Qx4$+bf@oZJ)C#*XVRuuVxFA?2<{Zv5pD)%e>E^$ zR|o9FM{}EBipyxDCcX@RMZ70PS0Rp^{4zL^X7Xv5?8~F&lY+>cPic3~t=)rFENy&T zE}xD4-_yMyy2Xo~sE0n`wLN9Yw%$6N`7@dHtq~EqIhfvHSNLmctU7v7ioA}(SdsJg z3btb0ZmqC6FYV-G6LyoZV>K*1>7<*B-2o~da6v;|Mm3Jr%1d|#vC96;3jK-=l1 zpuQ3KJg{DNYo|oGZ8Vj*XNbJpf~RPsKE~>o!p{*ia1gPQ#L{O;HBzvY=)vRw+%FXE z(1A!p@=rSxosb72PdSlCon?iCzO1|T$UJ!h_CF_&DBIQH85y&*#w=6lQR=wV-u4;o zjKnO}U_0eFJgrC0i$!iPy`3V&_8WPvoA+iFBQlmqjo2g)B6txP^T#G4HQJ$gH|#N$ zw{~Cy$+04KJMA6-`EVcYA>B@UNR>4X4q}wnbOESzR}_ECu({PSIje(qmRe)$rhJ21 z6Ia@~T+`B)Yec@K1N%vwPf#bWw2AzgEtiP=oDNTJdcIZUm$aNE@~s`7y!3pF$hS2Y z^7>2{wQ>#;2K^50j^}~eM6i6K(r|PucGn7Sd$_3jru54mgpGcXco}L$_#0+oFP`Fv z$$d}mT$5Jo1y?92xO*e{Ny+E%6;|U~36B1zum6?(#eLyEYsObGozyaClC#45KmSk}GO|3rGA3j>8~=pYt7{ z#PWyXO6tpkZaGrI^BYKd*&d4NTIE89MMWeK*aT;*M{Fe{5eLppTXTf2qcUL5NeEzp zht6=&5k<*S<<@cy?9;PF{5I(emv+Z3!aw-Wb98!>8@L74!B&@F_z1n@ISfxTtbSwO?CE+LD1rF{!u-;I4y=nBLV0|(8enLnI`i&(4R%cign+v=3 zv<(jgirh&$tU(9l@kXRWC=}C%{EKewJ)DSPq>e`~|0+SRO=oA)i+~foeA*AhK+T}n zYj7#+^l}m@Krf;daHGRTklP9`a4SzIRUwnA2CPXVm9Zbf_?{#1O}vY@;7pxZ!RtY( z>v0NDOb(>1Qu<27+hF@VXlh8lL+)sA;8_d+x3XOe-pZY)tpM2rbwP)9p+M|zlw7nB z?o=FMBt)x;lGSd%PiyCMK8RKMF@o`!ZWMI~9+C6$81vg4ZVT~x>rnzozRV{eoj^mk zR?_y9uwFxhG&IvKYbN{lFN(n4y1%UPIbkpcL(j zTv2R7FjHDLNs|rjW+ftDQ{9vzv|VuAt|I8M`Qr5ceT^sq^d;55?}QMM);?sJ&lX^Q{^fosbLT_975NBF%htQ)DB}CZQuk~qA-!DOXE(mv#G0sSq?6eV7 z?$P2!EKe}?pbSWN!QjM9I~maV{vw^=fo`M%i+(;|nTCaK?uhvY{V%7foM3r<=Ka_L zi@OXG>;*^?Zm;$qh>oJAG1e zmf3ZzewvTn&r6Tf&lAy4rX}(lw8uod1ANlyi~f$!=z79ppQMdHl83ghUqb_sVHb|&eo8dVB6WuUrRLrETL?>+CNJVu;j~J`5rLdW z6dGo9jv8U}KnkCoZ8G81xJ?Qd;tBEHD2?hQ+&-K`o>`9@vt~9R2z4f1k22GvJ@qOZ zspU{RIJwn+&M-8O(4lclU6$emFC`(BwB79*H!}M!mxk6n0U5mmLfGbe_+g#@Qs0c-}YHI zlECM-!}NfDS4)w0U>DgVwc82H&#R^APv{{zMq2X=JhUaVz(u7`ij?BFElE6;A};(5 z_lDv^6k^(G~@MJc06zx)gmMkJKsL>Jj}b^%h1)?ACqc z=P*lm>>>R6M@bPnN*C@wRf_E32ZYgd^8>CMM1-L{hm0Oy#lr3b8R8aImsaJdXhp@qaBGgXn-;c)MW@CFDW)iq2 z5Q*^!wE~%{*Ooa`twfOt808qSr+P{ARd|F%O&4V##O0x@>nV_fs*@3$@XND7!3G0S?&lq2^?o-0 zijc$&6jo`?uZS^Xd{CGN(XR3XDLfC08&wRf*o|gt`7RzW9yP+hZa-*OF|OXEyqEnV zr_h!;8`m@rYPAh8aWQ()hsYf%xGSEghU4FZ6@%u3dO)YK<_Y|o$jW1Q_k)aW{CZHO zA+R3lVjyYXtQ8rjV11!+2 z4;HKx8nnp+5a$928KT54F_OwR2@>rA&oBmTFnA5{1R191bC+-c2CXZk_6tcp`U3rg zE(8BN9Gw{N@rS6sm~4#tOI_vlsGs#2PYh0>6(TFBU}^MV^9=}jMua4;KkBgLSB<^d zjyP=_;a6d?}cG)BGlCT1#-)d?Q5s)9N_2+@!4#ve~`c;F~?jvL8ML0GeqN{j{=Lw@N1 zI4+!y#d;(68avb2qUrtLQ{W{?3@qbceEc%G(zC9G(SJI#^LoDAo zJ(e&3Vju&V@oKl;m#X|zRIX8T5Y7khtFGV4$UgLnVCG2VqX+2-lB@v1`g8Ie~@7&oFSq>jxj8ZdtY<|G|grAn+FZIES67t~SwDU?W^e;QnL z*+5YW(2TdD=wSMg-hfn@;@2RH!*x>{rYu#ta~d)+SnTSkfr(l!1wq9G-E=d}o{i$rID(4?Bh+EJwh$3;;jJYD=Z*CH zHL(UoEnI`8FGjg=sa`mJJulWf6*0g^*VstKuY<-ger@y%|NJ{J?e~%UmXysw5WZHn z;O{id#n+>ux!5bZQHp#@(o8qCV1uDznEV5tD<1kB_ZZ!TKdh&4KN|Ek%JrHs+p-Nh z+@rm>A(^CA_+kVpy0y`{M(kxaZmA-eb8U3v+6Q>VW)QA*x&?y3P6r6rf(EQW-7qh( z1hxg90nos5u->gb7r4Q0pbMv zgPC}6MYSRgi`R>Bauc=-2jy{j()Kef9&IaTA5cxdTqS-X239(#Zd4bbauR=HIk-?D z5I=!=XA3-LlnDG9Pl--K;K@|l&+x?e&!)*IhRkBLVJxhUBdgc#BJaboBz1*)2@W+t zPTL_!7vrF&Ns7QhH&kS`Pa(2B_%k4~6y!J{ve`J#`lT71#1&zP zy13HrAlc)!plk`w6Mo5TB_M}DX%;$h z%j$ddO2Ub&4fpN`Fd>bvSd~>)WhA5#>5xWl@l${XiBQM}mobhs8&YXSY;Ja^5WIo4m}skHO&R41wQSopK!y8*pjYvp`$s~ zkl!Bk1M<82SKQI~G$D0v?f0T;yZ{O5bsed?IU=m|2mmlJA)n8Xd=kM$8p&tkXDR<* z1rY?`9Dtgv*S>lKr~WWH-3UI{rswlnLv3%C`27I;UP7?IN*V=1lg)I^O(~ehKo_Yn zIG^`jZ4|}^1i9lNBdnBkL7GcWL!p!!>a~~9E=0iZtJfAo8WzokGy1l>zz0O7|7i(3 z3a@a3oD0-R4Qitm#TI_S^ubN0;17CdYWAa}_1Z9uBCrfT+SFl2gc%c7Vjo9AQIK~G zMiJBlSJ{OYGpk(5Rj?QII*J6Bi?^JBUi(uVjzqiq)a^{H$9|VF-l6U=-sd2j$Gcy< z=XJAbKH$4?=2bh0ixU;ZZ+Du-uaIXf>|Y`VNRfj;50jtUul*j?nA%dL4T&@|YH4I( zA7+7Tu1_mGdBB+mDV8D+p_Ipmnm^b{M>k3HW8>|IYOZM!IEoF&AciCAUg5(%U2dE` z$E}^G{9UKwOGHAcQ^j9&*spB@3??C8tVV7nB=Z$c1}V0oCsXf4?xeNZ^XE@cemu32qc7;-KGYBd~GQJ5dRK_y1r9Be(4ne+CY}W zfhR(hOM1Zp69-?U>^bWH(7KM2wr1SwZQ8cvW-F~!e?@KD~n zlm`a|wZeH7miS0L@56yS`8+DTmlELIL1DR&622Tr2#q4hC!69sq0i`XqdC3}k5ud* zR4g5VI#c{f!JB*4_8(Qrvd2}$EQUhkBX!msIndVVA z%m!^e9*GLOkw26A6 zAn~U6vAYU~CA+A8$mBREqcE$z@LU|b&TjsOn1cc)q2O4Dj0)y!*8&n` z8|<@1_X(0l$cyD}2z*cSpQZf<5+LONJ~4oTw+KX*27@&I*YcR*Nk}MPhJ77p&etUloy09einPKnh@*Tnl6l^_0gb7RS&A)Qb;v4Xr(m&Xbq$6a5&W^NG0R(HzU3EfO zk4Q+JKc`kt!-A6W#f!F4Ky?R)p<|mKWXz3P7*;}_7w9;FLo%r5Bz#Dii5a1b>W?$# zc5crzH7-gUb7RpKjX5cbB@4+BXxvn0_1ih?=z5TUq32>dz_5~@idTrE;G=J2(W^Di{ zvR|Fwy_7xa^DFghKsJMO6)#KCX~+eDS|XRetoHTl&nUK;jM(v5jYmf z)8r4@p+Kj30JO-jU5{cY5`7#+G&@?Ifk>t(*@YKZpM$AC`X6ealnvI%pcP-Nh;#p? z*r}n*A1mNytoqKt(qSN}JNyMI#e~ZC@X_R4De?rsxfz2B90L2~1sd&EI)S#89=F0>F=nVfDY(*?Xx%&^yC(3@Jcd(aX( zR80*{K||EW(`cCmxgJ!_(o6yt&s89w><ElmODrq z;t#;5li`0x9&H^FiGmggT6sgzB4$Q%1#2<1fnJuB;%*RIoTAn~{f`+J#2f~FY#K7*!@g4m#G#Hbm(DgGN)hE{YZ z*#b22nT6){&;Zqo)*bjfg;;_1xSvId`14D^ zQgW2tYik^z1-W4X@(&nVlJ({jQQy_qbD>!jS(1U)J`oX@Ic|@)hy#NIk#desR_tWFqabxu)%*NDA!F7Z|(DwK-1a{7riSKwM= zc;X-WWi~uI;wGTb3G*A-U|5uj{|0IO$(_ur&|ZAh1vQjMcq2x90)g;_$J!mN3pk1Q z)mm_lE;S!d|yJRqsch`1}Ld0{ndj{{Qg!X(MR|daM+>%Jyh)2q{-~C*DN4;H9Dq zlY>hkTq|LRLw-ds2l&hJQwWkpy!jsaOsW+UwN4HuV>NJGo;H!|>1%^#iEI&kz|2_h z34bqL@DF+9XO=@d&zR!q!G1R4C)H}lnMmek5ap>;9K-Rhpn#sAcK;W~CDl>f7NflW zuGeRHxF1Ycs|+to#SLQmd`IxJaO0SsFZ3-hS0z44!CuNC>dI>zuaLJt$h2&|A#LMH{rwB{0(hTt1dA#_XdM&LVbFun&;*}#|d z1BMHSK04$ZnXcEV)o4;)Ea*BaC2KGxQfOWz2zwPWfhZ_zAB1Otz!S6$L{f+nJu%f7 z3tCz&vMEQ7I)lE;fbWBM>GTCN#3F~inXJE7B|bdHdic2V;W(yz3`=z*Q$8r}5M)M3 zu1@Btz&>T)A4@)p)9?=f|J~{!`S@WfTc6bB5=S-?HkZU-t#WJ;#;`L8WF(q&jsdaoo?=2zm6XEHG#4jVpnGq(1v((c@89%zQT|j5=@{6?uxDA*6v2A7%I7hjxr}G%0_tWaw0|f}4hCop z#ew-Ky=SytT<&Q|>)9JL$RPzKh#@w5wvl>9*J+j4Hyop98TvYnzC4D>1Z8Uk9(*gd zhCmta!A}IGuSmz|e?@8NB>MCs)DxMHsn1Xzi$X8AGlfJ+@>iz$({%l#2VXnnMU3sW z>T-N*P`*lFdzvonV(T%Dd>5$M;!X=vKrSH|m_>Q$Vv$?oNu!@Wtd6CB!UOUVAhik+ zITch91}BmfoTXiSXbKZQ$Fm+)Im-qYVt1@)1YydwuT@0Ox--u)4H)1G$D42#{ z)Ggy%`sH-39tN?S2A-6+k1q!Bh6`w@D$&@Ca@@m3D zz0HLSC7GlvGup`=)Q-XO+<|rsmZM-=JF2e;{K)|slqoL!MNhL8cTsPRu&Qg=##Z`B za{2zTAsQ$U#fRRWPqUw}F2ZEOZex6(#46_*RQ(qOb(lJ(Krt&jg?%IJyMmNGM)4@5 z0qKihHhF^d{HuvG^$C~dE7}tp^sajG@hN#UeH&9gfG=j6e92X`v=6gCUlC$#mE%(=7I%&k}|U+u!&v4dOq- zAoj4iF5fd?5I-ag;(GxYro94#c*%5vXP@01&eRWveuWDzO8#}$t@z~y6K;Ek>d?C< z;@{QrX-lnS9fReNxJX_$q`T<L4jQbai0!@hCM)O3@LpbfArb84|g&#IJ3|+Lz(A%=|AUa(rw? zh1)|5$h}FZn% zISRlssq-Y1A`{6L-d0gh{tXhqpRQp61YTBBM+kl*ENNvfMf6Xld6PRy2yp;K7DE%} z>kgDkv)5>SI?bhe9O1oEp4)*BBPi~Io-}*X0RTnflpzo;A_GK!1r8U-2!}!OYc$uT zZJ&g@E8N;0cT%xJ*5?FI-0j9$YE)$D`Vg5ZeOXHD@wE4HoY*q-6CBJDKE6iQHRPk5 zbQQ*n=@vSF2+vt+Zi}~Gy`iu^FcW~|RD;*cZAWOmCq>Rig4Fo{p4*Pl8cKWx zx$Ov<>ewMct}n2dQ2G`LfiZ;Azpi7H@TDJ8R|Ihbq3Rx_FVxD|1IeRDj~;sOjDBqC zZnm7&ihr`!wzstvhn8|SYxeNwEK_&*{p_|lRrdtwuTQ&rKZF6BJs>;Vt8GVaXH@=z zzimgz?31E&#hO2G2X%$>T4@Fn;`xa7mGQanJwjLl|pd7GL`)q7nOkKIvw zK{EB3+F8k3coz;fpQPTUXO<E}T2H$g zX$|nG{_SM8NKkK~7*?olquQ%4GZ^Hs#-@B))wlssyr!EiWjos6^aPd|-F%U|IhB#J ziEiG@NFhS*M!*(GH9hI$yuh{8tTvY$wLxx3JbcRGH2SLMSQn4yoo8(MZ^m=+VNXNy zUY0$4P`U)&2ZIZsT^(i|9>aet{@W_CK_=aG2}wKV1$cApn&Q)L{s|Kq5>5#EmG{%;gZ~%!z3=f2#t+0TY}-nMhgD9@mlL0b`L=9Nycj>a zf3X1Zx9AQUf-A-o_FqPh-N`=BUI8vc4-U>B3} zuW`DyvnOG*vc?G@#a0TSG(ogXy`&Pc-z2&u_M$x6)rg@&!{j?`cH<-CbTJ1$ zh+EN2f$kNts#DvN9}UB#x8?$@OX_hR_<|-1;;*jPE`L1FRE%JFEEdeW&Di&cVMxxE zN|*{~(ITdfLEwuuwfYU`on)EE{Dq3cl^8!*Ue#G}jx?{LHlgl2!I+Ho66H&sXCi&Q z2|e(Rt3$Z&GOPJNZQJIdWAr6}Dy+F1v~Ry6F{AZG=Q-%M>bI)19I-+;MU_UY6s4ml zu}xH^?MRZag!itwjqFFVRI9w&1)~1JtuSLKo>KWF{MC2xE20zb*s>->$J^_*%Q&>o z9!_Q{+nPtVzBuV}0K8a=^Z|P1Cvsu&5w9AKScQEOr&s$9zlQH`7#qXXaEFGr)rPpnLuZv@kAW4B!95dXaNftyDJ*dq=!pJ9F zGUTC-aEy3Sx@=$Ueu9E;`N3qzf1C2&>cR~FW`{Jj^>DsC`jX0znhsQcBt?dU!UmdS zNXyr8m@Dn9Ujm-viYcDb6K@FoVGiJQr8Q{9XSclCc+mx-(jTcVWC+eEY_K=t)ax$^ zr|vl2Rjz2FAMuNjb{WKAE{j3*F!C;oAHZLCU_Ggu-=bMoJlfT#i}B{S#GirY9!!VV zNS9+8Iy_F-otTSA=f`F>2?wTDE8-x5*_t-M1pf-418$*}`7U_{=^0GaZY_p@w@bhT z>S;VqU7#rs^q0@CM3NM_2Q6_kHhft`Lho!HmQ&j!qlnN`Dqm^Zr_8d(m!VMG5gKk% z^eRHOPUt(KJL0pEMB<65)GGe~HyFJfdpMf_Fi2$JAYKaO2TkOlflT5%#ouu38E(j8 zoff|g_!&Avr{_eFZje1M^r5(~gpSmLUFvFQP^#Bfh%+|f7Q0y}yoSQwsbXtA7Zft} z(yjmW!!8+of8ZX$fQ#_8nPNocww##wG1g0X`PjT07U%`|0;dq*-qPe% z7a|Pl^3>s_p61Jnw}V1NpO$xe)5!`%ir`6EgQ${0o)i#@R2Q}}K!ZYetO19Xr!nQ7 zi7->~1uV3H2(qc8(Ptq9pr?Mjg`h0$0bp6#>{51bA{wkcbhP*_s6^^K>e=1ee|}2c zijSjJ4!I;tys%kbeE6t#i%Q>0FojjNp5BoQh*?>ZS|8zxCh18gf9^ye5LI^ zA;8%aYb=Q|dV7364&NKVkOWbq-K(C&aiz-Pp|4oHa6g?m)HI7A7baXcoZZGX2svbi?rqw{7t+?9-NZ;;PChs zyi$K8obURC3e&U}yhdFFY2C=u>S(~>kt_bmM2PDqeB~Au1D|EUXD9>O5_!DD zRvMit70s#@TR#vXQzKpKqpr02wZ|{Sxr9WR?&D{FI#4|*Pf#13;yZYk;%-AsC$xX5 z&T@LRn^3&B&dE&}QA-S%!JThM%#yRA6qj7dPTtzjhdARmGo}KdPM6wA8JRq>on8(xgP4OY$Rt6RehB3u!?V z(8*tm@wBI@K6Dq#rnWJu1dV3x0IzBg7}0d0{PDng5W3Zh_Fgl8RQ!ntd$$mG)ep;J zPFgd25*r94&E<5;LfIXs56p;Z&(KeEFtgD0BdPsbc@);FC!$`8V!R4CHpToO3X09L zsLKjee_^clWRvo0aZlwDDK-L9PX3R`J8xS6AT9_Tz{VD5GBMLVR*sZ~sOlBk)@HKMbPY z68cLzQPo+NUJ(G77?Yt-h@SZGD*iMt_O$z0eULd^diG01d3t>b2DmVn&5+g)i@#`_TC~8GX7zGHCt)e|Y}r zC($k%m_MpO-4=XA8NN~+#e<8J&R6mEC-i-#aQ|>AGJ;kg2le%>I-AA?ZjK0}v@s*3 zujl&l6_!&7(wd(#Bup>&P|^Nt@KxSLRC$lwt1HJvA3&+nVx}I-@n)A_`^CpZaaN8s zaVGx2v?F>&x3@&;drE2pYsg+hzv$XVxbTFwks|Y{AQ2unLwMu68KICAnZ^j+Dn-X5 zzjB;(7e&`Xv*^H->ues1Vkr0Ge%1%_#5oco?hPoO!EZPxvfaTDN>_Nb7DgYlTBxP$ zpt3(cd1(2vgSev=a}C0cSRYm5Bk*+Av9-#YFP~VQ_$C8>U3Drzi213mEC66F#bvfJ z>JgTIHAI*vYTf{VI|ZkJfmf5OqCxx05N|2^d(yeyE-l}ixRb~W0gXHdno_bVk#yA4o6mxkltpp4)ESO8 zY<|(SpyJRp%y8^d!SP>tl02p%a5c*EoKmm-;#^}s!P!eh9husyTvV&MNVT{sYbxw|KnSH4Zn% ziAQ^BH}OQNa%mx~ZV5|VC3G$zm{(Fn7TE_2-UA3k8YOXWYBG}I}6)4UT`@mP!BAz8$td6rG7vxh9v zeW53}=aO1Y&7xG3Afx-H#8$UD%8P4+>Aoc~C!Yh9h`X4Rn@>>Q$5gLR`|*VsXM!Y{ zq?hX}v^{{{WC^{Sc$nIN^gM`&;CltWFxMPly3?mUcee;h_n{E1>3jO9jRJA2p<(TH zb*+Eoc=q1%KyEcw^nb_j(dy1gH~FRlqZnjV`|(PcO|)P!toPm;-eZHmP$cs5R9I`A zAQr;uM%5tyVV&x80(N){HYBr~_{*x)Ji8R{nN7#TXTVt70Xh92I2VOm)vzyk2`J*& zj%|-k!v{D+Ck)tQx;(MJuv+&QRwkr1CxVBu$OVs}Kd`$55oTFd>l1m_kW7NaP3$UBL%-GhmI3N0LdTNiboGy(%_nnz?ZNl81{Bb zLZr?IQ3zzBTt|aHw&@YPH^l0|kY_^BK!}XS9xr@CByI99Dm6$|mXd(&bC&qrQ>nN3 zBQ4yeTa7S&NcQ%@#U`x@vAm@$(WASP$@v&ZxKiMNxDY}}91fJxfX5MEeNCq%;$I|ydPkzM4075F;pvtWlN!YAlX;o6AHQ>WYvnW`g2spjaO!MGdE5B3z%Qb@EJ%2Mbu zK5Kjd3Z?9cf$?4nJd5BfK^Z&wOQxW0RX9J;Mxs<7AO6_A6+|#EJ3SV@l}6W3kj$%p zI0=eBh6{j#eqg4<-AOsZ1JL<3o?~ksp}$idx-Uerxk#--pP&Z0J!sB>+XFvf6pg@g zm|VCMRX=2V4}p2D>Qc zz(ui3S|e^xf&Zo&Yhg4^)0@F3U{kLZAsre+$MG!f8Me*v*%OkK*o*(~#eGv<(R&n21JzFjoEQCGaP_h=HK=qDOS~wjOCY z$e(+3MCes@;<(!ouFN1^$@ZCzq+Sg)6ETgIvjjqqBw$F{B0J!iJxY|Z4-=X4C z;fHx#|86XwNZsU6Sod>ftzP1j--R4^d>bkZqL%sMlzLS$W=<6cpRRvf@u(yI1nb6t zlf~Zz|4YOqUZ8KVBYdX#gX?(<;IyVs#S3p%I)OZJ@W^@;Ttp9CqtrekDzQ1t!ssEJAB)b79>D!CXFnHBRVh1v$6Mkn zt19FgPCLArCdG+GhpsPH`*#j3Fd8I1!=kF_fjN=Q>hyrouvyy10~_Dmk$+g1iarq} zZr(&bHd%_+vpv_F4p@rHS>DwwQ$71^$$>j)%4n3FL29ceyuhpvI@ihL#B9NGyO?y=OtX?w`JnnU(RwOvrPhe$eV}4f7XnV9k_U-_3lH z-sKpgWC+KS(f-K4FVzbu$f65ip9U+^_O}7X*H)rbY+FeXRLX9({82`wK7WPI`=vfB z*^2*98kh9p8{n4!%vz13DdHE0p6Dy>55*Vyz()2D&DP06$gcbrX=rEt1uUu8fJ%F8 zuVYEdc@-9W;S*Rw^r2O{*cu6{c2oTC`d&~5>heFajurODDSh3wLHBigT3?Cs9>2ru zZXa|iQ||zoPQrOaEqz ztlCpU5nubi`jcf5lkg{Bq_edOX}$VFFTH=?xv;sF{TH5{8U)3&uMhWqKKOBU@aoIb z=`ZV>T6t%mz1|(DQmx0Kl~kg9ZZfZ}w9gSNfB?jsQ-VTh&;O+PM>9?yQqX=lHcHYf zO@E)Lo&3kBvtE&qtN7DH8>G7w`V0S$%PY6T4QiVvzNv%$@aQ{&?e6{A&@>2RsUty3 zpDntG9h;j@=Ef9~4+FHZLZvC*&z}8G9;5s5yFI`U310Q|nA^RMzmD#|GCXzB#cb1r zNt0O{d7X(PXpP2FAV~cZPSzQXzoCBHXuV<^tz?PJN9O1}bAEu=qE+MfC+v+27V{kb z$psR_DE@V8J|3QD1dm387@o`rU96dT=JX8yb$-ShYi2;XiwMy{r^)`JoD}}B@A?J6?e~SdN{Jqkx7A72WHD`@6Ms0J3WdfV2uA+|TX;va8;I zsURMCj8$ebJI>!@KR{&}cS`@er2p>KcCOq1ro;5F&nx5zF2F>!Jygcs+U5(v9~t%x z1yNT(Ou^(KgQL?N4D8&H@eGw~F`1r)`{e5^|Q}A$Kx~?35!G$Nq zT*IG{t}kkUs#d1l+8KB=a5>#bsGzwP%^>$Z*_zSbS4)F(}36tjXneB|Whx>?<6R5Am8B3=%Qll&%lNQF)8{t`9_nAN&6I zwiZR5L^n0bZaV5ynGcx>LFV4@0-RzT2$~TLUm3m8-yX)?B^`p?DQ16jT{ouMN`HkN zkBIPR<0(?u9;MKUPaoatV{DW3NH5YOmB6k_dpr99j3u!_r*srX3h2?({-v-oCClui zG~n?U)ziJYo|~ZZHuveCwDG$77cc8JW&d$Ve1BH>RAc!?P)XElz@Y5F7lEQp=Q)@;J$6-~3a2&q+dr$28r}b$-518Rohisi?O9PD7VYVTfoTh~qvfyS z{uyir%~H>r|B+YK_JxU8f7UoyO`k$u72i1~@_=5u&7(9+hsq_?UmDN*b5G>YS^-C^}f3=4(C)f0c7*Da%VH~r|r)v3XSFX4cqG{^{5n1o>9 z)kL9dRG~{K6eqUD&*>I>c(V(FVy#@Epd(xJ>GI^B$C5fI(9fR-?e#%wBfh{rCG*=Bw_n&l7E$PQtdaZGPozv8N?g632Q=zIWCV-u(mED-3-kApc0{()j3WV3Y-5nx7(Z8 zs4&}{y>QC&ylZzpl~vM#%LjU^b9|glT4}bsi zSt}I}Lbsk3HCAy=b^B{tdn9;JJ01EcRlznr^9x!)0b?`}NFqlXlph>;$g3mKhIOcG z0)5^5h*G8L8f+GVMH#+0&}vZyiSY{)N`fWaw>`?~UXQbl_*$X8tm4!41^px#dD zTkU(5nd4E|s;v_NNEz|Jrp-UDV)-SQqI)udjU$%4KCi~!cC{q^G27gUay!p^*~!E30b3tFtM{9R$3|P)o#r7va4A)a+WW936a0-7|dd;Fe}4; znvQ0frF{#xlDMieu(Y&S;ReZO^+iOQt*x+cqA+JAF(=7o z*49a!iKGdtk+*HAbW%Et<{FKULQf6QKo>oSs{`j92?E;39_xM}unT8~TkIn=7dl;Pu`@Wwm z6{+)p{ec(**5qP5)jml9Nupf)6hG>G_OHjPZ?&=ltd-hz*VWBqf}f-61f7=-fhStO zE%AMhx+|>q?}-<4RfUUGg*@5n*KKFnLsi=W$5d^oN$*=E3WHvZAJrSvLtaH)7Ibe{ zhca^*c$o(BMqiBw z(YfQu6DaiA-@0TNh5B8o)*mAiEe^B~c1vA@Dcd3$xn#cJa?yJm>$VF^? zGQrE@q}nbf03X~WHl+39L@ySVO`BMp^wK5W#Yrdk=^vuPNh7|E&L&TZM?3h3Tzk-} z4&vfDzYFW^oN!NU&ByoaxxM~e96A!m$ojuo!;6iPUc&$`#KX}!9ZE{=zzQ+s~m z@kGzso2q*LJ$$X1PW6nd;jg+}e5Yq$YkOfJBN{=|CDHviqJG9;osLgfr`Z?_zS8$E zSN4ObmVw{1Pqk$VQTYs+d((TFefiH)$?JOY;>{Qv&x)D(C87tt$y%*4P#F!c@IuUi zHvX#4<9Dd|rsst_!cEG!yfM$bi5R6k<)B46tz_pR6jboBLX3rk)%Tg~v`gOOUBGjM4axR1(&kxQo}* z^|<`A%o#b(h13?a@(S)7`lkj|_9`SBBCpoCuX1>$X!u-LC4J5g>2#FenW+CFon+!- zLGT%Kq{5ot`G9%6a6`q?37yIdcP*WsRY~H`s>}+E9pM-^HL2k8!1xXS`H$UrxREPO^EN$Ku2c0XM<{(54*K%M8* zijKD*)7DtW^}rRCcCM>tnmeusuKD}-)&r}9m1+AsqVMn}BI}ZMg6+z?7_ZV1jI{_b zYo&sHj%Nx=(#Bg9J8?&CnLXhV4oJW&+(&$yQYFEJqN}T~?Kx6gT2IbV_z|We|LH}- zLbju&h}YE$H4{md19Fj$qEmW{-gA!TEYbSIab$!3soVVc##6t z_I72+alLI}vr@6OOS~$zi}@9QF~$F@@TrFMzvO;&Vn0Kr*kd187`_*3 zNE&cmIrPDx%Ir>hQ0enre4E!EIucIU0600})8U{~B~(jWikW5hKe55cY{I5I=jBSW zgAv2i?+n6SF4k_O(*%ifSd+n?WpzCu(XTEL6z9}#@=9>&)y@0{`wRKR!c7;8|Z>%<@WZgT$T#=;!SS4y(8O_Df>6>dX;h^crqyG zX9!Xk?}(0pu4KRYMMRnXX4#8*i=W1dQ~4LKKG#ID5L0JszTu6YpY}2;wF{urent0R z#f&1gmW$n~wpaA6NF01q&CHH$!aX`jZ#@mv*%14Q>l(6D*t7Lm(#hgd&^*8;_xswE zgS!%4eh=;KYXc-=Yd+>TD~;O0a5`JCgi36ERY`yJ`&9SMCA?8PT}I;wfrxDz^}UVnI#&7&N~H0R%;tS5E0eP< zX05r|(T@U-1U(aBRKxzPsvBR+DJbVlHHiSnVy4u2G^ME%&feE1%K)QMOb+k2pv$A4 zxW$Be=RrN^0@@a0!R+Psv$aWLg)`KACuIsJJW=AO@i!{~pjZI7ov-CQTp$m> z;6dz5t&nopll_MgSmjF91+?Ml}viW3Nq>H&n=qXmP=bMtht1nBRj{uMTW|?x%EJ<#CG~*N{E4bhc*Xnf>`E-ek=P z#G_PnZsa3O!!SoN%)1yI(bNp{)~R}!D^m401Lab%9^q5#kGWvbsd%UR5r&X5|FY#7 z`7GtR<9rF*(L~_L`^9nTl@5MXpQAfCp|IJM)Ezp$h4$m0c5L5V8-Lp!w(u{4me@~r zmvZ0ev0%iDv!#+m(Pj1)d8e$n^)>bta3f64o^nrjPublDoR}H?HgkU5u5{r#VbMvl zt$QrO&>VhM>a(LfO4p9BQ^Spas34E;gg;XEljkIWNZ(Q(k^qD?<*q+!Kz7yVxN$Nr zAll#O?k|5}w@%S`N`pssmSl(dpSqFwbKir*k68G8kHOS*kTqG z-6PKmMeH#TOF-6L8-a=pdcXT~d(h$F;W|cT_x{L*r!5^G^n_yVP2$nDo6Z=`5OCT8;Z(D+XFl1!{5D48f!1kCE7I|I)?fK+IFHNz$rt&`m4^%JsVp>~EoDw-$jIZ$AMm zed_-qf4DjOQ)w3~iR?rLk{?M>L=q(VlmB+n&YQAGZn-Sfz@rwc0&`PY{x;kHW7Ru7 zN&Ip{khw3vbsS-{P5t@FPI%La{B?&P2>XlPrlq$Fce!se!&&>}dY$-vWv(0P0Lk$(m?J?F$+EgGB@VVJTdwn&(=Pvp#M7;B8qolx*(*4}8)c&cD{CpJ zNCR%p52LB@`@sF}htc#dp8+ph!1FK#mxX{UBtCSb@#pAviI@z)W!jY#7U02e3ty#t9Q>Cz7XH;d2amx9AE+;;46^H>+K71s#%5SJWa@GeOK zION=w-+HAllXAZDdQZ+}CB&&SS09|qq?#h;t8ot~_YA z>MR0<&#d+{psW>~!=3{YHh7p}WeP;hPqO&K7O8TzL z{F&{cC+HBG$!ywbG;ZSSed~B*`Dai z*)Qs~iVa;z!JwxZ4PWwx!%x%ZRX88YDpSmaNJh0?__@ap(>P**k?@Lc z7-W|fZPAv(+yXhz9DXYHn9_@p6unqoZm&qT06;}s%&zk`V@j66l&~Rl`Fc)_GC@v* zRi;_bhSo8Li%nz6I-!>e=P=fM6&pohZ>4d|bzVK`;LcyMXftTn!j;Z3bV zXlEdA7au%@WU{m>lOA!3a?h8@5rEDsbN;|niv4wXGYgly{pF_`e;{59(_a8Xke6a) z%{@yjwlG(M9%fo8`%E=pb_L97N|5Jm0aImSlL_{k^0EYZ&UREyI=_j(>ihgAZ|Q%0 zEPJuBW?O99SmCkpj-2ma!=N%8(bG(SC0XHK#)_AYrgt82^Yy^ChQ&5h#>$H744J;) zAb%s|RML3KKtGB6F%{&74u=*%B6g~Of3RxhIZ&T~IO2B%m#nf!I!`MgKJmdz=dO%@+K&1J2R(deieg)hH{FP15C zn;!hu)H+=8S~JCDK}J&#CYzF@=F$|*%t-E*rk3fZ`XriyY(#zTpRL1Dmmyk@0k1ofVn+vzP3>OYfT72-DY8BiZ?zZ?~UKD=0=73p|fQy!t6%FEILzbz*ZGlAzJ*Hqv+q2^s!ENo_=7u?Byu&6g+GBzK@OO z4)Sf>yhUD$j6ZQ$8K0eOgwt1>T!C9*Xq04yd&nDXyPpPV=5fsj{e-m>Na^AHRzyR8 z^WGu{WEQ+e`gVsi&bfl8l|L|^VRnb{{4@J7<2mhz$8&eG5g8BY7~^UVkE-Z48gdDo=SB>qkTbwj(sag*xBYcLdYbLA%x4Q2m$&qb8Z zMgI}X_!gTa;hsA>|C*^~%$pk`2jAMB+6)AqRIxJ@+R^!L3IK`o&hF?sA1SyZnZu>I zA8u)5HFe+KxMKr#(wZF~npDoGohi)?V~*Uku+ymgZLvE-_X00zYzLE#f^2(FAuA+K z`*!J4@m=sHXZ6o!&6)*p*)Z|orm?&CeQth9a^#k8`{!?jA+`5naPRuZ!NcGjqu~sp zTsSm%eZ%0zw3V?&>YLlCw!LM4e`@BhjrnfNY|d|uEz_Nk4;3tpx~`IfreV!8*doc= zw{2k?jTX@8CgBn8kz7Aw@MUUpIUQio8AJFxv~gE#PJ6ghY|eLpIrQ{N)Z|dD(NqK^ z;{Fg`+DqhvcsSmcA9#C)^-$lI6D%VRg&|lM^&*;g3iRGQICq?OblgqvN8sp@%)r-t z*ny*c(g>=NNwjFK>`G;1%zFZ8xh_mksjS|Iu=12?b)v_h_#o@0XMy4>G1Ju2+=bvg z?2R9Q;Ij`pKyVx7j|hR$u!qlTVNQQCNlcT4`7h{w2pS|J5OmXqqP7P-JUjz$9|4N# z>{R`%BNVU0Ob|2G5;-Cs=zK61>he24mY@NTz+mbQ2ADN>`xf)oXgFPu3J;WvmI%|( z5dfNyQEtEfeh0MNPWkTxBb^Ib-Y3?W!5%Pa=K zBfuiwxu@UjfScQ7K;I))0yk%YIuAE097pCf=NAOv$c*dv><}hiD>0L%nr(sKIHeRu z3OZR0DcT=V}KF6O`f2Q(COQbJo5t6@)7VNJahlM9q@Ae595V<6|~RNW15K|h8&mm zMC>(g){@oGc%a07PsGmK%J3#yh z<$r*ZvVK7Ug|v55bGpF1v=foIN5S)B$j~~#SSIn>;vTx6uoxj5dG3Mu4ta8M`;f42 zD$kW48B*c7Q5_*|`e8_~hNs;neG<%3h8YkT5^&B1PUkdi6?9+lKq#;e}cU|>4M{Bu|%t7~$2coQm*RBShK zTjKwqQHqVBlaw{Ups}74f0|Q43(6lcSvmG{Q$AHv>##6H(;63R{jkNRr4_psKgMB; zos(AVXSAl4gK)#NQgf%X9c-n*KL6q*>>I^cl%=CrRcfpkR+nDO*YRDa{n5 zd^+by1Gww$R3y$9$h4P}DYe90>&&HpMY176);>a3`7SzBeeX)2>ia|T{g@Egc^gFRVQpJZ2lOIw| zUEfNKS+4{>lg5s$vL!D|+APncF%Y>KD0AA4u5_>4qkvPoQu4m10rb$30TOw>?#&JW zJ#tup-1*FCc$&{?@;qumlT!)+D-X+kPbQlO%Uf9`y3!!)=}R5MTYsluuvp1G?mTdf*p4rdR^Xg^Ws@<{T|wk_9`-<6+!2@z31> zTm}0%R1D|Hl}DIog?G-^A*HG+dxMcHIx_Ne-sr=Zr}2!Ks<%01bej?naM+>EPjtJ6!x;dp@Gw|CtcN)~CrmG4QE!<&%k1mqQB9y>k$w#T z2sA-y;m152cJ$Zp@CZAQP88vuBj()sodljseQ`T4G8%6YvPUv-oQ-jAPAyHK+j%#oMcMmEjMF@} zP2X+cocU=26~^#O6jek<;FWz@a4~$nPh>xOxpDem1Szt~Ezxj(} zL$;`8Us3I+)t2F=wL|MONQZX^YFbj6HK@O3F>LGd7|Te>re7ebRR+qFeVh$-*9Boz ziF_I={>Z$_vTA$gFC?UCGP;1{`ExbX*#~HO$vS~aq9!`#Qz%1X^U5o$$yXq421swS zA-#gT@#V&ws_lg!jp6wW#_kRc;U}I(HO}z=FwUVb07qVkcncY6<&j2s4oBLOBRpTy ztmRd3G0;to`c{ZM$-}c!GnSpJ02ZL!xDXYIsPmDPfuy{$>KYrX)*YV%2us#|ueh6E z<5Ty(80^q8kj+H~f*^(e!maYh{>?`kHnOWVZe0!UwnxOoCM>D_rq1eF#R}>D(CDXm zdzPys+ul|)AMqT_!g--QYSCWpRc_;`H@Vm^qv#UyVr_i%dwe>o6x zIm@Z(+&93vwJN*UMT91H%HLylqTt@q$7Ds$=Hwj|99kINpAkPw?JqC&o!{i^KD?;+ zqHo}elt5uuo-*fgE+O%r{H4v2FUYPz)E(*kRBTrrun4NRtMU|PrK=SPxJL5tS0#n1}x%81*UYZqryPOh#$m3scJR=wn z8ij=YY?!$x)n_oGyQw7i84!!z!cSwxUHq$Jv57(82Xo>kek9=snT758JiDl631o~f zoEqIf&uCb!8fa4W=0;uv$EWMa#t^Bx@VlKrXlb9lqZBvES!3FlAXW20g))@v0CPsb zygC?{M6)jatYRTwYJdLlL>2oySBOF^CLxaTT?zO|AzvK+m!v|jB6I8oHw#FOqot^7 zJ}>@=RgAabj+Q0ww)4$=CH{M>J`n#c%y%bAc;dIhVt~3-f!UkWt>)&_JLR{S<9uds z=UUn+lYIGg_z!Nw0Spy%K3AujPl6AR-z7+Ny1IwtlX^NA7G_^(`9J6r;0kvYwccb% z`NczfcMo?iEoxo(O%X>kUgUczgdH%ZXgfh$k|D#r&g21UhQybFqPGzk;<*?Qxi)cT z?ybL}yxO|NCtFVi752_zxTGs$!_o4<-jOhC1s@TcJS8^ArO!8VyNk%raf`lD{7-9s zgf+C@cOj$m1P6#@0#&4#DXA?p=lkNX3jD~^zpRMJ(+f(Ts_ec1O3(2p+~@B4+4&I| z^78%X!i!>?59B-`;8L7vlGGsP3kh+X2Xl&hN3>6bgDf1$tF-UA2_#%Nc}gtrEM#rY z9~9#tYZs@=LYh80oV+bi$1{&CZ~D@X8hhpW3N*Q%D7F2Lvi5Mw^g#Z@^!b5`$^2_d z*}L(~N>R41likm76KE4 zo+#SncSfuvCvkI{l6Z9nS0=uod~O>a(tifX9~}fN=w+$m_BgDruC9~4zPpa9>gz|A^@f^XUzq3_lng|v$#lOV<)e<+qyfm!6=0xQS= z35Ln2Z9mFs2@%xSXo^#-M{(2SJ?ANok3SHjK=pYa& z3qHjjv5fe$Ob_v=%W8k)(YtmGzwDbgs%l933*k##QmXS}mG-G8Dsr_-J|&^$b&fUP zZ$2Ti($*-j$$wUVE^N8TyvA={<1^2Sjt&4pHhHgSz&KE>ZS(*sSbG}(x?INDm`?Vt zE?$azNfcvU;y3>+E$VjSt2I4`FxAO2tc>YGZT42!QQUBo5sb-Y_OGdkep6|>n~eqs zLmO0(7j*?}2FMaC^(rr!bH*?Ujj=MLtz0T#G@Zqov1NZ}>nfj^Sdo)uigm9%ipnYJ znM%8+$uFY4gI;rE3p-2bcc|zFzp?BKX?&dAY7!bWG20p>u`1TtvAn-o!+qx1?D)}1 z{oLjgvB@YY+Q0B?DNkEt%o}{xf)Y5K4X-#_4#|BHbHSI~c6_CC9Cj@hXI=Lm2`+1(kU+{#Qj!fTUypken?lMO_=Hs^# zZFAaVT%GY)Hdtkq5?J^k^#jqhzIgvMdib6MlpNFd53ie1MXxV(6rRPBtUp!Iz5I}8 zk0O^0+ZR$rmdF6eWuB)u+a+>1Iizg+Y&oUa6^ z!~89}5ZS1(-`|h)uvh>$ns+tZT0MqgX7w-{cZ(;z)K`Lhm->vxE$Y!<@-!qRV~yX- zw?EwS-Ckn)ZbvnKJ=$4CsdW1>+T7pSfaoD$9M-$%goJ?2L(H9hd;y#iH%*5lhBzg; zJ|*mxbeg5jguSojTkON33|Dg04`!sqi*c{6{d>k_lGg1>+?7g>Nvt%u=!@{!>MD}@ z-iPx}y$uJ&pHaXuAX)O;8}Rw-_Y$@tvzp8p)pm>cVbxMf-;%24gNkZen$O;J{BiXx z=h+}lUTmCCTl(d8K4x63GS|!r+5O~a=|IyLvU~Hht+Ye{Z7Urif1+}G1og9P`-0Tk zJz_wOu}>EVjjZj}io2N4DNVn8e4^=A`Kg;;%}**MTbCio={e3Cd+T>r0;lV?%IyKk zR!&H^a>l}Q@X}0%G}FZ@K%ELDm6Xbul{@bCzQ8XbI2DTs6%G5tOOorXYbbZjC{}s|M~jhnL8}4xKEKsNia{m7BSX z^ZKV_E9kaD(iTVvl-JbMW!NVr+Uo(xKE^Kekur8yam6#F=(_PtN(gq@g>IHK@)}m! zO+;gO`Sis<1&N_$PSIE77a+fmKUVGIUOy5-L^v5RR zKdT&AVNXc;fDvhpDlvPTr6uN0?ZWcbcV1%NP;INn`qktI`1_y0W`7y&u1Iw)Nusn= zRM&N}@8DCdoV8YRQeD^0zS%7tgtafOeIxm)yZtjhMc3uIwv4Ne^?IpxN3viS`@-~s z%ny}m!gx;$PP5Mx`cWm*K(4oJQIiw-@G;Fpg8A{!z8nuOjBIRd8PWP(tyn1 zCoSw9iOhDj`a9oX*(eV9rd565)qA{3S+RNcZkTyqd=N8y)=j>mgE#dc>g%BSxwXJ& zzF_XmKOhq$xnt=uSUlWstaw$p8>2J>dr5ennb@4{v(B}0YcP1rIm@2GqQlvqO0s(*Q83! zkE~d*(oar2=Z|DF(@zszBGoE)1+7HkP$cppCUJwn-##4GEaKsaOXz#^p`QaaT{Iq1 zB8?|eCIvnShzP0MI#T?Vd4P%g(?|0|tU90Bmj8uVghtc7qJl7ggu+F0FoCA>0=yM~ zx1Y+lL8=PAK&yw9#U(LFDclk;H+c%nXj}`ZMPEe*#Y>TAZl7rfMd9VOaHG_fZ>i*$=!%F!ot$D+^SzXJ18pP~aor4>qIJ}%m5+}WJJajZ%QW%h;y6GX#@ z&jtfImOt981ce8*0hSU=vzMrH8&x{gh=?I7x%3+{paANzom-$8|535>_DiU#+LrX% z@50cG^COki0NsP#KKdBN?Lxl0_$qDy4W}Xn@~40|=lYytEmO6t!9Z_V_DRUPfUua;r;eHY4;K4RWTA@k{^)oK3f%=)$m|lI?N{nT&1ZMM>6iPz-yf7QpD2Ugk(Q<{8ExVvq*KKK2w zYY&$Cib%5bjroqTtXarB-N*6`iQWjC`9z|b|B8qC&76?20`2Dswdgfz4odmVDL(cA zt-fW}S51ZWWousq(HU16qT?^eSJ`)jrG!pP3iBsVhqepE)=LdVChB;TIp@akXRq7fL@~K{Z!DlJ+j4AE}AR z=gRGcK4N#k=EG(dc1_IpSAl?`WVSYLjb(SyVh<|&fvUc-n)bN%KXk+w4at$>AgiRq za*=goHolg|^TCezci@K2_=z_QX^YU~%ife;wI07uek$I5qn`o=`DxnB%QEV#`%Gqe_3t&jEjqTNNlmLlT;Lvgv z^dY1-U4JQlDBPOh2VsK>ySkSPuoIISDgHeD{oKU&W9WmE*042}{zZCSAmcF_)>2a# z&&?9%{FP_#JbE6?K3H@Rs4iGT6Yb$L>ssH$1!Jsk$Wab!?}UY!4lgarW>&V9{SV1f zoNcZx{H(IuC#3v*kzl8ZLFGT2c>Q+Cx{>E?N2S2;@hj_al@UMaG3(O`if;J01pLTx z-%}sn{weOVHwZhXUQQ5V=JR(cAX8p=r=q6Dd~4f>cDUHZplj)xrO7 zl%N5Eq@zXCzx1R9mByBuir^+?ym`0XIbH29%|eh=@J{;j^{}_ zq;Jf8gnHX2iijQN8&PbmyVYH_0i_JW)2#V&RdfT}KDhJ{&d8MLcbp1(L!OPoU1|Qp z{LQBCH%bypE#Gg7HqZT>Klz){QNG_qdc;Px7j4(K&8W6#tz`O2)3=rpUHIssc5_>} zg4^f2u2fk&h8MkVZm;ib`EJeJUes#(Zk=xzy}qbRtUA-I@u|+-iyTl@ntuPCzyRg-Giet(IY#bAu};hikm@P-zb`WN-I zdhp8}d@Ej#_*&YnG8d9V{36 zb+Z2|N=7_OTnmEy_;IT+w;GLNHYkcj43wX5eU}0`tLEdCxqjk;pLCFqXW%MgJ(Bx7 zU9Fr)ns#ja#Q5i0%rqhFfR_Ef=z9h#+qZOa0T3-+D*rZ(%xIdB8GRv3?i78FIVmF= z_h0|~#OqhdOy1!_&Gi{)z&3t4_sJ>iKf7oaDpFke-Om3A((zh>uD za#hM|>II`=l+3}|_|J`oVt%qFsB1vX|Q&JU_sG=RZm_hGHw0HjDqzeGj%FT zCV7mQ0QDdp{)z`t(bNMVe0KUht7iNIj8xidj~tDsSCT4K+1FE6EgF^DQ@4J>xjt*G zx*cW%qo8|PHP>cr{B1Gq2g>ZudS;SxtcO~CID*l8yR6bfWwnzbrs?o0kpE@u#zV1b z-8*3%p8S@wE#pQyJw`3QFe7rHcxcFI7z7!GsIl=UWKi?cPQpgAb~vu0b{cKc$lGt~ zMtUUX^HJlZfp2Lb@~ZPCa?FtGZtxhP7Dkxzg<=GiA`+aP5sPF3tuuh$Pz#zL@<0>^ zCS(p`xEbN@KtEVzh*tNFYvX+A8Aii$h=WECCi=I|P{)>U%HZ5`%_)mDvhqvae|R$43#2GN(j3d>LAla_Ls z;F)G5MMf;=ceIod8{S`Rpve>zH$O#+N3r&mL~__nxfqOn`N^{AEsxiYx{g2=LyuR}qF)#%Qv1vWw%lxlA`^Z?+BQ|U^0Se~Y>rRdh`+anO=UDllugi@M zdtNT8={%qH&W03uZ~53avD+J^ggU+;8~{vybLgEC;2wy6=5PFL(Q-(Pzr@^OEEjVg zEs%@PTvm1*onaBSJR@n&Dgy0|>U1zAdw7 zYO~6DN$~~ukJ>MF)%@S=?B<FO@3kdw*F*zgPWb%4}R;2q5btpN6U<1Lzq zHTFJKuT;6tC7^^^yK2Xy7xN*un%qDD^XS!8<6>~4G;Da9x!x^5(P)wh@EN?}8S(5{ zE8U67x8Nh|ILr4KGn*aVZMZYr#}##%onT0i$k4#Y2_8Rx2(Qt(xX`&H<8tRHDC*BFZU43T%;V(!6Zv5;^ z`P&9Egg=Z1HUdQCf?OL|S65?SJyQWlBOz;!U}LX`jfh-HA5v;vUqocCSM&rQ_a6Zt zQa;P_$!wp3llK!isjG31#4clK0FLf|M|i2}zN?6#0%KYov&>;mt3%y>FfqkRm4pDI(z*(DNxN;6y-y z>n~x(#iI1zCjhmwf6dJ!$CH3L{3E0`}#fstMxuL>q=_xfk<_MS}D=iSt_-gjpE4uoE&kVHF89@Ye$iw+t9;FhyiH>j_8hmVmxS@$@nR&cK;Q)rM+;FMKe?%)De4aiR<-8WcZK#>al+f5op~ z<2O8e(abU7pQo=c1PE9*2uA%f@sW?$9O*OXyN=ua>^6!~QDOf>JF2liQ;at?p_Kb% zbID=yAqjD($X`2RD1KLpym;ehu*|qP4L@geWoaVuEsi3^Y(|4v?(`8Xdhx#t2Z;0G z%~CPjOmc*bYgGPH(5KdU?lr|SN?E_*5*HRaA8f`?@#yn^z^6X)YpjqC#^`HC(dP^d z9IC~rv@Sr^3O>yP_{4Y$pWv~l;FUM57Hba4w15(b428&qe}qqyqGGj$=UXLcAfv&g z7xdJ%!0Z!UHIkduc_nJbYfv!>jETl~j@8sK8n5CL;eNm8!RUAd1M$dQCptgxdI1^ zsF%INMDP5Fr^*z)GbvQk@?KVj{rLkfD?rK|(K~}WQH^%30`yL$H94Vo0`ZfyJ^eq( zL%`EFx6&=QM)R-IHwy(OAqD$D^B5=Z?+yyVx-r-lL}U@jnU* zZfv7r84aci%6sxmIuMb{Q=~o4A1SznkCo0TjGyp-k`I|BjODv&SK+s-G%7<^@{z5i zZv-y&RQbpb&l9YH9n4;Lb5c#*N&D)1`ux}Gz^Q>M{)FMokKyMy#O3C3E5~?6G9*T{1^&TJ^xvL z2+uN3P^s=r`kq97z!L0xFRziBLWNNQ=LT9moXFnwmM{Uz)LEyOcVIZT!d|DjmDjQl zA{A4nBj*MEuEPGEE`|(ebLp`oyDOBfv>Ww%C+t@4XmOm9=v7i(eNVs3mR26unS<9L zSXmYGi95d5V&D%X8BEuu>CqLn{4ma^#QQNzA({fq*2@--JKczI4k;I zR`jDxw#mDFXAKN>+iVUaOMSLEPZ?bj3)8{A^z1uj+nv;=kDVj}5D1{^P#z#K9?Q8> zrNVxsyYeU;rc!b}0Cs)R8+@6o!BqPqU)V?n+&>S0kl)P8NL>#k{M|(1L&}``3q`x* zr~2hus|V?+sN&s5P-s^ArcPBWsu>A;e(Kc1W)*mz;C~hJa|(wr74OC$OP|lZ`BlzW z?S)&F)qn~XXDj}$6A-~d$${F?%JVof_t;q_W?m@zekWXGkP6%5F}{c+O~%6>FGTSZ z%E9d}?rwMZq!fB{7v9JIr4us7b$ex`9EQ6@g?%{ONl84^+18n{NijFFQJ)|5;vtXI z@d(UTI~K)L+LNeoiO}$N(Iptf>L)OmL!)C0+n?zKRPnnK^4+G-mB!mduB+{y-?(@& z8izq3VJI=7lqD&AC^PD8CD)PJFnWc3_q|Go3R$h;fp|^2DRMP#M1|(?od2%xxg(b@ zP}X~mMhQuF&K3xm&-DtUuS}GpJ4OfE@Uo>&+&}_v1gh<$G&D@HABf0yl)AsG1DeQs za(`CnT@Iadt&wUYkxueBIe4+U19pr^uN~UL!>xnQ?vPHd@mA|Fs-A?AgZT9G`} zPr&tNs}B*ZVh?Bk!peyjZHf5DT>n*JGhTwCjeVM#yf0u!YN6zcW)lUk) z%s>iQ^xc9n=4<8~E&u7vG9~(P|DsRo&s&Xbw42{M;oB74pgtH0nTBhf< z5O~ySXH^F|Suow))^f13`FixD{zcP$^<*6gmnj}Fza$wyo~oRX`NqD@&LMpO>Zm8jsnE(9sC??XXR8wWUvH-?BLGi>HU7>+)`;5=6}5zqEqYP$>C`dW z5$5?LHb?EOcuqn;rpbevONute*C*#QR~}@V$OEp0c=$k4Y+n0Pl}ehR*< z2Ib-1TN3b*7(h@M>ArVVz(-nFCOgU5IvVP)Yx}|7s)sk&Ra@(<*!BzuMcU za@t;=%!W_{t=8CMc{zpi^zO^WysWg(;U(;EEepw>f!d3jL*NJrG)c-WmiHXln!E^K z4TzjA^Xnjf({r_b!>6v~U~@N}CvV@|oxk~+AnoS2R>7PLLkp2F!lRXZk=TmaL0XKq z2HS>ZQ%o`*sJjSqC6hq_0LmD`Vm?&a3zPL)q@E-Pm&+Cw_+m3Nttwy9&Nh8=$yna1 z+YPfnZe8jpj$#GJ0i~y~z#i4L!mn}0Ip_Vaccmmiz-jYSw6ohe~$lR{S*D=GwQ3q!LpVBTbJgF()=2E;CDk1c%e zkhD0`9OL8Vg5p|EaF7&WBk?@7064%{aSm1KXn~OAUh2v|$Z_&+QfM}zb^;YG50vm4 zwb^Yf?*V>kbQ<+@c`F+m(&H6YZZSi#<{aYXB;nie<;2+T+(*_i*%cLB{}7 zzb>5$p?!ygFb zuD^AfSaaTvRUmdf8k(wMIok6rCtB`MQR}F*bCDC-eX%ahE(35vio>CW)gJHKwfF;o zglCCzXIaEndna?UxEW@9UGl4lB{bY*y67LULG1MEVXTp%wNgFDUs^%7*ir zYnQAO?SSr(ki(T?Mmgo;7FY1>f0h7UoD3fygi#ApAUS7)h zpVi8QZKtD!tCxkuwJEYV16X7?s#?K>4msB*%C1=y$Zi28p+K05^NB14l)I@~WnX=a zYFSL)zSQNtG>IaC4YOTEPzIX0FuK82BzK4IrAVbcC|UI#lqTQRUKrt~W#UmV64r7kaCR9^21^c$9zCFcQ@rjGFlLdC#8D^bF z6|r?=Bknm=Ji(7G16<2^fjAZzD|QNQpH(GLE1zcZ#M%30>P30%4h3(89ZReynw}Gc zGP)`XDNuS<0GszgXW6P60T)5PNTjXH8w7KFoTf`&E!AOJJ?vx zU5b+VWwE@#%0tV9_$h>Fl0Uh{cWcpC3TT8o4zMkCBO z3U^uM`_V+iw$szlc#dm;tMw}SD2P6iRo+mgkA@`lQ7-x@r_(CcIx1;IvHm&lUz8nD zMiTY*BZgK`v{4G{q;!SVTPrM0J*~4oNb0O2+E3A0 zff@jp{*r3zQ5~xcNa<=zG=nx5D(&el?d@tu6Zv-L9z=tne>T#LtAFaQLiyO6gfD`g zqVl3)m77Z%(*{Y69iTDV!H4rBT8fqRcjO3LY;=}2T}|Kj2*)tlqds3WH^*wSNY}kw z(%j2SPI0vPme4b{|Jp@g^A$}TX8XlCb-w~Dp+NuKOrXM60k}mrg_wM*P_|)?j6_S?NVq~Y0b!S=$?742P}N9E@#t^7Yzkx<}H%#V!030GrpBKZ`sC?q-sloX73y8dr#)kk=Jwj=4Sge|M+OwnAyEAlAt ziab`@MFiPdH7>m(RIJv7Nu1_=@*=E&B>iU*54= z8$$%Ba}B=}CbihpQF1iR4h=!$DkU6e9!# z5+EN)l;ueux}9-(CauT^e7zObcDLjQ*+y2^Nd(c)BD{nj#npN&D zKJju&Pmw;5Vl@0sgTwOnLvrn@mka9E@xREgSmxB5`vKqts+FODIIMAzhj4X`n4p2t z5Fz+8oC9DX;eD2coVy-VkagOdkr`Q{yWf@Vb}zT(0;;_+@#3UW@te(W8T=+^KD^D* z)EfKl1hkD;^A25bC4bBv=2zy+>Wb

v_FCS|;eZTiM@xO9LEn)+yikJAf}4kzOb< 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�z=(6g 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

x8tM1U&vI%f85M-kh&a< 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

g+4142`ZBK&MBTSqQdq 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Ν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�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 )kSS+GRQxR@w-N8N-a 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 z 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%0D1655rzvs+Oyh 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

vzSOwr$JF8yftkS%wjj+^HfQnO2H+? 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 diff --git a/mingwm10.dll b/mingwm10.dll deleted file mode 100644 index bf8544ce13ad82018a4aa13abc1594297e4d891b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 11673 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 x4(UFf=#(JFDkOW?G&iQ|nyfl8)l*8X1Z) Date: Thu, 24 Sep 2009 04:09:56 +0000 Subject: [PATCH 004/133] tray icon + ask before closing git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@10 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- bugs.txt | 4 + changelog.txt | 4 + db.cpp | 5 + headers.h | 8 +- main.cpp | 3 - main.h | 1 - makefile | 4 +- ui.cpp | 229 +++- ui.h | 61 +- uibase.cpp | 3467 +++++++++++++++++++++++++------------------------ uibase.h | 1196 +++++++++-------- uiproject.fbp | 825 ++++++++++-- 12 files changed, 3362 insertions(+), 2445 deletions(-) create mode 100644 bugs.txt diff --git a/bugs.txt b/bugs.txt new file mode 100644 index 00000000..348c359e --- /dev/null +++ b/bugs.txt @@ -0,0 +1,4 @@ +Known bugs: +- For some reason, CreateHardLink doesn't add a shortcut to the startup folder +- When the program is minimized to tray, double clicking the icon only restores it to the task bar +- Window flickers when blocks are added (problem with repainting?) \ No newline at end of file diff --git a/changelog.txt b/changelog.txt index 19120c3e..685eb629 100644 --- a/changelog.txt +++ b/changelog.txt @@ -1,2 +1,6 @@ Changes after 0.1.5: -------------------- ++ Options dialog layout changed - added the UI options panel ++ Minimize to tray feature ++ Startup on system boot feature ++ Ask before closing \ No newline at end of file diff --git a/db.cpp b/db.cpp index f38f3ffa..6607ff59 100644 --- a/db.cpp +++ b/db.cpp @@ -575,6 +575,11 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) if (strKey == "fGenerateBitcoins") ssValue >> fGenerateBitcoins; if (strKey == "nTransactionFee") ssValue >> nTransactionFee; if (strKey == "addrIncoming") ssValue >> addrIncoming; + if (strKey == "minimizeToTray") ssValue >> minimizeToTray; + if (strKey == "closeToTray") ssValue >> closeToTray; + if (strKey == "startOnSysBoot") ssValue >> startOnSysBoot; + if (strKey == "askBeforeClosing") ssValue >> askBeforeClosing; + if (strKey == "alwaysShowTrayIcon") ssValue >> alwaysShowTrayIcon; } } } diff --git a/headers.h b/headers.h index 7bd68a19..92911505 100644 --- a/headers.h +++ b/headers.h @@ -10,11 +10,16 @@ #ifdef _WIN32_WINNT #undef _WIN32_WINNT #endif -#define _WIN32_WINNT 0x0400 +#define _WIN32_WINNT 0x0500 +#ifdef _WIN32_IE +#undef _WIN32_IE +#endif +#define _WIN32_IE 0x0500 #define WIN32_LEAN_AND_MEAN 1 #include #include #include +#include #include #include #include @@ -32,6 +37,7 @@ #include #include #include +#include #include #define BOUNDSCHECK 1 #include diff --git a/main.cpp b/main.cpp index 97000db3..ebf9d727 100644 --- a/main.cpp +++ b/main.cpp @@ -54,9 +54,6 @@ CAddress addrIncoming; - - - ////////////////////////////////////////////////////////////////////////////// // // mapKeys diff --git a/main.h b/main.h index 3432b316..9dd29bb6 100644 --- a/main.h +++ b/main.h @@ -47,7 +47,6 @@ extern CAddress addrIncoming; - string GetAppDir(); bool CheckDiskSpace(int64 nAdditionalBytes=0); FILE* OpenBlockFile(unsigned int nFile, unsigned int nBlockPos, const char* pszMode="rb"); diff --git a/makefile b/makefile index 534eb521..221684e8 100644 --- a/makefile +++ b/makefile @@ -12,7 +12,7 @@ ifeq "$(BUILD)" "debug" D=d # note: gcc 3.x profile doesn't work #DEBUGFLAGS=-O0 -g -pg -D__WXDEBUG__ -DEBUGFLAGS=-g -D__WXDEBUG__ +DEBUGFLAGS=-g -D__WXDEBUG__ -Wall -Wextra endif @@ -22,7 +22,7 @@ LIBPATHS=-L"/DB/build_unix" -L"/OpenSSL/out" -L"/wxWidgets/lib/gcc_lib" LIBS= \ -l db_cxx \ -l eay32 \ - -l wxmsw28$(D)_richtext -l wxmsw28$(D)_html -l wxmsw28$(D)_core -l wxbase28$(D) -l wxtiff$(D) -l wxjpeg$(D) -l wxpng$(D) -l wxzlib$(D) -l wxregex$(D) -l wxexpat$(D) \ + -l wxmsw28$(D)_richtext -l wxmsw28$(D)_html -l wxmsw28$(D)_core -l wxmsw28$(D)_adv -l wxbase28$(D) -l wxtiff$(D) -l wxjpeg$(D) -l wxpng$(D) -l wxzlib$(D) -l wxregex$(D) -l wxexpat$(D) \ -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) diff --git a/ui.cpp b/ui.cpp index 96c5a83d..751a50be 100644 --- a/ui.cpp +++ b/ui.cpp @@ -7,8 +7,6 @@ #include #endif - - DEFINE_EVENT_TYPE(wxEVT_CROSSTHREADCALL) DEFINE_EVENT_TYPE(wxEVT_REPLY1) DEFINE_EVENT_TYPE(wxEVT_REPLY2) @@ -19,6 +17,7 @@ DEFINE_EVENT_TYPE(wxEVT_TABLEDELETED) CMainFrame* pframeMain = NULL; map mapAddressBook; +CBitcoinTBIcon* taskBarIcon = NULL; // Tray icon void ThreadRequestProductDetails(void* parg); @@ -27,8 +26,12 @@ bool fRandSendTest = false; void RandSend(); extern int g_isPainting; - - +// UI settings and their default values +int minimizeToTray = 1; +int closeToTray = 1; +int startOnSysBoot = 1; +int askBeforeClosing = 1; +int alwaysShowTrayIcon = 1; @@ -359,8 +362,28 @@ void Shutdown(void* parg) void CMainFrame::OnClose(wxCloseEvent& event) { - Destroy(); - _beginthread(Shutdown, 0, NULL); + if (closeToTray && event.CanVeto()) { + event.Veto(); + SendToTray(); + } + else if (!event.CanVeto() || !askBeforeClosing || wxMessageBox("Quit program?", "Confirm", wxYES_NO, this) == wxYES) { + delete taskBarIcon; + Destroy(); + _beginthread(Shutdown, 0, NULL); + } +} + +void CMainFrame::OnIconize(wxIconizeEvent& event) +{ + if (minimizeToTray) { + SendToTray(); + } +} + +void CMainFrame::SendToTray() +{ + Hide(); + taskBarIcon->Show(); } void CMainFrame::OnMouseEvents(wxMouseEvent& event) @@ -836,16 +859,22 @@ void CMainFrame::OnMenuFileExit(wxCommandEvent& event) Close(true); } -void CMainFrame::OnMenuOptionsGenerate(wxCommandEvent& event) +void GenerateBitcoins(bool flag) { - fGenerateBitcoins = event.IsChecked(); + fGenerateBitcoins = flag; nTransactionsUpdated++; CWalletDB().WriteSetting("fGenerateBitcoins", fGenerateBitcoins); - if (fGenerateBitcoins) if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); + taskBarIcon->UpdateTooltip(); +} + +void CMainFrame::OnMenuOptionsGenerate(wxCommandEvent& event) +{ + GenerateBitcoins(event.IsChecked()); + Refresh(); wxPaintEvent eventPaint; AddPendingEvent(eventPaint); @@ -868,6 +897,10 @@ void CMainFrame::OnMenuHelpAbout(wxCommandEvent& event) dialog.ShowModal(); } +void CMainFrame::OnUpdateMenuGenerate( wxUpdateUIEvent& event ) { + event.Check(fGenerateBitcoins); +} + void CMainFrame::OnButtonSend(wxCommandEvent& event) { /// debug test @@ -1231,23 +1264,57 @@ void CTxDetailsDialog::OnButtonOK(wxCommandEvent& event) COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) { - m_textCtrlTransactionFee->SetValue(FormatMoney(nTransactionFee)); m_buttonOK->SetFocus(); + m_treeCtrl->AddRoot(wxT("Settings")); + m_treeCtrl->AppendItem(m_treeCtrl->GetRootItem(), wxT("Bitcoin")); + m_treeCtrl->AppendItem(m_treeCtrl->GetRootItem(), wxT("UI")); + + panelUI = new COptionsPanelUI(this); + panelBitcoin = new COptionsPanelBitcoin(this); + currentPanel = panelBitcoin; + + panelSizer->Add(panelUI); + panelSizer->Hide(panelUI); + panelSizer->Add(panelBitcoin); + panelSizer->Layout(); + } -void COptionsDialog::OnKillFocusTransactionFee(wxFocusEvent& event) +void COptionsDialog::MenuSelChanged( wxTreeEvent& event ) { - int64 nTmp = nTransactionFee; - ParseMoney(m_textCtrlTransactionFee->GetValue(), nTmp); - m_textCtrlTransactionFee->SetValue(FormatMoney(nTmp)); + panelSizer->Hide(currentPanel); + wxString text = m_treeCtrl->GetItemText(event.GetItem()); + if (text == "Bitcoin") { + panelSizer->Show(panelBitcoin); + currentPanel = panelBitcoin; + } + else { + panelSizer->Show(panelUI); + currentPanel = panelUI; + } + panelSizer->Layout(); } void COptionsDialog::OnButtonOK(wxCommandEvent& event) { // nTransactionFee int64 nPrevTransactionFee = nTransactionFee; - if (ParseMoney(m_textCtrlTransactionFee->GetValue(), nTransactionFee) && nTransactionFee != nPrevTransactionFee) - CWalletDB().WriteSetting("nTransactionFee", nTransactionFee); + if (ParseMoney(panelBitcoin->m_textCtrlTransactionFee->GetValue(), nTransactionFee) && nTransactionFee != nPrevTransactionFee) + CWalletDB().WriteSetting("transactionFee", nTransactionFee); + + minimizeToTray = panelUI->m_checkMinToTray->IsChecked(); + closeToTray = panelUI->m_checkCloseToTray->IsChecked(); + startOnSysBoot = panelUI->m_checkStartOnSysBoot->IsChecked(); + askBeforeClosing = panelUI->m_checkAskBeforeClosing->IsChecked(); + alwaysShowTrayIcon = panelUI->m_checkAlwaysShowTray->IsChecked(); + + CWalletDB().WriteSetting("minimizeToTray", minimizeToTray); + CWalletDB().WriteSetting("closeToTray", closeToTray); + CWalletDB().WriteSetting("startOnSysBoot", startOnSysBoot); + CWalletDB().WriteSetting("askBeforeClosing", askBeforeClosing); + CWalletDB().WriteSetting("alwaysShowTrayIcon", alwaysShowTrayIcon); + + ApplyUISettings(); Close(); } @@ -1259,6 +1326,39 @@ void COptionsDialog::OnButtonCancel(wxCommandEvent& event) +////////////////////////////////////////////////////////////////////////////// +// +// COptionsPanelBitcoin +// + +COptionsPanelBitcoin::COptionsPanelBitcoin(wxWindow* parent) : COptionsPanelBitcoinBase(parent) +{ + m_textCtrlTransactionFee->SetValue(FormatMoney(nTransactionFee)); +} + +void COptionsPanelBitcoin::OnKillFocusTransactionFee(wxFocusEvent& event) +{ + int64 nTmp = nTransactionFee; + ParseMoney(m_textCtrlTransactionFee->GetValue(), nTmp); + m_textCtrlTransactionFee->SetValue(FormatMoney(nTmp)); +} + + +////////////////////////////////////////////////////////////////////////////// +// +// COptionsPanelUI +// + +COptionsPanelUI::COptionsPanelUI(wxWindow* parent) : COptionsPanelUIBase(parent) +{ + m_checkMinToTray->SetValue(minimizeToTray); + m_checkCloseToTray->SetValue(closeToTray); + m_checkStartOnSysBoot->SetValue(startOnSysBoot); + m_checkAskBeforeClosing->SetValue(askBeforeClosing); + m_checkAlwaysShowTray->SetValue(alwaysShowTrayIcon); +} + + @@ -2862,10 +2962,79 @@ void CEditReviewDialog::GetReview(CReview& review) +////////////////////////////////////////////////////////////////////////////// +// +// BitcoinTBIcon +// + +enum { + PU_RESTORE = 10001, + PU_GENERATE, + PU_EXIT, +}; +BEGIN_EVENT_TABLE(CBitcoinTBIcon, wxTaskBarIcon) + EVT_TASKBAR_LEFT_DCLICK (CBitcoinTBIcon::OnLeftButtonDClick) + EVT_MENU(PU_RESTORE, CBitcoinTBIcon::OnMenuRestore) + EVT_MENU(PU_GENERATE, CBitcoinTBIcon::OnMenuGenerate) + EVT_MENU(PU_EXIT, CBitcoinTBIcon::OnMenuExit) +END_EVENT_TABLE() +void CBitcoinTBIcon::Show() +{ + string tooltip = "Bitcoin"; + tooltip += fGenerateBitcoins ? " - Generating" : ""; + SetIcon(wxICON(bitcoin), tooltip); +} +void CBitcoinTBIcon::Hide() +{ + RemoveIcon(); +} +void CBitcoinTBIcon::OnLeftButtonDClick(wxTaskBarIconEvent&) +{ + Restore(); +} + +void CBitcoinTBIcon::OnMenuExit(wxCommandEvent&) +{ + pframeMain->Close(true); +} + +void CBitcoinTBIcon::OnMenuGenerate(wxCommandEvent& event) +{ + GenerateBitcoins(event.IsChecked()); + pframeMain->Refresh(); +} + +void CBitcoinTBIcon::OnMenuRestore(wxCommandEvent&) { + Restore(); +} + +void CBitcoinTBIcon::Restore() { + pframeMain->Show(); + pframeMain->Raise(); + if (!alwaysShowTrayIcon) + Hide(); +} + +void CBitcoinTBIcon::UpdateTooltip() { + if (IsIconInstalled()) + Show(); +} + +wxMenu *CBitcoinTBIcon::CreatePopupMenu() +{ + wxMenu *menu = new wxMenu; + wxMenuItem* generateCheck = menu->AppendCheckItem(PU_GENERATE, _T("Generate Coins")); + menu->Append(PU_RESTORE, _T("Open Bitcoin")); + menu->Append(PU_EXIT, _T("Exit")); + + generateCheck->Check(fGenerateBitcoins); + + return menu; +} @@ -3137,6 +3306,9 @@ bool CMyApp::OnInit2() } } + taskBarIcon = new CBitcoinTBIcon(); + ApplyUISettings(); + return true; } @@ -3214,6 +3386,31 @@ void MainFrameRepaint() +void ApplyUISettings() { + // Show the tray icon? + if (alwaysShowTrayIcon) + taskBarIcon->Show(); + else + taskBarIcon->Hide(); + + // Autostart on system startup? + if (startOnSysBoot) { + // Get the startup folder path + char targetPath[ MAX_PATH ]; + SHGetSpecialFolderPath(0, targetPath, CSIDL_STARTUP, 0); + strcat(targetPath, "\\bitcoin.lnk"); + + // And the current executable path + char currentPath[ MAX_PATH ]; + GetModuleFileName(NULL, currentPath, _MAX_PATH + 1); + + // Create the shortcut + CreateHardLink(targetPath, currentPath, NULL); + } +} + + + diff --git a/ui.h b/ui.h index 163554a5..5f3897c1 100644 --- a/ui.h +++ b/ui.h @@ -27,9 +27,14 @@ extern string FormatTxStatus(const CWalletTx& wtx); extern void CrossThreadCall(int nID, void* pdata); extern void MainFrameRepaint(); extern void Shutdown(void* parg); +void ApplyUISettings(); - - +// UI settings +extern int minimizeToTray; +extern int closeToTray; +extern int startOnSysBoot; +extern int askBeforeClosing; +extern int alwaysShowTrayIcon; @@ -38,6 +43,7 @@ class CMainFrame : public CMainFrameBase protected: // Event handlers void OnClose(wxCloseEvent& event); + void OnIconize( wxIconizeEvent& event ); void OnMouseEvents(wxMouseEvent& event); void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } void OnIdle(wxIdleEvent& event); @@ -59,6 +65,7 @@ protected: void OnListItemActivatedProductsSent(wxListEvent& event); void OnListItemActivatedOrdersSent(wxListEvent& event); void OnListItemActivatedOrdersReceived(wxListEvent& event); + void OnUpdateMenuGenerate( wxUpdateUIEvent& event ); public: /** Constructor */ @@ -77,6 +84,7 @@ public: void InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); void RefreshListCtrl(); void RefreshStatus(); + void SendToTray(); }; @@ -98,14 +106,44 @@ public: +class COptionsPanelBitcoin : public COptionsPanelBitcoinBase +{ +protected: + // Event handlers + void OnKillFocusTransactionFee( wxFocusEvent& event ); + +public: + /** Constructor */ + COptionsPanelBitcoin(wxWindow* parent); +}; + + + +class COptionsPanelUI : public COptionsPanelUIBase +{ +protected: + // Event handlers + void OnOptionsChanged( wxCommandEvent& event ); + +public: + /** Constructor */ + COptionsPanelUI(wxWindow* parent); +}; + + + class COptionsDialog : public COptionsDialogBase { protected: // Event handlers - void OnKillFocusTransactionFee(wxFocusEvent& event); + void MenuSelChanged( wxTreeEvent& event ); void OnButtonOK(wxCommandEvent& event); void OnButtonCancel(wxCommandEvent& event); + // Panels + COptionsPanelBitcoin* panelBitcoin; + COptionsPanelUI* panelUI; + wxPanel* currentPanel; public: /** Constructor */ COptionsDialog(wxWindow* parent); @@ -416,5 +454,22 @@ public: +class CBitcoinTBIcon : public wxTaskBarIcon +{ +protected: + void Restore(); + + // Event handlers + void OnLeftButtonDClick(wxTaskBarIconEvent&); + void OnMenuExit(wxCommandEvent&); + void OnMenuGenerate(wxCommandEvent&); + void OnMenuRestore(wxCommandEvent&); +public: + void Show(); + void Hide(); + void UpdateTooltip(); + virtual wxMenu *CreatePopupMenu(); +DECLARE_EVENT_TABLE() +}; diff --git a/uibase.cpp b/uibase.cpp index 2972e9af..dcaa6975 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -1,7 +1,3 @@ -// Copyright (c) 2009 Satoshi Nakamoto -// Distributed under the MIT/X11 software license, see the accompanying -// file license.txt or http://www.opensource.org/licenses/mit-license.php. - /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -13,1813 +9,1884 @@ /////////////////////////////////////////////////////////////////////////// -CMainFrameBase::CMainFrameBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE)); - - m_menubar = new wxMenuBar(0); - m_menubar->SetBackgroundColour(wxColour(240, 240, 240)); - - m_menuFile = new wxMenu(); - wxMenuItem* m_menuFileExit; - m_menuFileExit = new wxMenuItem(m_menuFile, wxID_ANY, wxString(wxT("E&xit")) , wxEmptyString, wxITEM_NORMAL); - m_menuFile->Append(m_menuFileExit); - - m_menubar->Append(m_menuFile, wxT("&File")); - - m_menuOptions = new wxMenu(); - wxMenuItem* m_menuOptionsGenerateBitcoins; - m_menuOptionsGenerateBitcoins = new wxMenuItem(m_menuOptions, wxID_OPTIONSGENERATEBITCOINS, wxString(wxT("&Generate Coins")) , wxEmptyString, wxITEM_CHECK); - m_menuOptions->Append(m_menuOptionsGenerateBitcoins); - - wxMenuItem* m_menuChangeYourAddress; - m_menuChangeYourAddress = new wxMenuItem(m_menuOptions, wxID_ANY, wxString(wxT("&Change Your Address...")) , wxEmptyString, wxITEM_NORMAL); - m_menuOptions->Append(m_menuChangeYourAddress); - - wxMenuItem* m_menuOptionsOptions; - m_menuOptionsOptions = new wxMenuItem(m_menuOptions, wxID_ANY, wxString(wxT("&Options...")) , wxEmptyString, wxITEM_NORMAL); - m_menuOptions->Append(m_menuOptionsOptions); - - m_menubar->Append(m_menuOptions, wxT("&Options")); - - m_menuHelp = new wxMenu(); - wxMenuItem* m_menuHelpAbout; - m_menuHelpAbout = new wxMenuItem(m_menuHelp, wxID_ANY, wxString(wxT("&About...")) , wxEmptyString, wxITEM_NORMAL); - m_menuHelp->Append(m_menuHelpAbout); - - m_menubar->Append(m_menuHelp, wxT("&Help")); - - this->SetMenuBar(m_menubar); - - m_toolBar = this->CreateToolBar(wxTB_FLAT|wxTB_HORZ_TEXT, wxID_ANY); - m_toolBar->SetToolBitmapSize(wxSize(20,20)); - m_toolBar->SetToolSeparation(1); - m_toolBar->SetFont(wxFont(wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString)); - - m_toolBar->AddTool(wxID_BUTTONSEND, wxT("&Send Coins"), wxBitmap(wxT("send20"), wxBITMAP_TYPE_RESOURCE), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); - m_toolBar->AddTool(wxID_BUTTONRECEIVE, wxT("&Address Book"), wxBitmap(wxT("addressbook20"), wxBITMAP_TYPE_RESOURCE), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); - m_toolBar->Realize(); - - m_statusBar = this->CreateStatusBar(1, wxST_SIZEGRIP, wxID_ANY); - m_statusBar->SetBackgroundColour(wxColour(240, 240, 240)); - - wxBoxSizer* bSizer2; - bSizer2 = new wxBoxSizer(wxVERTICAL); - - - bSizer2->Add(0, 2, 0, wxEXPAND, 5); - - wxBoxSizer* bSizer85; - bSizer85 = new wxBoxSizer(wxHORIZONTAL); - - m_staticText32 = new wxStaticText(this, wxID_ANY, wxT("Your Bitcoin Address:"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText32->Wrap(-1); - bSizer85->Add(m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxLEFT, 5); - - m_textCtrlAddress = new wxTextCtrl(this, wxID_TEXTCTRLADDRESS, wxEmptyString, wxDefaultPosition, wxSize(250,-1), wxTE_READONLY); - m_textCtrlAddress->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); - - bSizer85->Add(m_textCtrlAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_buttonCopy = new wxButton(this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); - bSizer85->Add(m_buttonCopy, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5); - - m_button91 = new wxButton(this, wxID_BUTTONCHANGE, wxT("C&hange..."), wxDefaultPosition, wxDefaultSize, 0); - m_button91->Hide(); - - bSizer85->Add(m_button91, 0, wxRIGHT, 5); - - - bSizer85->Add(0, 0, 0, wxEXPAND, 5); - - bSizer2->Add(bSizer85, 0, wxEXPAND|wxRIGHT|wxLEFT, 5); - - wxBoxSizer* bSizer3; - bSizer3 = new wxBoxSizer(wxHORIZONTAL); - - m_panel14 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); - wxBoxSizer* bSizer66; - bSizer66 = new wxBoxSizer(wxHORIZONTAL); - - m_staticText41 = new wxStaticText(m_panel14, wxID_ANY, wxT("Balance:"), wxDefaultPosition, wxSize(-1,15), 0); - m_staticText41->Wrap(-1); - bSizer66->Add(m_staticText41, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5); - - m_staticTextBalance = new wxStaticText(m_panel14, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(120,15), wxALIGN_RIGHT|wxST_NO_AUTORESIZE); - m_staticTextBalance->Wrap(-1); - m_staticTextBalance->SetFont(wxFont(8, 70, 90, 90, false, wxEmptyString)); - m_staticTextBalance->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); - - bSizer66->Add(m_staticTextBalance, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); - - m_panel14->SetSizer(bSizer66); - m_panel14->Layout(); - bSizer66->Fit(m_panel14); - bSizer3->Add(m_panel14, 1, wxEXPAND|wxALIGN_BOTTOM|wxALL, 5); - - - bSizer3->Add(0, 0, 0, wxEXPAND, 5); - - wxString m_choiceFilterChoices[] = { wxT(" All"), wxT(" Sent"), wxT(" Received"), wxT(" In Progress") }; - int m_choiceFilterNChoices = sizeof(m_choiceFilterChoices) / sizeof(wxString); - m_choiceFilter = new wxChoice(this, wxID_ANY, wxDefaultPosition, wxSize(110,-1), m_choiceFilterNChoices, m_choiceFilterChoices, 0); - m_choiceFilter->SetSelection(0); - m_choiceFilter->Hide(); - - bSizer3->Add(m_choiceFilter, 0, wxALIGN_BOTTOM|wxTOP|wxRIGHT|wxLEFT, 5); - - bSizer2->Add(bSizer3, 0, wxEXPAND, 5); - - m_notebook = new wxNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0); - m_panel7 = new wxPanel(m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); - wxBoxSizer* bSizer157; - bSizer157 = new wxBoxSizer(wxVERTICAL); - - m_listCtrl = new wxListCtrl(m_panel7, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxALWAYS_SHOW_SB); - bSizer157->Add(m_listCtrl, 1, wxEXPAND|wxALL, 5); - - m_panel7->SetSizer(bSizer157); - m_panel7->Layout(); - bSizer157->Fit(m_panel7); - m_notebook->AddPage(m_panel7, wxT("All Transactions"), false); - - bSizer2->Add(m_notebook, 1, wxEXPAND, 5); - - wxBoxSizer* bSizer_TabsForFutureUse; - bSizer_TabsForFutureUse = new wxBoxSizer(wxVERTICAL); - - m_panel9 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); - m_panel9->Hide(); - - wxBoxSizer* bSizer159; - bSizer159 = new wxBoxSizer(wxVERTICAL); - - m_listCtrlEscrows = new wxListCtrl(m_panel9, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); - bSizer159->Add(m_listCtrlEscrows, 1, wxALL|wxEXPAND, 5); - - m_panel9->SetSizer(bSizer159); - m_panel9->Layout(); - bSizer159->Fit(m_panel9); - bSizer_TabsForFutureUse->Add(m_panel9, 1, wxEXPAND | wxALL, 5); - - m_panel8 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); - m_panel8->Hide(); - - wxBoxSizer* bSizer158; - bSizer158 = new wxBoxSizer(wxVERTICAL); - - m_listCtrlOrdersSent = new wxListCtrl(m_panel8, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); - bSizer158->Add(m_listCtrlOrdersSent, 1, wxALL|wxEXPAND, 5); - - m_panel8->SetSizer(bSizer158); - m_panel8->Layout(); - bSizer158->Fit(m_panel8); - bSizer_TabsForFutureUse->Add(m_panel8, 1, wxEXPAND | wxALL, 5); - - m_panel10 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); - m_panel10->Hide(); - - wxBoxSizer* bSizer160; - bSizer160 = new wxBoxSizer(wxVERTICAL); - - m_listCtrlProductsSent = new wxListCtrl(m_panel10, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); - bSizer160->Add(m_listCtrlProductsSent, 1, wxALL|wxEXPAND, 5); - - m_panel10->SetSizer(bSizer160); - m_panel10->Layout(); - bSizer160->Fit(m_panel10); - bSizer_TabsForFutureUse->Add(m_panel10, 1, wxEXPAND | wxALL, 5); - - m_panel11 = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL); - m_panel11->Hide(); - - wxBoxSizer* bSizer161; - bSizer161 = new wxBoxSizer(wxVERTICAL); - - m_listCtrlOrdersReceived = new wxListCtrl(m_panel11, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); - bSizer161->Add(m_listCtrlOrdersReceived, 1, wxALL|wxEXPAND, 5); - - m_panel11->SetSizer(bSizer161); - m_panel11->Layout(); - bSizer161->Fit(m_panel11); - bSizer_TabsForFutureUse->Add(m_panel11, 1, wxEXPAND | wxALL, 5); - - bSizer2->Add(bSizer_TabsForFutureUse, 1, wxEXPAND, 5); - - this->SetSizer(bSizer2); - this->Layout(); - - // Connect Events - this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CMainFrameBase::OnClose)); - this->Connect(wxEVT_IDLE, wxIdleEventHandler(CMainFrameBase::OnIdle)); - this->Connect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Connect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaint)); - this->Connect(m_menuFileExit->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuFileExit)); - this->Connect(m_menuOptionsGenerateBitcoins->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsGenerate)); - this->Connect(m_menuChangeYourAddress->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsChangeYourAddress)); - this->Connect(m_menuOptionsOptions->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsOptions)); - this->Connect(m_menuHelpAbout->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuHelpAbout)); - this->Connect(wxID_BUTTONSEND, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonSend)); - this->Connect(wxID_BUTTONRECEIVE, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonAddressBook)); - m_textCtrlAddress->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CMainFrameBase::OnKeyDown), NULL, this); - m_textCtrlAddress->Connect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Connect(wxEVT_SET_FOCUS, wxFocusEventHandler(CMainFrameBase::OnSetFocusAddress), NULL, this); - m_buttonCopy->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonCopy), NULL, this); - m_button91->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonChange), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler(CMainFrameBase::OnListColBeginDrag), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedAllTransactions), NULL, this); - m_listCtrl->Connect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaintListCtrl), NULL, this); - m_listCtrlOrdersSent->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersSent), NULL, this); - m_listCtrlProductsSent->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedProductsSent), NULL, this); - m_listCtrlOrdersReceived->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersReceived), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE ) ); + + m_menubar = new wxMenuBar( 0 ); + m_menubar->SetBackgroundColour( wxColour( 240, 240, 240 ) ); + + m_menuFile = new wxMenu(); + wxMenuItem* m_menuFileExit; + m_menuFileExit = new wxMenuItem( m_menuFile, wxID_ANY, wxString( wxT("E&xit") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuFile->Append( m_menuFileExit ); + + m_menubar->Append( m_menuFile, wxT("&File") ); + + m_menuOptions = new wxMenu(); + wxMenuItem* m_menuOptionsGenerateBitcoins; + m_menuOptionsGenerateBitcoins = new wxMenuItem( m_menuOptions, wxID_OPTIONSGENERATEBITCOINS, wxString( wxT("&Generate Coins") ) , wxEmptyString, wxITEM_CHECK ); + m_menuOptions->Append( m_menuOptionsGenerateBitcoins ); + + wxMenuItem* m_menuChangeYourAddress; + m_menuChangeYourAddress = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( wxT("&Change Your Address...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptions->Append( m_menuChangeYourAddress ); + + wxMenuItem* m_menuOptionsOptions; + m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( wxT("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptions->Append( m_menuOptionsOptions ); + + m_menubar->Append( m_menuOptions, wxT("&Options") ); + + m_menuHelp = new wxMenu(); + wxMenuItem* m_menuHelpAbout; + m_menuHelpAbout = new wxMenuItem( m_menuHelp, wxID_ANY, wxString( wxT("&About...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuHelp->Append( m_menuHelpAbout ); + + m_menubar->Append( m_menuHelp, wxT("&Help") ); + + this->SetMenuBar( m_menubar ); + + m_toolBar = this->CreateToolBar( wxTB_FLAT|wxTB_HORZ_TEXT, wxID_ANY ); + m_toolBar->SetToolBitmapSize( wxSize( 20,20 ) ); + m_toolBar->SetToolSeparation( 1 ); + m_toolBar->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); + + m_toolBar->AddTool( wxID_BUTTONSEND, wxT("&Send Coins"), wxBitmap( wxT("send20"), wxBITMAP_TYPE_RESOURCE ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->AddTool( wxID_BUTTONRECEIVE, wxT("&Address Book"), wxBitmap( wxT("addressbook20"), wxBITMAP_TYPE_RESOURCE ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->Realize(); + + m_statusBar = this->CreateStatusBar( 1, wxST_SIZEGRIP, wxID_ANY ); + m_statusBar->SetBackgroundColour( wxColour( 240, 240, 240 ) ); + + wxBoxSizer* bSizer2; + bSizer2 = new wxBoxSizer( wxVERTICAL ); + + + bSizer2->Add( 0, 2, 0, wxEXPAND, 5 ); + + wxBoxSizer* bSizer85; + bSizer85 = new wxBoxSizer( wxHORIZONTAL ); + + m_staticText32 = new wxStaticText( this, wxID_ANY, wxT("Your Bitcoin Address:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText32->Wrap( -1 ); + bSizer85->Add( m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxLEFT, 5 ); + + m_textCtrlAddress = new wxTextCtrl( this, wxID_TEXTCTRLADDRESS, wxEmptyString, wxDefaultPosition, wxSize( 250,-1 ), wxTE_READONLY ); + m_textCtrlAddress->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); + + bSizer85->Add( m_textCtrlAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT ); + bSizer85->Add( m_buttonCopy, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); + + m_button91 = new wxButton( this, wxID_BUTTONCHANGE, wxT("C&hange..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_button91->Hide(); + + bSizer85->Add( m_button91, 0, wxRIGHT, 5 ); + + + bSizer85->Add( 0, 0, 0, wxEXPAND, 5 ); + + bSizer2->Add( bSizer85, 0, wxEXPAND|wxRIGHT|wxLEFT, 5 ); + + wxBoxSizer* bSizer3; + bSizer3 = new wxBoxSizer( wxHORIZONTAL ); + + m_panel14 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer66; + bSizer66 = new wxBoxSizer( wxHORIZONTAL ); + + m_staticText41 = new wxStaticText( m_panel14, wxID_ANY, wxT("Balance:"), wxDefaultPosition, wxSize( -1,15 ), 0 ); + m_staticText41->Wrap( -1 ); + bSizer66->Add( m_staticText41, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_staticTextBalance = new wxStaticText( m_panel14, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( 120,15 ), wxALIGN_RIGHT|wxST_NO_AUTORESIZE ); + m_staticTextBalance->Wrap( -1 ); + m_staticTextBalance->SetFont( wxFont( 8, 70, 90, 90, false, wxEmptyString ) ); + m_staticTextBalance->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW ) ); + + bSizer66->Add( m_staticTextBalance, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + m_panel14->SetSizer( bSizer66 ); + m_panel14->Layout(); + bSizer66->Fit( m_panel14 ); + bSizer3->Add( m_panel14, 1, wxEXPAND|wxALIGN_BOTTOM|wxALL, 5 ); + + + bSizer3->Add( 0, 0, 0, wxEXPAND, 5 ); + + wxString m_choiceFilterChoices[] = { wxT(" All"), wxT(" Sent"), wxT(" Received"), wxT(" In Progress") }; + int m_choiceFilterNChoices = sizeof( m_choiceFilterChoices ) / sizeof( wxString ); + m_choiceFilter = new wxChoice( this, wxID_ANY, wxDefaultPosition, wxSize( 110,-1 ), m_choiceFilterNChoices, m_choiceFilterChoices, 0 ); + m_choiceFilter->SetSelection( 0 ); + m_choiceFilter->Hide(); + + bSizer3->Add( m_choiceFilter, 0, wxALIGN_BOTTOM|wxTOP|wxRIGHT|wxLEFT, 5 ); + + bSizer2->Add( bSizer3, 0, wxEXPAND, 5 ); + + m_notebook = new wxNotebook( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 ); + m_panel7 = new wxPanel( m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer157; + bSizer157 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrl = new wxListCtrl( m_panel7, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxALWAYS_SHOW_SB ); + bSizer157->Add( m_listCtrl, 1, wxEXPAND|wxALL, 5 ); + + m_panel7->SetSizer( bSizer157 ); + m_panel7->Layout(); + bSizer157->Fit( m_panel7 ); + m_notebook->AddPage( m_panel7, wxT("All Transactions"), false ); + + bSizer2->Add( m_notebook, 1, wxEXPAND, 5 ); + + wxBoxSizer* bSizer_TabsForFutureUse; + bSizer_TabsForFutureUse = new wxBoxSizer( wxVERTICAL ); + + m_panel9 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + m_panel9->Hide(); + + wxBoxSizer* bSizer159; + bSizer159 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlEscrows = new wxListCtrl( m_panel9, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); + bSizer159->Add( m_listCtrlEscrows, 1, wxALL|wxEXPAND, 5 ); + + m_panel9->SetSizer( bSizer159 ); + m_panel9->Layout(); + bSizer159->Fit( m_panel9 ); + bSizer_TabsForFutureUse->Add( m_panel9, 1, wxEXPAND | wxALL, 5 ); + + m_panel8 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + m_panel8->Hide(); + + wxBoxSizer* bSizer158; + bSizer158 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlOrdersSent = new wxListCtrl( m_panel8, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); + bSizer158->Add( m_listCtrlOrdersSent, 1, wxALL|wxEXPAND, 5 ); + + m_panel8->SetSizer( bSizer158 ); + m_panel8->Layout(); + bSizer158->Fit( m_panel8 ); + bSizer_TabsForFutureUse->Add( m_panel8, 1, wxEXPAND | wxALL, 5 ); + + m_panel10 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + m_panel10->Hide(); + + wxBoxSizer* bSizer160; + bSizer160 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlProductsSent = new wxListCtrl( m_panel10, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); + bSizer160->Add( m_listCtrlProductsSent, 1, wxALL|wxEXPAND, 5 ); + + m_panel10->SetSizer( bSizer160 ); + m_panel10->Layout(); + bSizer160->Fit( m_panel10 ); + bSizer_TabsForFutureUse->Add( m_panel10, 1, wxEXPAND | wxALL, 5 ); + + m_panel11 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + m_panel11->Hide(); + + wxBoxSizer* bSizer161; + bSizer161 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlOrdersReceived = new wxListCtrl( m_panel11, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); + bSizer161->Add( m_listCtrlOrdersReceived, 1, wxALL|wxEXPAND, 5 ); + + m_panel11->SetSizer( bSizer161 ); + m_panel11->Layout(); + bSizer161->Fit( m_panel11 ); + bSizer_TabsForFutureUse->Add( m_panel11, 1, wxEXPAND | wxALL, 5 ); + + bSizer2->Add( bSizer_TabsForFutureUse, 1, wxEXPAND, 5 ); + + this->SetSizer( bSizer2 ); + this->Layout(); + + // Connect Events + this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CMainFrameBase::OnClose ) ); + this->Connect( wxEVT_ICONIZE, wxIconizeEventHandler( CMainFrameBase::OnIconize ) ); + this->Connect( wxEVT_IDLE, wxIdleEventHandler( CMainFrameBase::OnIdle ) ); + this->Connect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_LEFT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_RIGHT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_MOTION, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaint ) ); + this->Connect( m_menuFileExit->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuFileExit ) ); + this->Connect( m_menuOptionsGenerateBitcoins->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsGenerate ) ); + this->Connect( m_menuOptionsGenerateBitcoins->GetId(), wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateMenuGenerate ) ); + this->Connect( m_menuChangeYourAddress->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsChangeYourAddress ) ); + this->Connect( m_menuOptionsOptions->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsOptions ) ); + this->Connect( m_menuHelpAbout->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuHelpAbout ) ); + this->Connect( wxID_BUTTONSEND, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonSend ) ); + this->Connect( wxID_BUTTONRECEIVE, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonAddressBook ) ); + m_textCtrlAddress->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CMainFrameBase::OnKeyDown ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_LEFT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_RIGHT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_MOTION, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); + m_buttonCopy->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); + m_button91->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedAllTransactions ), NULL, this ); + m_listCtrl->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlOrdersSent->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersSent ), NULL, this ); + m_listCtrlProductsSent->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedProductsSent ), NULL, this ); + m_listCtrlOrdersReceived->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersReceived ), NULL, this ); } CMainFrameBase::~CMainFrameBase() { - // Disconnect Events - this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CMainFrameBase::OnClose)); - this->Disconnect(wxEVT_IDLE, wxIdleEventHandler(CMainFrameBase::OnIdle)); - this->Disconnect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEvents)); - this->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaint)); - this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuFileExit)); - this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsGenerate)); - this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsChangeYourAddress)); - this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuOptionsOptions)); - this->Disconnect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(CMainFrameBase::OnMenuHelpAbout)); - this->Disconnect(wxID_BUTTONSEND, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonSend)); - this->Disconnect(wxID_BUTTONRECEIVE, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonAddressBook)); - m_textCtrlAddress->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CMainFrameBase::OnKeyDown), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_LEFT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_RIGHT_UP, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_MOTION, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CMainFrameBase::OnMouseEventsAddress), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_SET_FOCUS, wxFocusEventHandler(CMainFrameBase::OnSetFocusAddress), NULL, this); - m_buttonCopy->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonCopy), NULL, this); - m_button91->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CMainFrameBase::OnButtonChange), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler(CMainFrameBase::OnListColBeginDrag), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedAllTransactions), NULL, this); - m_listCtrl->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CMainFrameBase::OnPaintListCtrl), NULL, this); - m_listCtrlOrdersSent->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersSent), NULL, this); - m_listCtrlProductsSent->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedProductsSent), NULL, this); - m_listCtrlOrdersReceived->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CMainFrameBase::OnListItemActivatedOrdersReceived), NULL, this); + // Disconnect Events + this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CMainFrameBase::OnClose ) ); + this->Disconnect( wxEVT_ICONIZE, wxIconizeEventHandler( CMainFrameBase::OnIconize ) ); + this->Disconnect( wxEVT_IDLE, wxIdleEventHandler( CMainFrameBase::OnIdle ) ); + this->Disconnect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_LEFT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_RIGHT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MOTION, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaint ) ); + this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuFileExit ) ); + this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsGenerate ) ); + this->Disconnect( wxID_ANY, wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateMenuGenerate ) ); + this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsChangeYourAddress ) ); + this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsOptions ) ); + this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuHelpAbout ) ); + this->Disconnect( wxID_BUTTONSEND, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonSend ) ); + this->Disconnect( wxID_BUTTONRECEIVE, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonAddressBook ) ); + m_textCtrlAddress->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CMainFrameBase::OnKeyDown ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_LEFT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_RIGHT_UP, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_MOTION, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); + m_buttonCopy->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); + m_button91->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedAllTransactions ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlOrdersSent->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersSent ), NULL, this ); + m_listCtrlProductsSent->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedProductsSent ), NULL, this ); + m_listCtrlOrdersReceived->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersReceived ), NULL, this ); } -CTxDetailsDialogBase::CTxDetailsDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CTxDetailsDialogBase::CTxDetailsDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer64; - bSizer64 = new wxBoxSizer(wxVERTICAL); - - wxBoxSizer* bSizer66; - bSizer66 = new wxBoxSizer(wxVERTICAL); - - m_htmlWin = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO); - bSizer66->Add(m_htmlWin, 1, wxALL|wxEXPAND, 5); - - bSizer64->Add(bSizer66, 1, wxEXPAND, 5); - - wxBoxSizer* bSizer65; - bSizer65 = new wxBoxSizer(wxVERTICAL); - - m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(85,25), 0); - bSizer65->Add(m_buttonOK, 0, wxALL, 5); - - bSizer64->Add(bSizer65, 0, wxALIGN_RIGHT, 5); - - this->SetSizer(bSizer64); - this->Layout(); - - // Connect Events - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CTxDetailsDialogBase::OnButtonOK), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer64; + bSizer64 = new wxBoxSizer( wxVERTICAL ); + + wxBoxSizer* bSizer66; + bSizer66 = new wxBoxSizer( wxVERTICAL ); + + m_htmlWin = new wxHtmlWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO ); + bSizer66->Add( m_htmlWin, 1, wxALL|wxEXPAND, 5 ); + + bSizer64->Add( bSizer66, 1, wxEXPAND, 5 ); + + wxBoxSizer* bSizer65; + bSizer65 = new wxBoxSizer( wxVERTICAL ); + + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( 85,25 ), 0 ); + bSizer65->Add( m_buttonOK, 0, wxALL, 5 ); + + bSizer64->Add( bSizer65, 0, wxALIGN_RIGHT, 5 ); + + this->SetSizer( bSizer64 ); + this->Layout(); + + // Connect Events + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CTxDetailsDialogBase::OnButtonOK ), NULL, this ); } CTxDetailsDialogBase::~CTxDetailsDialogBase() { - // Disconnect Events - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CTxDetailsDialogBase::OnButtonOK), NULL, this); + // Disconnect Events + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CTxDetailsDialogBase::OnButtonOK ), NULL, this ); } -COptionsDialogBase::COptionsDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer55; - bSizer55 = new wxBoxSizer(wxVERTICAL); - - wxBoxSizer* bSizer57; - bSizer57 = new wxBoxSizer(wxVERTICAL); - - - bSizer57->Add(0, 20, 0, wxEXPAND, 5); - - m_staticText32 = new wxStaticText(this, wxID_ANY, wxT("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0); - m_staticText32->Wrap(-1); - bSizer57->Add(m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); - - wxBoxSizer* bSizer56; - bSizer56 = new wxBoxSizer(wxHORIZONTAL); - - m_staticText31 = new wxStaticText(this, wxID_ANY, wxT("Transaction fee:"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText31->Wrap(-1); - bSizer56->Add(m_staticText31, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5); - - m_textCtrlTransactionFee = new wxTextCtrl(this, wxID_TRANSACTIONFEE, wxEmptyString, wxDefaultPosition, wxSize(70,-1), 0); - bSizer56->Add(m_textCtrlTransactionFee, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - bSizer57->Add(bSizer56, 0, wxEXPAND, 5); - - bSizer55->Add(bSizer57, 1, wxEXPAND|wxLEFT, 5); - - wxBoxSizer* bSizer58; - bSizer58 = new wxBoxSizer(wxHORIZONTAL); - - m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(85,25), 0); - bSizer58->Add(m_buttonOK, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer58->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer55->Add(bSizer58, 0, wxALIGN_RIGHT, 5); - - this->SetSizer(bSizer55); - this->Layout(); - - // Connect Events - m_textCtrlTransactionFee->Connect(wxEVT_KILL_FOCUS, wxFocusEventHandler(COptionsDialogBase::OnKillFocusTransactionFee), NULL, this); - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer55; + bSizer55 = new wxBoxSizer( wxVERTICAL ); + + panelSizer = new wxBoxSizer( wxHORIZONTAL ); + + m_treeCtrl = new wxTreeCtrl( this, wxID_ANY, wxDefaultPosition, wxSize( 100,-1 ), wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT ); + panelSizer->Add( m_treeCtrl, 0, wxALL|wxEXPAND, 5 ); + + bSizer55->Add( panelSizer, 1, wxEXPAND, 5 ); + + m_staticline1 = new wxStaticLine( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLI_HORIZONTAL ); + bSizer55->Add( m_staticline1, 0, wxEXPAND | wxALL, 5 ); + + wxBoxSizer* bSizer58; + bSizer58 = new wxBoxSizer( wxHORIZONTAL ); + + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( 85,25 ), 0 ); + bSizer58->Add( m_buttonOK, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer58->Add( m_buttonCancel, 0, wxALL, 5 ); + + m_buttonApply = new wxButton( this, wxID_ANY, wxT("Apply"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonApply->Enable( false ); + m_buttonApply->Hide(); + + bSizer58->Add( m_buttonApply, 0, wxALL, 5 ); + + bSizer55->Add( bSizer58, 0, wxALIGN_RIGHT, 5 ); + + this->SetSizer( bSizer55 ); + this->Layout(); + + // Connect Events + m_treeCtrl->Connect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( COptionsDialogBase::MenuSelChanged ), NULL, this ); + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonCancel ), NULL, this ); } COptionsDialogBase::~COptionsDialogBase() { - // Disconnect Events - m_textCtrlTransactionFee->Disconnect(wxEVT_KILL_FOCUS, wxFocusEventHandler(COptionsDialogBase::OnKillFocusTransactionFee), NULL, this); - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(COptionsDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + m_treeCtrl->Disconnect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( COptionsDialogBase::MenuSelChanged ), NULL, this ); + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonCancel ), NULL, this ); } -CAboutDialogBase::CAboutDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer60; - bSizer60 = new wxBoxSizer(wxVERTICAL); - - wxBoxSizer* bSizer62; - bSizer62 = new wxBoxSizer(wxHORIZONTAL); - - - bSizer62->Add(60, 0, 0, wxEXPAND, 5); - - wxBoxSizer* bSizer63; - bSizer63 = new wxBoxSizer(wxVERTICAL); - - - bSizer63->Add(0, 50, 0, wxEXPAND, 5); - - wxBoxSizer* bSizer64; - bSizer64 = new wxBoxSizer(wxHORIZONTAL); - - m_staticText40 = new wxStaticText(this, wxID_ANY, wxT("Bitcoin "), wxDefaultPosition, wxDefaultSize, 0); - m_staticText40->Wrap(-1); - m_staticText40->SetFont(wxFont(10, 74, 90, 92, false, wxT("Tahoma"))); - - bSizer64->Add(m_staticText40, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxLEFT, 5); - - m_staticTextVersion = new wxStaticText(this, wxID_ANY, wxT("version"), wxDefaultPosition, wxDefaultSize, 0); - m_staticTextVersion->Wrap(-1); - m_staticTextVersion->SetFont(wxFont(10, 74, 90, 90, false, wxT("Tahoma"))); - - bSizer64->Add(m_staticTextVersion, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxRIGHT, 5); - - bSizer63->Add(bSizer64, 0, wxEXPAND, 5); - - - bSizer63->Add(0, 4, 0, wxEXPAND, 5); - - m_staticTextMain = new wxStaticText(this, wxID_ANY, wxT("Copyright © 2009 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0); - m_staticTextMain->Wrap(400); - bSizer63->Add(m_staticTextMain, 0, wxALL, 5); - - - bSizer63->Add(0, 0, 1, wxEXPAND, 5); - - bSizer62->Add(bSizer63, 1, wxEXPAND, 5); - - bSizer60->Add(bSizer62, 1, wxEXPAND, 5); - - wxBoxSizer* bSizer61; - bSizer61 = new wxBoxSizer(wxHORIZONTAL); - - - bSizer61->Add(0, 0, 1, wxEXPAND, 5); - - m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(85,25), 0); - bSizer61->Add(m_buttonOK, 0, wxALL, 5); - - bSizer60->Add(bSizer61, 0, wxALIGN_RIGHT|wxEXPAND, 5); - - this->SetSizer(bSizer60); - this->Layout(); - - // Connect Events - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAboutDialogBase::OnButtonOK), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer60; + bSizer60 = new wxBoxSizer( wxVERTICAL ); + + wxBoxSizer* bSizer62; + bSizer62 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer62->Add( 60, 0, 0, wxEXPAND, 5 ); + + wxBoxSizer* bSizer63; + bSizer63 = new wxBoxSizer( wxVERTICAL ); + + + bSizer63->Add( 0, 50, 0, wxEXPAND, 5 ); + + wxBoxSizer* bSizer64; + bSizer64 = new wxBoxSizer( wxHORIZONTAL ); + + m_staticText40 = new wxStaticText( this, wxID_ANY, wxT("Bitcoin "), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText40->Wrap( -1 ); + m_staticText40->SetFont( wxFont( 10, 74, 90, 92, false, wxT("Tahoma") ) ); + + bSizer64->Add( m_staticText40, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_staticTextVersion = new wxStaticText( this, wxID_ANY, wxT("version"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextVersion->Wrap( -1 ); + m_staticTextVersion->SetFont( wxFont( 10, 74, 90, 90, false, wxT("Tahoma") ) ); + + bSizer64->Add( m_staticTextVersion, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxRIGHT, 5 ); + + bSizer63->Add( bSizer64, 0, wxEXPAND, 5 ); + + + bSizer63->Add( 0, 4, 0, wxEXPAND, 5 ); + + m_staticTextMain = new wxStaticText( this, wxID_ANY, wxT("Copyright © 2009 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMain->Wrap( 400 ); + bSizer63->Add( m_staticTextMain, 0, wxALL, 5 ); + + + bSizer63->Add( 0, 0, 1, wxEXPAND, 5 ); + + bSizer62->Add( bSizer63, 1, wxEXPAND, 5 ); + + bSizer60->Add( bSizer62, 1, wxEXPAND, 5 ); + + wxBoxSizer* bSizer61; + bSizer61 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer61->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( 85,25 ), 0 ); + bSizer61->Add( m_buttonOK, 0, wxALL, 5 ); + + bSizer60->Add( bSizer61, 0, wxALIGN_RIGHT|wxEXPAND, 5 ); + + this->SetSizer( bSizer60 ); + this->Layout(); + + // Connect Events + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAboutDialogBase::OnButtonOK ), NULL, this ); } CAboutDialogBase::~CAboutDialogBase() { - // Disconnect Events - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAboutDialogBase::OnButtonOK), NULL, this); + // Disconnect Events + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAboutDialogBase::OnButtonOK ), NULL, this ); } -CSendDialogBase::CSendDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer21; - bSizer21 = new wxBoxSizer(wxVERTICAL); - - - bSizer21->Add(0, 5, 0, wxEXPAND, 5); - - wxFlexGridSizer* fgSizer1; - fgSizer1 = new wxFlexGridSizer(3, 2, 0, 0); - fgSizer1->AddGrowableCol(1); - fgSizer1->SetFlexibleDirection(wxBOTH); - fgSizer1->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); - - - fgSizer1->Add(0, 0, 0, wxEXPAND, 5); - - m_staticText14 = new wxStaticText(this, wxID_ANY, wxT("Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, \nor Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online."), wxDefaultPosition, wxDefaultSize, 0); - m_staticText14->Wrap(-1); - fgSizer1->Add(m_staticText14, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - wxBoxSizer* bSizer47; - bSizer47 = new wxBoxSizer(wxHORIZONTAL); - - bSizer47->SetMinSize(wxSize(70,-1)); - - bSizer47->Add(0, 0, 1, wxEXPAND, 5); - - m_bitmapCheckMark = new wxStaticBitmap(this, wxID_ANY, wxICON(check), wxDefaultPosition, wxSize(16,16), 0); - bSizer47->Add(m_bitmapCheckMark, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_staticText36 = new wxStaticText(this, wxID_ANY, wxT("Pay &To:"), wxDefaultPosition, wxSize(-1,-1), wxALIGN_RIGHT); - m_staticText36->Wrap(-1); - bSizer47->Add(m_staticText36, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5); - - fgSizer1->Add(bSizer47, 1, wxEXPAND|wxLEFT, 5); - - wxBoxSizer* bSizer19; - bSizer19 = new wxBoxSizer(wxHORIZONTAL); - - m_textCtrlAddress = new wxTextCtrl(this, wxID_TEXTCTRLPAYTO, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - bSizer19->Add(m_textCtrlAddress, 1, wxALIGN_CENTER_VERTICAL|wxALL, 5); - - m_buttonPaste = new wxButton(this, wxID_BUTTONPASTE, wxT("&Paste"), wxDefaultPosition, wxSize(-1,-1), wxBU_EXACTFIT); - bSizer19->Add(m_buttonPaste, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5); - - m_buttonAddress = new wxButton(this, wxID_BUTTONADDRESSBOOK, wxT(" Address &Book..."), wxDefaultPosition, wxDefaultSize, 0); - bSizer19->Add(m_buttonAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5); - - fgSizer1->Add(bSizer19, 1, wxEXPAND|wxRIGHT, 5); - - m_staticText19 = new wxStaticText(this, wxID_ANY, wxT("&Amount:"), wxDefaultPosition, wxSize(-1,-1), wxALIGN_RIGHT); - m_staticText19->Wrap(-1); - fgSizer1->Add(m_staticText19, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT|wxALIGN_RIGHT, 5); - - m_textCtrlAmount = new wxTextCtrl(this, wxID_TEXTCTRLAMOUNT, wxEmptyString, wxDefaultPosition, wxSize(145,-1), 0); - m_textCtrlAmount->SetMaxLength(20); - m_textCtrlAmount->SetFont(wxFont(wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString)); - - fgSizer1->Add(m_textCtrlAmount, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); - - m_staticText20 = new wxStaticText(this, wxID_ANY, wxT("T&ransfer:"), wxDefaultPosition, wxSize(-1,-1), wxALIGN_RIGHT); - m_staticText20->Wrap(-1); - m_staticText20->Hide(); - - fgSizer1->Add(m_staticText20, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); - - wxString m_choiceTransferTypeChoices[] = { wxT(" Standard") }; - int m_choiceTransferTypeNChoices = sizeof(m_choiceTransferTypeChoices) / sizeof(wxString); - m_choiceTransferType = new wxChoice(this, wxID_CHOICETRANSFERTYPE, wxDefaultPosition, wxDefaultSize, m_choiceTransferTypeNChoices, m_choiceTransferTypeChoices, 0); - m_choiceTransferType->SetSelection(0); - m_choiceTransferType->Hide(); - - fgSizer1->Add(m_choiceTransferType, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5); - - - fgSizer1->Add(0, 3, 0, wxEXPAND, 5); - - - fgSizer1->Add(0, 0, 0, wxEXPAND, 5); - - bSizer21->Add(fgSizer1, 0, wxEXPAND|wxLEFT, 5); - - wxBoxSizer* bSizer672; - bSizer672 = new wxBoxSizer(wxHORIZONTAL); - - wxBoxSizer* bSizer681; - bSizer681 = new wxBoxSizer(wxVERTICAL); - - m_staticTextFrom = new wxStaticText(this, wxID_ANY, wxT("&From:"), wxDefaultPosition, wxDefaultSize, 0); - m_staticTextFrom->Wrap(-1); - bSizer681->Add(m_staticTextFrom, 0, wxBOTTOM|wxLEFT, 5); - - m_textCtrlFrom = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - bSizer681->Add(m_textCtrlFrom, 0, wxLEFT|wxEXPAND, 5); - - bSizer672->Add(bSizer681, 1, wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT, 5); - - bSizer21->Add(bSizer672, 0, wxEXPAND, 5); - - wxBoxSizer* bSizer67; - bSizer67 = new wxBoxSizer(wxHORIZONTAL); - - wxBoxSizer* bSizer68; - bSizer68 = new wxBoxSizer(wxVERTICAL); - - m_staticTextMessage = new wxStaticText(this, wxID_ANY, wxT("&Message:"), wxDefaultPosition, wxDefaultSize, 0); - m_staticTextMessage->Wrap(-1); - bSizer68->Add(m_staticTextMessage, 0, wxTOP|wxBOTTOM|wxLEFT, 5); - - m_textCtrlMessage = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); - bSizer68->Add(m_textCtrlMessage, 1, wxEXPAND|wxLEFT, 5); - - bSizer67->Add(bSizer68, 1, wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT, 5); - - bSizer21->Add(bSizer67, 1, wxEXPAND, 5); - - wxBoxSizer* bSizer23; - bSizer23 = new wxBoxSizer(wxHORIZONTAL); - - - bSizer23->Add(0, 0, 1, wxEXPAND, 5); - - m_buttonSend = new wxButton(this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonSend->SetFont(wxFont(wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString)); - m_buttonSend->SetMinSize(wxSize(85,25)); - - bSizer23->Add(m_buttonSend, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer23->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer21->Add(bSizer23, 0, wxEXPAND, 5); - - this->SetSizer(bSizer21); - this->Layout(); - - // Connect Events - m_textCtrlAddress->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_textCtrlAddress->Connect(wxEVT_COMMAND_TEXT_UPDATED, wxCommandEventHandler(CSendDialogBase::OnTextAddress), NULL, this); - m_buttonPaste->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonPaste), NULL, this); - m_buttonAddress->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonAddressBook), NULL, this); - m_textCtrlAmount->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_textCtrlAmount->Connect(wxEVT_KILL_FOCUS, wxFocusEventHandler(CSendDialogBase::OnKillFocusAmount), NULL, this); - m_textCtrlFrom->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_textCtrlMessage->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_buttonSend->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonSend), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer21; + bSizer21 = new wxBoxSizer( wxVERTICAL ); + + + bSizer21->Add( 0, 5, 0, wxEXPAND, 5 ); + + wxFlexGridSizer* fgSizer1; + fgSizer1 = new wxFlexGridSizer( 3, 2, 0, 0 ); + fgSizer1->AddGrowableCol( 1 ); + fgSizer1->SetFlexibleDirection( wxBOTH ); + fgSizer1->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED ); + + + fgSizer1->Add( 0, 0, 0, wxEXPAND, 5 ); + + m_staticText14 = new wxStaticText( this, wxID_ANY, wxT("Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, \nor Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText14->Wrap( -1 ); + fgSizer1->Add( m_staticText14, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + wxBoxSizer* bSizer47; + bSizer47 = new wxBoxSizer( wxHORIZONTAL ); + + bSizer47->SetMinSize( wxSize( 70,-1 ) ); + + bSizer47->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_bitmapCheckMark = new wxStaticBitmap( this, wxID_ANY, wxICON( check ), wxDefaultPosition, wxSize( 16,16 ), 0 ); + bSizer47->Add( m_bitmapCheckMark, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_staticText36 = new wxStaticText( this, wxID_ANY, wxT("Pay &To:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); + m_staticText36->Wrap( -1 ); + bSizer47->Add( m_staticText36, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + fgSizer1->Add( bSizer47, 1, wxEXPAND|wxLEFT, 5 ); + + wxBoxSizer* bSizer19; + bSizer19 = new wxBoxSizer( wxHORIZONTAL ); + + m_textCtrlAddress = new wxTextCtrl( this, wxID_TEXTCTRLPAYTO, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + bSizer19->Add( m_textCtrlAddress, 1, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + m_buttonPaste = new wxButton( this, wxID_BUTTONPASTE, wxT("&Paste"), wxDefaultPosition, wxSize( -1,-1 ), wxBU_EXACTFIT ); + bSizer19->Add( m_buttonPaste, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); + + m_buttonAddress = new wxButton( this, wxID_BUTTONADDRESSBOOK, wxT(" Address &Book..."), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer19->Add( m_buttonAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); + + fgSizer1->Add( bSizer19, 1, wxEXPAND|wxRIGHT, 5 ); + + m_staticText19 = new wxStaticText( this, wxID_ANY, wxT("&Amount:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); + m_staticText19->Wrap( -1 ); + fgSizer1->Add( m_staticText19, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT|wxALIGN_RIGHT, 5 ); + + m_textCtrlAmount = new wxTextCtrl( this, wxID_TEXTCTRLAMOUNT, wxEmptyString, wxDefaultPosition, wxSize( 145,-1 ), 0 ); + m_textCtrlAmount->SetMaxLength( 20 ); + m_textCtrlAmount->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); + + fgSizer1->Add( m_textCtrlAmount, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + m_staticText20 = new wxStaticText( this, wxID_ANY, wxT("T&ransfer:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); + m_staticText20->Wrap( -1 ); + m_staticText20->Hide(); + + fgSizer1->Add( m_staticText20, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + wxString m_choiceTransferTypeChoices[] = { wxT(" Standard") }; + int m_choiceTransferTypeNChoices = sizeof( m_choiceTransferTypeChoices ) / sizeof( wxString ); + m_choiceTransferType = new wxChoice( this, wxID_CHOICETRANSFERTYPE, wxDefaultPosition, wxDefaultSize, m_choiceTransferTypeNChoices, m_choiceTransferTypeChoices, 0 ); + m_choiceTransferType->SetSelection( 0 ); + m_choiceTransferType->Hide(); + + fgSizer1->Add( m_choiceTransferType, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + + fgSizer1->Add( 0, 3, 0, wxEXPAND, 5 ); + + + fgSizer1->Add( 0, 0, 0, wxEXPAND, 5 ); + + bSizer21->Add( fgSizer1, 0, wxEXPAND|wxLEFT, 5 ); + + wxBoxSizer* bSizer672; + bSizer672 = new wxBoxSizer( wxHORIZONTAL ); + + wxBoxSizer* bSizer681; + bSizer681 = new wxBoxSizer( wxVERTICAL ); + + m_staticTextFrom = new wxStaticText( this, wxID_ANY, wxT("&From:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextFrom->Wrap( -1 ); + bSizer681->Add( m_staticTextFrom, 0, wxBOTTOM|wxLEFT, 5 ); + + m_textCtrlFrom = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + bSizer681->Add( m_textCtrlFrom, 0, wxLEFT|wxEXPAND, 5 ); + + bSizer672->Add( bSizer681, 1, wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT, 5 ); + + bSizer21->Add( bSizer672, 0, wxEXPAND, 5 ); + + wxBoxSizer* bSizer67; + bSizer67 = new wxBoxSizer( wxHORIZONTAL ); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer( wxVERTICAL ); + + m_staticTextMessage = new wxStaticText( this, wxID_ANY, wxT("&Message:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMessage->Wrap( -1 ); + bSizer68->Add( m_staticTextMessage, 0, wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_textCtrlMessage = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE ); + bSizer68->Add( m_textCtrlMessage, 1, wxEXPAND|wxLEFT, 5 ); + + bSizer67->Add( bSizer68, 1, wxEXPAND|wxBOTTOM|wxRIGHT|wxLEFT, 5 ); + + bSizer21->Add( bSizer67, 1, wxEXPAND, 5 ); + + wxBoxSizer* bSizer23; + bSizer23 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer23->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_buttonSend = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonSend->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); + m_buttonSend->SetMinSize( wxSize( 85,25 ) ); + + bSizer23->Add( m_buttonSend, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer23->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer21->Add( bSizer23, 0, wxEXPAND, 5 ); + + this->SetSizer( bSizer21 ); + this->Layout(); + + // Connect Events + m_textCtrlAddress->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlAddress->Connect( wxEVT_COMMAND_TEXT_UPDATED, wxCommandEventHandler( CSendDialogBase::OnTextAddress ), NULL, this ); + m_buttonPaste->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonPaste ), NULL, this ); + m_buttonAddress->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonAddressBook ), NULL, this ); + m_textCtrlAmount->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlAmount->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( CSendDialogBase::OnKillFocusAmount ), NULL, this ); + m_textCtrlFrom->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlMessage->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_buttonSend->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonSend ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonCancel ), NULL, this ); } CSendDialogBase::~CSendDialogBase() { - // Disconnect Events - m_textCtrlAddress->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_textCtrlAddress->Disconnect(wxEVT_COMMAND_TEXT_UPDATED, wxCommandEventHandler(CSendDialogBase::OnTextAddress), NULL, this); - m_buttonPaste->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonPaste), NULL, this); - m_buttonAddress->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonAddressBook), NULL, this); - m_textCtrlAmount->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_textCtrlAmount->Disconnect(wxEVT_KILL_FOCUS, wxFocusEventHandler(CSendDialogBase::OnKillFocusAmount), NULL, this); - m_textCtrlFrom->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_textCtrlMessage->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CSendDialogBase::OnKeyDown), NULL, this); - m_buttonSend->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonSend), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + m_textCtrlAddress->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlAddress->Disconnect( wxEVT_COMMAND_TEXT_UPDATED, wxCommandEventHandler( CSendDialogBase::OnTextAddress ), NULL, this ); + m_buttonPaste->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonPaste ), NULL, this ); + m_buttonAddress->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonAddressBook ), NULL, this ); + m_textCtrlAmount->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlAmount->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( CSendDialogBase::OnKillFocusAmount ), NULL, this ); + m_textCtrlFrom->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlMessage->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CSendDialogBase::OnKeyDown ), NULL, this ); + m_buttonSend->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonSend ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendDialogBase::OnButtonCancel ), NULL, this ); } -CSendingDialogBase::CSendingDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CSendingDialogBase::CSendingDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer68; - bSizer68 = new wxBoxSizer(wxVERTICAL); - - m_staticTextSending = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,14), 0); - m_staticTextSending->Wrap(-1); - bSizer68->Add(m_staticTextSending, 0, wxALIGN_CENTER_VERTICAL|wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 8); - - m_textCtrlStatus = new wxTextCtrl(this, wxID_ANY, wxT("\n\nConnecting..."), wxDefaultPosition, wxDefaultSize, wxTE_CENTRE|wxTE_MULTILINE|wxTE_NO_VSCROLL|wxTE_READONLY|wxNO_BORDER); - m_textCtrlStatus->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE)); - - bSizer68->Add(m_textCtrlStatus, 1, wxEXPAND|wxRIGHT|wxLEFT, 10); - - wxBoxSizer* bSizer69; - bSizer69 = new wxBoxSizer(wxHORIZONTAL); - - - bSizer69->Add(0, 0, 1, wxEXPAND, 5); - - m_buttonOK = new wxButton(this, wxID_ANY, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonOK->Enable(false); - m_buttonOK->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonOK, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer68->Add(bSizer69, 0, wxEXPAND, 5); - - this->SetSizer(bSizer68); - this->Layout(); - - // Connect Events - this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CSendingDialogBase::OnClose)); - this->Connect(wxEVT_PAINT, wxPaintEventHandler(CSendingDialogBase::OnPaint)); - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer( wxVERTICAL ); + + m_staticTextSending = new wxStaticText( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,14 ), 0 ); + m_staticTextSending->Wrap( -1 ); + bSizer68->Add( m_staticTextSending, 0, wxALIGN_CENTER_VERTICAL|wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 8 ); + + m_textCtrlStatus = new wxTextCtrl( this, wxID_ANY, wxT("\n\nConnecting..."), wxDefaultPosition, wxDefaultSize, wxTE_CENTRE|wxTE_MULTILINE|wxTE_NO_VSCROLL|wxTE_READONLY|wxNO_BORDER ); + m_textCtrlStatus->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE ) ); + + bSizer68->Add( m_textCtrlStatus, 1, wxEXPAND|wxRIGHT|wxLEFT, 10 ); + + wxBoxSizer* bSizer69; + bSizer69 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_buttonOK = new wxButton( this, wxID_ANY, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonOK->Enable( false ); + m_buttonOK->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); + + this->SetSizer( bSizer68 ); + this->Layout(); + + // Connect Events + this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CSendingDialogBase::OnClose ) ); + this->Connect( wxEVT_PAINT, wxPaintEventHandler( CSendingDialogBase::OnPaint ) ); + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendingDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendingDialogBase::OnButtonCancel ), NULL, this ); } CSendingDialogBase::~CSendingDialogBase() { - // Disconnect Events - this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CSendingDialogBase::OnClose)); - this->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CSendingDialogBase::OnPaint)); - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CSendingDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CSendingDialogBase::OnClose ) ); + this->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CSendingDialogBase::OnPaint ) ); + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendingDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CSendingDialogBase::OnButtonCancel ), NULL, this ); } -CYourAddressDialogBase::CYourAddressDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CYourAddressDialogBase::CYourAddressDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer68; - bSizer68 = new wxBoxSizer(wxVERTICAL); - - - bSizer68->Add(0, 5, 0, wxEXPAND, 5); - - m_staticText45 = new wxStaticText(this, wxID_ANY, wxT("These are your Bitcoin addresses for receiving payments. You may want to give a different one to each sender so you can keep track of who is paying you. The highlighted address is displayed in the main window."), wxDefaultPosition, wxDefaultSize, 0); - m_staticText45->Wrap(590); - bSizer68->Add(m_staticText45, 0, wxALL, 5); - - m_listCtrl = new wxListCtrl(this, wxID_LISTCTRL, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING); - bSizer68->Add(m_listCtrl, 1, wxALL|wxEXPAND, 5); - - wxBoxSizer* bSizer69; - bSizer69 = new wxBoxSizer(wxHORIZONTAL); - - - bSizer69->Add(0, 0, 1, wxEXPAND, 5); - - m_buttonRename = new wxButton(this, wxID_BUTTONRENAME, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0); - m_buttonRename->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonRename, 0, wxALL, 5); - - m_buttonNew = new wxButton(this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonNew->SetMinSize(wxSize(110,25)); - - bSizer69->Add(m_buttonNew, 0, wxALL, 5); - - m_buttonCopy = new wxButton(this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonCopy->SetMinSize(wxSize(120,25)); - - bSizer69->Add(m_buttonCopy, 0, wxALL, 5); - - m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonOK->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonOK, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonCancel->Hide(); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer68->Add(bSizer69, 0, wxEXPAND, 5); - - this->SetSizer(bSizer68); - this->Layout(); - - // Connect Events - this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CYourAddressDialogBase::OnClose)); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CYourAddressDialogBase::OnListEndLabelEdit), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CYourAddressDialogBase::OnListItemActivated), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CYourAddressDialogBase::OnListItemSelected), NULL, this); - m_buttonRename->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonRename), NULL, this); - m_buttonNew->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonNew), NULL, this); - m_buttonCopy->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCopy), NULL, this); - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer( wxVERTICAL ); + + + bSizer68->Add( 0, 5, 0, wxEXPAND, 5 ); + + m_staticText45 = new wxStaticText( this, wxID_ANY, wxT("These are your Bitcoin addresses for receiving payments. You may want to give a different one to each sender so you can keep track of who is paying you. The highlighted address is displayed in the main window."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText45->Wrap( 590 ); + bSizer68->Add( m_staticText45, 0, wxALL, 5 ); + + m_listCtrl = new wxListCtrl( this, wxID_LISTCTRL, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING ); + bSizer68->Add( m_listCtrl, 1, wxALL|wxEXPAND, 5 ); + + wxBoxSizer* bSizer69; + bSizer69 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_buttonRename = new wxButton( this, wxID_BUTTONRENAME, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonRename->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonRename, 0, wxALL, 5 ); + + m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonNew->SetMinSize( wxSize( 110,25 ) ); + + bSizer69->Add( m_buttonNew, 0, wxALL, 5 ); + + m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCopy->SetMinSize( wxSize( 120,25 ) ); + + bSizer69->Add( m_buttonCopy, 0, wxALL, 5 ); + + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonOK->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel->Hide(); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); + + this->SetSizer( bSizer68 ); + this->Layout(); + + // Connect Events + this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CYourAddressDialogBase::OnClose ) ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CYourAddressDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CYourAddressDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CYourAddressDialogBase::OnListItemSelected ), NULL, this ); + m_buttonRename->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonRename ), NULL, this ); + m_buttonNew->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonNew ), NULL, this ); + m_buttonCopy->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonCopy ), NULL, this ); + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonCancel ), NULL, this ); } CYourAddressDialogBase::~CYourAddressDialogBase() { - // Disconnect Events - this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CYourAddressDialogBase::OnClose)); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CYourAddressDialogBase::OnListEndLabelEdit), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CYourAddressDialogBase::OnListItemActivated), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CYourAddressDialogBase::OnListItemSelected), NULL, this); - m_buttonRename->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonRename), NULL, this); - m_buttonNew->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonNew), NULL, this); - m_buttonCopy->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCopy), NULL, this); - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CYourAddressDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CYourAddressDialogBase::OnClose ) ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CYourAddressDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CYourAddressDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CYourAddressDialogBase::OnListItemSelected ), NULL, this ); + m_buttonRename->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonRename ), NULL, this ); + m_buttonNew->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonNew ), NULL, this ); + m_buttonCopy->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonCopy ), NULL, this ); + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CYourAddressDialogBase::OnButtonCancel ), NULL, this ); } -CAddressBookDialogBase::CAddressBookDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CAddressBookDialogBase::CAddressBookDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer68; - bSizer68 = new wxBoxSizer(wxVERTICAL); - - - bSizer68->Add(0, 5, 0, wxEXPAND, 5); - - m_staticText55 = new wxStaticText(this, wxID_ANY, wxT("Bitcoin Address"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText55->Wrap(-1); - m_staticText55->Hide(); - - bSizer68->Add(m_staticText55, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - m_listCtrl = new wxListCtrl(this, wxID_LISTCTRL, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING); - bSizer68->Add(m_listCtrl, 1, wxALL|wxEXPAND, 5); - - wxBoxSizer* bSizer69; - bSizer69 = new wxBoxSizer(wxHORIZONTAL); - - - bSizer69->Add(0, 0, 1, wxEXPAND, 5); - - m_buttonEdit = new wxButton(this, wxID_BUTTONEDIT, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0); - m_buttonEdit->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonEdit, 0, wxALL, 5); - - m_buttonNew = new wxButton(this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxDefaultSize, 0); - m_buttonNew->SetMinSize(wxSize(110,25)); - - bSizer69->Add(m_buttonNew, 0, wxALL, 5); - - m_buttonDelete = new wxButton(this, wxID_BUTTONDELETE, wxT("&Delete"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonDelete->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonDelete, 0, wxALL, 5); - - m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonOK->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonOK, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer69->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer68->Add(bSizer69, 0, wxEXPAND, 5); - - this->SetSizer(bSizer68); - this->Layout(); - - // Connect Events - this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CAddressBookDialogBase::OnClose)); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CAddressBookDialogBase::OnListEndLabelEdit), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CAddressBookDialogBase::OnListItemActivated), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CAddressBookDialogBase::OnListItemSelected), NULL, this); - m_buttonEdit->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonEdit), NULL, this); - m_buttonNew->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonNew), NULL, this); - m_buttonDelete->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonDelete), NULL, this); - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer68; + bSizer68 = new wxBoxSizer( wxVERTICAL ); + + + bSizer68->Add( 0, 5, 0, wxEXPAND, 5 ); + + m_staticText55 = new wxStaticText( this, wxID_ANY, wxT("Bitcoin Address"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText55->Wrap( -1 ); + m_staticText55->Hide(); + + bSizer68->Add( m_staticText55, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_listCtrl = new wxListCtrl( this, wxID_LISTCTRL, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING ); + bSizer68->Add( m_listCtrl, 1, wxALL|wxEXPAND, 5 ); + + wxBoxSizer* bSizer69; + bSizer69 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_buttonEdit = new wxButton( this, wxID_BUTTONEDIT, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonEdit->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonEdit, 0, wxALL, 5 ); + + m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonNew->SetMinSize( wxSize( 110,25 ) ); + + bSizer69->Add( m_buttonNew, 0, wxALL, 5 ); + + m_buttonDelete = new wxButton( this, wxID_BUTTONDELETE, wxT("&Delete"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonDelete->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonDelete, 0, wxALL, 5 ); + + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonOK->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); + + this->SetSizer( bSizer68 ); + this->Layout(); + + // Connect Events + this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CAddressBookDialogBase::OnClose ) ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_buttonEdit->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonEdit ), NULL, this ); + m_buttonNew->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonNew ), NULL, this ); + m_buttonDelete->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonDelete ), NULL, this ); + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonCancel ), NULL, this ); } CAddressBookDialogBase::~CAddressBookDialogBase() { - // Disconnect Events - this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CAddressBookDialogBase::OnClose)); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler(CAddressBookDialogBase::OnListEndLabelEdit), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CAddressBookDialogBase::OnListItemActivated), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CAddressBookDialogBase::OnListItemSelected), NULL, this); - m_buttonEdit->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonEdit), NULL, this); - m_buttonNew->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonNew), NULL, this); - m_buttonDelete->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonDelete), NULL, this); - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CAddressBookDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CAddressBookDialogBase::OnClose ) ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_buttonEdit->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonEdit ), NULL, this ); + m_buttonNew->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonNew ), NULL, this ); + m_buttonDelete->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonDelete ), NULL, this ); + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonCancel ), NULL, this ); } -CProductsDialogBase::CProductsDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CProductsDialogBase::CProductsDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer22; - bSizer22 = new wxBoxSizer(wxVERTICAL); - - wxBoxSizer* bSizer23; - bSizer23 = new wxBoxSizer(wxHORIZONTAL); - - m_comboBoxCategory = new wxComboBox(this, wxID_ANY, wxT("(Any Category)"), wxDefaultPosition, wxSize(150,-1), 0, NULL, 0); - m_comboBoxCategory->Append(wxT("(Any Category)")); - bSizer23->Add(m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlSearch = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - bSizer23->Add(m_textCtrlSearch, 1, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonSearch = new wxButton(this, wxID_ANY, wxT("&Search"), wxDefaultPosition, wxDefaultSize, 0); - bSizer23->Add(m_buttonSearch, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - bSizer22->Add(bSizer23, 0, wxEXPAND|wxTOP|wxBOTTOM|wxRIGHT, 5); - - m_listCtrl = new wxListCtrl(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); - bSizer22->Add(m_listCtrl, 1, wxALL|wxEXPAND, 5); - - this->SetSizer(bSizer22); - this->Layout(); - - // Connect Events - m_comboBoxCategory->Connect(wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler(CProductsDialogBase::OnCombobox), NULL, this); - m_textCtrlSearch->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CProductsDialogBase::OnKeyDown), NULL, this); - m_buttonSearch->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CProductsDialogBase::OnButtonSearch), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CProductsDialogBase::OnListItemActivated), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer22; + bSizer22 = new wxBoxSizer( wxVERTICAL ); + + wxBoxSizer* bSizer23; + bSizer23 = new wxBoxSizer( wxHORIZONTAL ); + + m_comboBoxCategory = new wxComboBox( this, wxID_ANY, wxT("(Any Category)"), wxDefaultPosition, wxSize( 150,-1 ), 0, NULL, 0 ); + m_comboBoxCategory->Append( wxT("(Any Category)") ); + bSizer23->Add( m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlSearch = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + bSizer23->Add( m_textCtrlSearch, 1, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonSearch = new wxButton( this, wxID_ANY, wxT("&Search"), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer23->Add( m_buttonSearch, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + bSizer22->Add( bSizer23, 0, wxEXPAND|wxTOP|wxBOTTOM|wxRIGHT, 5 ); + + m_listCtrl = new wxListCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); + bSizer22->Add( m_listCtrl, 1, wxALL|wxEXPAND, 5 ); + + this->SetSizer( bSizer22 ); + this->Layout(); + + // Connect Events + m_comboBoxCategory->Connect( wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler( CProductsDialogBase::OnCombobox ), NULL, this ); + m_textCtrlSearch->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CProductsDialogBase::OnKeyDown ), NULL, this ); + m_buttonSearch->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CProductsDialogBase::OnButtonSearch ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CProductsDialogBase::OnListItemActivated ), NULL, this ); } CProductsDialogBase::~CProductsDialogBase() { - // Disconnect Events - m_comboBoxCategory->Disconnect(wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler(CProductsDialogBase::OnCombobox), NULL, this); - m_textCtrlSearch->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CProductsDialogBase::OnKeyDown), NULL, this); - m_buttonSearch->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CProductsDialogBase::OnButtonSearch), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CProductsDialogBase::OnListItemActivated), NULL, this); + // Disconnect Events + m_comboBoxCategory->Disconnect( wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler( CProductsDialogBase::OnCombobox ), NULL, this ); + m_textCtrlSearch->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CProductsDialogBase::OnKeyDown ), NULL, this ); + m_buttonSearch->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CProductsDialogBase::OnButtonSearch ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CProductsDialogBase::OnListItemActivated ), NULL, this ); } -CEditProductDialogBase::CEditProductDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +CEditProductDialogBase::CEditProductDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); - - wxBoxSizer* bSizer20; - bSizer20 = new wxBoxSizer(wxVERTICAL); - - m_scrolledWindow = new wxScrolledWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL); - m_scrolledWindow->SetScrollRate(5, 5); - m_scrolledWindow->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); - - wxBoxSizer* bSizer21; - bSizer21 = new wxBoxSizer(wxVERTICAL); - - wxFlexGridSizer* fgSizer8; - fgSizer8 = new wxFlexGridSizer(0, 2, 0, 0); - fgSizer8->AddGrowableCol(1); - fgSizer8->SetFlexibleDirection(wxBOTH); - fgSizer8->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); - - m_staticText106 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Category"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); - m_staticText106->Wrap(-1); - fgSizer8->Add(m_staticText106, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); - - m_comboBoxCategory = new wxComboBox(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0, NULL, 0); - m_comboBoxCategory->SetMinSize(wxSize(180,-1)); - - fgSizer8->Add(m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_staticText108 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Title"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); - m_staticText108->Wrap(-1); - fgSizer8->Add(m_staticText108, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); - - m_textCtrlTitle = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - fgSizer8->Add(m_textCtrlTitle, 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5); - - m_staticText107 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Price"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); - m_staticText107->Wrap(-1); - fgSizer8->Add(m_staticText107, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5); - - m_textCtrlPrice = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlPrice->SetMinSize(wxSize(105,-1)); - - fgSizer8->Add(m_textCtrlPrice, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - bSizer21->Add(fgSizer8, 0, wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 5); - - m_staticText22 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Page 1: Description"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText22->Wrap(-1); - bSizer21->Add(m_staticText22, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - m_textCtrlDescription = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); - m_textCtrlDescription->SetMinSize(wxSize(-1,170)); - - bSizer21->Add(m_textCtrlDescription, 0, wxALL|wxEXPAND, 5); - - m_staticText23 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Page 2: Order Form"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText23->Wrap(-1); - bSizer21->Add(m_staticText23, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - m_textCtrlInstructions = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); - m_textCtrlInstructions->SetMinSize(wxSize(-1,120)); - - bSizer21->Add(m_textCtrlInstructions, 0, wxEXPAND|wxALL, 5); - - fgSizer5 = new wxFlexGridSizer(0, 3, 0, 0); - fgSizer5->AddGrowableCol(1); - fgSizer5->SetFlexibleDirection(wxBOTH); - fgSizer5->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); - - m_staticText24 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Label"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText24->Wrap(-1); - fgSizer5->Add(m_staticText24, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5); - - m_staticText25 = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Comma separated list of choices, or leave blank for text field"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText25->Wrap(-1); - fgSizer5->Add(m_staticText25, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5); - - - fgSizer5->Add(0, 0, 1, wxEXPAND, 5); - - m_textCtrlLabel0 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel0->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel0, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField0 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField0, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel0 = new wxButton(m_scrolledWindow, wxID_DEL0, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel0, 0, wxRIGHT|wxLEFT|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlLabel1 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel1->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel1, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField1 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField1, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel1 = new wxButton(m_scrolledWindow, wxID_DEL1, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel1, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel2 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel2->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel2, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField2 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField2, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel2 = new wxButton(m_scrolledWindow, wxID_DEL2, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel2, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel3 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel3->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel3, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField3 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField3, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel3 = new wxButton(m_scrolledWindow, wxID_DEL3, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel3, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel4 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel4->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel4, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField4 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField4, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel4 = new wxButton(m_scrolledWindow, wxID_DEL4, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel4, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel5 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel5->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel5, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField5 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField5, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel5 = new wxButton(m_scrolledWindow, wxID_DEL5, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel5, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel6 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel6->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel6, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField6 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField6, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel6 = new wxButton(m_scrolledWindow, wxID_DEL6, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel6, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel7 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel7->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel7, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField7 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField7, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel7 = new wxButton(m_scrolledWindow, wxID_DEL7, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel7, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel8 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel8->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel8, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField8 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField8, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel8 = new wxButton(m_scrolledWindow, wxID_DEL8, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel8, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel9 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel9->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel9, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField9 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField9, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel9 = new wxButton(m_scrolledWindow, wxID_DEL9, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel9, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel10 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel10->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel10, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField10 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField10, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel10 = new wxButton(m_scrolledWindow, wxID_DEL10, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel10, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel11 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel11->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel11, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField11 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField11, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel11 = new wxButton(m_scrolledWindow, wxID_DEL11, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel11, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel12 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel12->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel12, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField12 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField12, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel12 = new wxButton(m_scrolledWindow, wxID_DEL12, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel12, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel13 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel13->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel13, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField13 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField13, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel13 = new wxButton(m_scrolledWindow, wxID_DEL13, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel13, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel14 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel14->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel14, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField14 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField14, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel14 = new wxButton(m_scrolledWindow, wxID_DEL14, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel14, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel15 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel15->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel15, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField15 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField15, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel15 = new wxButton(m_scrolledWindow, wxID_DEL15, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel15, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel16 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel16->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel16, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField16 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField16, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel16 = new wxButton(m_scrolledWindow, wxID_DEL16, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel16, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel17 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel17->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel17, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField17 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField17, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel17 = new wxButton(m_scrolledWindow, wxID_DEL17, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel17, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel18 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel18->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel18, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField18 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField18, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel18 = new wxButton(m_scrolledWindow, wxID_DEL18, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel18, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - m_textCtrlLabel19 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_textCtrlLabel19->SetMinSize(wxSize(150,-1)); - - fgSizer5->Add(m_textCtrlLabel19, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - - m_textCtrlField19 = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,-1), 0); - fgSizer5->Add(m_textCtrlField19, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5); - - m_buttonDel19 = new wxButton(m_scrolledWindow, wxID_DEL19, wxT("Delete"), wxDefaultPosition, wxSize(60,20), 0); - fgSizer5->Add(m_buttonDel19, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5); - - bSizer21->Add(fgSizer5, 0, wxEXPAND, 5); - - wxBoxSizer* bSizer25; - bSizer25 = new wxBoxSizer(wxHORIZONTAL); - - m_buttonAddField = new wxButton(m_scrolledWindow, wxID_ANY, wxT("&Add Field"), wxDefaultPosition, wxDefaultSize, 0); - bSizer25->Add(m_buttonAddField, 0, wxALL, 5); - - bSizer21->Add(bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5); - - m_scrolledWindow->SetSizer(bSizer21); - m_scrolledWindow->Layout(); - bSizer21->Fit(m_scrolledWindow); - bSizer20->Add(m_scrolledWindow, 1, wxEXPAND|wxALL, 5); - - wxBoxSizer* bSizer26; - bSizer26 = new wxBoxSizer(wxHORIZONTAL); - - m_buttonOK = new wxButton(this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonOK->SetMinSize(wxSize(85,25)); - - bSizer26->Add(m_buttonOK, 0, wxALL, 5); - - m_buttonPreview = new wxButton(this, wxID_BUTTONPREVIEW, wxT("&Preview"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonPreview->SetMinSize(wxSize(85,25)); - - bSizer26->Add(m_buttonPreview, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer26->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer20->Add(bSizer26, 0, wxALIGN_RIGHT, 5); - - this->SetSizer(bSizer20); - this->Layout(); - - // Connect Events - m_textCtrlTitle->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlPrice->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlDescription->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlInstructions->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlLabel0->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField0->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel0->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel0), NULL, this); - m_textCtrlLabel1->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField1->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel1->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel1), NULL, this); - m_textCtrlLabel2->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField2->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel2->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel2), NULL, this); - m_textCtrlLabel3->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField3->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel3->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel3), NULL, this); - m_textCtrlLabel4->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField4->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel4->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel4), NULL, this); - m_textCtrlLabel5->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField5->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel5->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel5), NULL, this); - m_textCtrlLabel6->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField6->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel6->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel6), NULL, this); - m_textCtrlLabel7->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField7->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel7->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel7), NULL, this); - m_textCtrlLabel8->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField8->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel8->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel8), NULL, this); - m_textCtrlLabel9->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField9->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel9->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel9), NULL, this); - m_textCtrlLabel10->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField10->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel10->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel10), NULL, this); - m_textCtrlLabel11->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField11->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel11->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel11), NULL, this); - m_textCtrlLabel12->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField12->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel12->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel12), NULL, this); - m_textCtrlLabel13->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField13->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel13->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel13), NULL, this); - m_textCtrlLabel14->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField14->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel14->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel14), NULL, this); - m_textCtrlLabel15->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField15->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel15->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel15), NULL, this); - m_textCtrlLabel16->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField16->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel16->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel16), NULL, this); - m_textCtrlLabel17->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField17->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel17->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel17), NULL, this); - m_textCtrlLabel18->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField18->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel18->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel18), NULL, this); - m_textCtrlLabel19->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField19->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel19->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel19), NULL, this); - m_buttonAddField->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonAddField), NULL, this); - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonSend), NULL, this); - m_buttonPreview->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonPreview), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); + + wxBoxSizer* bSizer20; + bSizer20 = new wxBoxSizer( wxVERTICAL ); + + m_scrolledWindow = new wxScrolledWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL ); + m_scrolledWindow->SetScrollRate( 5, 5 ); + m_scrolledWindow->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW ) ); + + wxBoxSizer* bSizer21; + bSizer21 = new wxBoxSizer( wxVERTICAL ); + + wxFlexGridSizer* fgSizer8; + fgSizer8 = new wxFlexGridSizer( 0, 2, 0, 0 ); + fgSizer8->AddGrowableCol( 1 ); + fgSizer8->SetFlexibleDirection( wxBOTH ); + fgSizer8->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED ); + + m_staticText106 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Category"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT ); + m_staticText106->Wrap( -1 ); + fgSizer8->Add( m_staticText106, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_comboBoxCategory = new wxComboBox( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0, NULL, 0 ); + m_comboBoxCategory->SetMinSize( wxSize( 180,-1 ) ); + + fgSizer8->Add( m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_staticText108 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Title"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT ); + m_staticText108->Wrap( -1 ); + fgSizer8->Add( m_staticText108, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_textCtrlTitle = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + fgSizer8->Add( m_textCtrlTitle, 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); + + m_staticText107 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Price"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT ); + m_staticText107->Wrap( -1 ); + fgSizer8->Add( m_staticText107, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_textCtrlPrice = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlPrice->SetMinSize( wxSize( 105,-1 ) ); + + fgSizer8->Add( m_textCtrlPrice, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + bSizer21->Add( fgSizer8, 0, wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_staticText22 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Page 1: Description"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText22->Wrap( -1 ); + bSizer21->Add( m_staticText22, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlDescription = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE ); + m_textCtrlDescription->SetMinSize( wxSize( -1,170 ) ); + + bSizer21->Add( m_textCtrlDescription, 0, wxALL|wxEXPAND, 5 ); + + m_staticText23 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Page 2: Order Form"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText23->Wrap( -1 ); + bSizer21->Add( m_staticText23, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlInstructions = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE ); + m_textCtrlInstructions->SetMinSize( wxSize( -1,120 ) ); + + bSizer21->Add( m_textCtrlInstructions, 0, wxEXPAND|wxALL, 5 ); + + fgSizer5 = new wxFlexGridSizer( 0, 3, 0, 0 ); + fgSizer5->AddGrowableCol( 1 ); + fgSizer5->SetFlexibleDirection( wxBOTH ); + fgSizer5->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED ); + + m_staticText24 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Label"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText24->Wrap( -1 ); + fgSizer5->Add( m_staticText24, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_staticText25 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Comma separated list of choices, or leave blank for text field"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText25->Wrap( -1 ); + fgSizer5->Add( m_staticText25, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5 ); + + + fgSizer5->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_textCtrlLabel0 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel0->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel0, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField0 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField0, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel0 = new wxButton( m_scrolledWindow, wxID_DEL0, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel0, 0, wxRIGHT|wxLEFT|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlLabel1 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel1->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel1, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField1 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField1, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel1 = new wxButton( m_scrolledWindow, wxID_DEL1, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel1, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel2 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel2->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel2, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField2 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField2, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel2 = new wxButton( m_scrolledWindow, wxID_DEL2, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel2, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel3 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel3->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel3, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField3 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField3, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel3 = new wxButton( m_scrolledWindow, wxID_DEL3, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel3, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel4 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel4->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel4, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField4 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField4, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel4 = new wxButton( m_scrolledWindow, wxID_DEL4, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel4, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel5 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel5->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel5, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField5 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField5, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel5 = new wxButton( m_scrolledWindow, wxID_DEL5, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel5, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel6 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel6->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel6, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField6 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField6, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel6 = new wxButton( m_scrolledWindow, wxID_DEL6, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel6, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel7 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel7->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel7, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField7 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField7, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel7 = new wxButton( m_scrolledWindow, wxID_DEL7, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel7, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel8 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel8->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel8, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField8 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField8, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel8 = new wxButton( m_scrolledWindow, wxID_DEL8, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel8, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel9 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel9->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel9, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField9 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField9, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel9 = new wxButton( m_scrolledWindow, wxID_DEL9, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel9, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel10 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel10->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel10, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField10 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField10, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel10 = new wxButton( m_scrolledWindow, wxID_DEL10, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel10, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel11 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel11->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel11, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField11 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField11, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel11 = new wxButton( m_scrolledWindow, wxID_DEL11, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel11, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel12 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel12->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel12, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField12 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField12, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel12 = new wxButton( m_scrolledWindow, wxID_DEL12, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel12, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel13 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel13->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel13, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField13 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField13, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel13 = new wxButton( m_scrolledWindow, wxID_DEL13, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel13, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel14 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel14->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel14, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField14 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField14, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel14 = new wxButton( m_scrolledWindow, wxID_DEL14, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel14, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel15 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel15->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel15, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField15 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField15, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel15 = new wxButton( m_scrolledWindow, wxID_DEL15, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel15, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel16 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel16->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel16, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField16 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField16, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel16 = new wxButton( m_scrolledWindow, wxID_DEL16, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel16, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel17 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel17->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel17, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField17 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField17, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel17 = new wxButton( m_scrolledWindow, wxID_DEL17, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel17, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel18 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel18->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel18, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField18 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField18, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel18 = new wxButton( m_scrolledWindow, wxID_DEL18, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel18, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlLabel19 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_textCtrlLabel19->SetMinSize( wxSize( 150,-1 ) ); + + fgSizer5->Add( m_textCtrlLabel19, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlField19 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); + fgSizer5->Add( m_textCtrlField19, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); + + m_buttonDel19 = new wxButton( m_scrolledWindow, wxID_DEL19, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); + fgSizer5->Add( m_buttonDel19, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); + + bSizer21->Add( fgSizer5, 0, wxEXPAND, 5 ); + + wxBoxSizer* bSizer25; + bSizer25 = new wxBoxSizer( wxHORIZONTAL ); + + m_buttonAddField = new wxButton( m_scrolledWindow, wxID_ANY, wxT("&Add Field"), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer25->Add( m_buttonAddField, 0, wxALL, 5 ); + + bSizer21->Add( bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5 ); + + m_scrolledWindow->SetSizer( bSizer21 ); + m_scrolledWindow->Layout(); + bSizer21->Fit( m_scrolledWindow ); + bSizer20->Add( m_scrolledWindow, 1, wxEXPAND|wxALL, 5 ); + + wxBoxSizer* bSizer26; + bSizer26 = new wxBoxSizer( wxHORIZONTAL ); + + m_buttonOK = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonOK->SetMinSize( wxSize( 85,25 ) ); + + bSizer26->Add( m_buttonOK, 0, wxALL, 5 ); + + m_buttonPreview = new wxButton( this, wxID_BUTTONPREVIEW, wxT("&Preview"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonPreview->SetMinSize( wxSize( 85,25 ) ); + + bSizer26->Add( m_buttonPreview, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer26->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); + + this->SetSizer( bSizer20 ); + this->Layout(); + + // Connect Events + m_textCtrlTitle->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlPrice->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlDescription->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlInstructions->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlLabel0->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField0->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel0->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel0 ), NULL, this ); + m_textCtrlLabel1->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField1->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel1->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel1 ), NULL, this ); + m_textCtrlLabel2->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField2->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel2->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel2 ), NULL, this ); + m_textCtrlLabel3->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField3->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel3->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel3 ), NULL, this ); + m_textCtrlLabel4->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField4->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel4->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel4 ), NULL, this ); + m_textCtrlLabel5->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField5->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel5->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel5 ), NULL, this ); + m_textCtrlLabel6->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField6->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel6->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel6 ), NULL, this ); + m_textCtrlLabel7->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField7->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel7->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel7 ), NULL, this ); + m_textCtrlLabel8->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField8->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel8->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel8 ), NULL, this ); + m_textCtrlLabel9->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField9->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel9->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel9 ), NULL, this ); + m_textCtrlLabel10->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField10->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel10->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel10 ), NULL, this ); + m_textCtrlLabel11->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField11->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel11->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel11 ), NULL, this ); + m_textCtrlLabel12->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField12->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel12->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel12 ), NULL, this ); + m_textCtrlLabel13->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField13->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel13->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel13 ), NULL, this ); + m_textCtrlLabel14->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField14->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel14->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel14 ), NULL, this ); + m_textCtrlLabel15->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField15->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel15->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel15 ), NULL, this ); + m_textCtrlLabel16->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField16->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel16->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel16 ), NULL, this ); + m_textCtrlLabel17->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField17->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel17->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel17 ), NULL, this ); + m_textCtrlLabel18->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField18->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel18->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel18 ), NULL, this ); + m_textCtrlLabel19->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField19->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel19->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel19 ), NULL, this ); + m_buttonAddField->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonAddField ), NULL, this ); + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonSend ), NULL, this ); + m_buttonPreview->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonPreview ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonCancel ), NULL, this ); } CEditProductDialogBase::~CEditProductDialogBase() { - // Disconnect Events - m_textCtrlTitle->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlPrice->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlDescription->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlInstructions->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlLabel0->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField0->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel0->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel0), NULL, this); - m_textCtrlLabel1->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField1->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel1->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel1), NULL, this); - m_textCtrlLabel2->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField2->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel2->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel2), NULL, this); - m_textCtrlLabel3->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField3->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel3->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel3), NULL, this); - m_textCtrlLabel4->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField4->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel4->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel4), NULL, this); - m_textCtrlLabel5->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField5->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel5->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel5), NULL, this); - m_textCtrlLabel6->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField6->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel6->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel6), NULL, this); - m_textCtrlLabel7->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField7->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel7->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel7), NULL, this); - m_textCtrlLabel8->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField8->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel8->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel8), NULL, this); - m_textCtrlLabel9->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField9->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel9->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel9), NULL, this); - m_textCtrlLabel10->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField10->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel10->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel10), NULL, this); - m_textCtrlLabel11->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField11->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel11->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel11), NULL, this); - m_textCtrlLabel12->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField12->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel12->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel12), NULL, this); - m_textCtrlLabel13->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField13->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel13->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel13), NULL, this); - m_textCtrlLabel14->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField14->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel14->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel14), NULL, this); - m_textCtrlLabel15->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField15->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel15->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel15), NULL, this); - m_textCtrlLabel16->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField16->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel16->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel16), NULL, this); - m_textCtrlLabel17->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField17->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel17->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel17), NULL, this); - m_textCtrlLabel18->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField18->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel18->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel18), NULL, this); - m_textCtrlLabel19->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_textCtrlField19->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditProductDialogBase::OnKeyDown), NULL, this); - m_buttonDel19->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonDel19), NULL, this); - m_buttonAddField->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonAddField), NULL, this); - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonSend), NULL, this); - m_buttonPreview->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonPreview), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditProductDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + m_textCtrlTitle->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlPrice->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlDescription->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlInstructions->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlLabel0->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField0->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel0->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel0 ), NULL, this ); + m_textCtrlLabel1->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField1->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel1->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel1 ), NULL, this ); + m_textCtrlLabel2->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField2->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel2->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel2 ), NULL, this ); + m_textCtrlLabel3->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField3->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel3->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel3 ), NULL, this ); + m_textCtrlLabel4->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField4->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel4->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel4 ), NULL, this ); + m_textCtrlLabel5->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField5->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel5->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel5 ), NULL, this ); + m_textCtrlLabel6->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField6->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel6->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel6 ), NULL, this ); + m_textCtrlLabel7->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField7->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel7->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel7 ), NULL, this ); + m_textCtrlLabel8->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField8->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel8->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel8 ), NULL, this ); + m_textCtrlLabel9->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField9->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel9->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel9 ), NULL, this ); + m_textCtrlLabel10->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField10->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel10->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel10 ), NULL, this ); + m_textCtrlLabel11->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField11->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel11->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel11 ), NULL, this ); + m_textCtrlLabel12->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField12->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel12->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel12 ), NULL, this ); + m_textCtrlLabel13->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField13->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel13->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel13 ), NULL, this ); + m_textCtrlLabel14->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField14->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel14->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel14 ), NULL, this ); + m_textCtrlLabel15->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField15->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel15->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel15 ), NULL, this ); + m_textCtrlLabel16->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField16->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel16->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel16 ), NULL, this ); + m_textCtrlLabel17->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField17->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel17->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel17 ), NULL, this ); + m_textCtrlLabel18->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField18->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel18->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel18 ), NULL, this ); + m_textCtrlLabel19->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_textCtrlField19->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); + m_buttonDel19->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel19 ), NULL, this ); + m_buttonAddField->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonAddField ), NULL, this ); + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonSend ), NULL, this ); + m_buttonPreview->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonPreview ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonCancel ), NULL, this ); } -CViewProductDialogBase::CViewProductDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +CViewProductDialogBase::CViewProductDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); - - wxBoxSizer* bSizer20; - bSizer20 = new wxBoxSizer(wxVERTICAL); - - wxBoxSizer* bSizer116; - bSizer116 = new wxBoxSizer(wxHORIZONTAL); - - m_htmlWinReviews = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO); - m_htmlWinReviews->Hide(); - - bSizer116->Add(m_htmlWinReviews, 1, wxALL|wxEXPAND, 5); - - m_scrolledWindow = new wxScrolledWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL); - m_scrolledWindow->SetScrollRate(5, 5); - m_scrolledWindow->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); - - wxBoxSizer* bSizer21; - bSizer21 = new wxBoxSizer(wxVERTICAL); - - m_richTextHeading = new wxRichTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1,50), wxTE_READONLY|wxNO_BORDER); - bSizer21->Add(m_richTextHeading, 0, wxEXPAND, 5); - - m_staticTextInstructions = new wxStaticText(m_scrolledWindow, wxID_ANY, wxT("Order Form instructions here\nmultiple lines\n1\n2\n3\n4\n5\n6"), wxDefaultPosition, wxDefaultSize, 0); - m_staticTextInstructions->Wrap(-1); - bSizer21->Add(m_staticTextInstructions, 0, wxALL|wxEXPAND, 5); - - wxBoxSizer* bSizer25; - bSizer25 = new wxBoxSizer(wxHORIZONTAL); - - m_buttonSubmitForm = new wxButton(m_scrolledWindow, wxID_BUTTONSAMPLE, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0); - bSizer25->Add(m_buttonSubmitForm, 0, wxALL, 5); - - m_buttonCancelForm = new wxButton(m_scrolledWindow, wxID_CANCEL2, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); - bSizer25->Add(m_buttonCancelForm, 0, wxALL, 5); - - bSizer21->Add(bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5); - - m_scrolledWindow->SetSizer(bSizer21); - m_scrolledWindow->Layout(); - bSizer21->Fit(m_scrolledWindow); - bSizer116->Add(m_scrolledWindow, 1, wxEXPAND|wxALL, 5); - - bSizer20->Add(bSizer116, 1, wxEXPAND, 5); - - wxBoxSizer* bSizer26; - bSizer26 = new wxBoxSizer(wxHORIZONTAL); - - m_buttonBack = new wxButton(this, wxID_BUTTONBACK, wxT("< &Back "), wxDefaultPosition, wxDefaultSize, 0); - m_buttonBack->Enable(false); - m_buttonBack->SetMinSize(wxSize(85,25)); - - bSizer26->Add(m_buttonBack, 0, wxALL, 5); - - m_buttonNext = new wxButton(this, wxID_BUTTONNEXT, wxT(" &Next >"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonNext->SetMinSize(wxSize(85,25)); - - bSizer26->Add(m_buttonNext, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer26->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer20->Add(bSizer26, 0, wxALIGN_RIGHT, 5); - - this->SetSizer(bSizer20); - this->Layout(); - - // Connect Events - m_buttonSubmitForm->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonSubmitForm), NULL, this); - m_buttonCancelForm->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancelForm), NULL, this); - m_buttonBack->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonBack), NULL, this); - m_buttonNext->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonNext), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); + + wxBoxSizer* bSizer20; + bSizer20 = new wxBoxSizer( wxVERTICAL ); + + wxBoxSizer* bSizer116; + bSizer116 = new wxBoxSizer( wxHORIZONTAL ); + + m_htmlWinReviews = new wxHtmlWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO ); + m_htmlWinReviews->Hide(); + + bSizer116->Add( m_htmlWinReviews, 1, wxALL|wxEXPAND, 5 ); + + m_scrolledWindow = new wxScrolledWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL ); + m_scrolledWindow->SetScrollRate( 5, 5 ); + m_scrolledWindow->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW ) ); + + wxBoxSizer* bSizer21; + bSizer21 = new wxBoxSizer( wxVERTICAL ); + + m_richTextHeading = new wxRichTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,50 ), wxTE_READONLY|wxNO_BORDER ); + bSizer21->Add( m_richTextHeading, 0, wxEXPAND, 5 ); + + m_staticTextInstructions = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Order Form instructions here\nmultiple lines\n1\n2\n3\n4\n5\n6"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextInstructions->Wrap( -1 ); + bSizer21->Add( m_staticTextInstructions, 0, wxALL|wxEXPAND, 5 ); + + wxBoxSizer* bSizer25; + bSizer25 = new wxBoxSizer( wxHORIZONTAL ); + + m_buttonSubmitForm = new wxButton( m_scrolledWindow, wxID_BUTTONSAMPLE, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer25->Add( m_buttonSubmitForm, 0, wxALL, 5 ); + + m_buttonCancelForm = new wxButton( m_scrolledWindow, wxID_CANCEL2, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer25->Add( m_buttonCancelForm, 0, wxALL, 5 ); + + bSizer21->Add( bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5 ); + + m_scrolledWindow->SetSizer( bSizer21 ); + m_scrolledWindow->Layout(); + bSizer21->Fit( m_scrolledWindow ); + bSizer116->Add( m_scrolledWindow, 1, wxEXPAND|wxALL, 5 ); + + bSizer20->Add( bSizer116, 1, wxEXPAND, 5 ); + + wxBoxSizer* bSizer26; + bSizer26 = new wxBoxSizer( wxHORIZONTAL ); + + m_buttonBack = new wxButton( this, wxID_BUTTONBACK, wxT("< &Back "), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonBack->Enable( false ); + m_buttonBack->SetMinSize( wxSize( 85,25 ) ); + + bSizer26->Add( m_buttonBack, 0, wxALL, 5 ); + + m_buttonNext = new wxButton( this, wxID_BUTTONNEXT, wxT(" &Next >"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonNext->SetMinSize( wxSize( 85,25 ) ); + + bSizer26->Add( m_buttonNext, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer26->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); + + this->SetSizer( bSizer20 ); + this->Layout(); + + // Connect Events + m_buttonSubmitForm->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonSubmitForm ), NULL, this ); + m_buttonCancelForm->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancelForm ), NULL, this ); + m_buttonBack->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonBack ), NULL, this ); + m_buttonNext->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonNext ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancel ), NULL, this ); } CViewProductDialogBase::~CViewProductDialogBase() { - // Disconnect Events - m_buttonSubmitForm->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonSubmitForm), NULL, this); - m_buttonCancelForm->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancelForm), NULL, this); - m_buttonBack->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonBack), NULL, this); - m_buttonNext->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonNext), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewProductDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + m_buttonSubmitForm->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonSubmitForm ), NULL, this ); + m_buttonCancelForm->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancelForm ), NULL, this ); + m_buttonBack->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonBack ), NULL, this ); + m_buttonNext->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonNext ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancel ), NULL, this ); } -CViewOrderDialogBase::CViewOrderDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +CViewOrderDialogBase::CViewOrderDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); - - wxBoxSizer* bSizer20; - bSizer20 = new wxBoxSizer(wxVERTICAL); - - wxBoxSizer* bSizer116; - bSizer116 = new wxBoxSizer(wxHORIZONTAL); - - m_htmlWin = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO); - bSizer116->Add(m_htmlWin, 1, wxALL|wxEXPAND, 5); - - bSizer20->Add(bSizer116, 1, wxEXPAND, 5); - - wxBoxSizer* bSizer26; - bSizer26 = new wxBoxSizer(wxHORIZONTAL); - - m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonOK->SetMinSize(wxSize(85,25)); - - bSizer26->Add(m_buttonOK, 0, wxALL, 5); - - bSizer20->Add(bSizer26, 0, wxALIGN_RIGHT, 5); - - this->SetSizer(bSizer20); - this->Layout(); - - // Connect Events - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewOrderDialogBase::OnButtonOK), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); + + wxBoxSizer* bSizer20; + bSizer20 = new wxBoxSizer( wxVERTICAL ); + + wxBoxSizer* bSizer116; + bSizer116 = new wxBoxSizer( wxHORIZONTAL ); + + m_htmlWin = new wxHtmlWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO ); + bSizer116->Add( m_htmlWin, 1, wxALL|wxEXPAND, 5 ); + + bSizer20->Add( bSizer116, 1, wxEXPAND, 5 ); + + wxBoxSizer* bSizer26; + bSizer26 = new wxBoxSizer( wxHORIZONTAL ); + + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonOK->SetMinSize( wxSize( 85,25 ) ); + + bSizer26->Add( m_buttonOK, 0, wxALL, 5 ); + + bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); + + this->SetSizer( bSizer20 ); + this->Layout(); + + // Connect Events + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewOrderDialogBase::OnButtonOK ), NULL, this ); } CViewOrderDialogBase::~CViewOrderDialogBase() { - // Disconnect Events - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CViewOrderDialogBase::OnButtonOK), NULL, this); + // Disconnect Events + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewOrderDialogBase::OnButtonOK ), NULL, this ); } -CEditReviewDialogBase::CEditReviewDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +CEditReviewDialogBase::CEditReviewDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_MENU)); - - wxBoxSizer* bSizer112; - bSizer112 = new wxBoxSizer(wxVERTICAL); - - - bSizer112->Add(0, 3, 0, 0, 5); - - m_staticTextSeller = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_staticTextSeller->Wrap(-1); - bSizer112->Add(m_staticTextSeller, 0, wxALL|wxEXPAND, 5); - - - bSizer112->Add(0, 3, 0, 0, 5); - - m_staticText110 = new wxStaticText(this, wxID_ANY, wxT("Rating"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText110->Wrap(-1); - bSizer112->Add(m_staticText110, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - wxString m_choiceStarsChoices[] = { wxT(" 1 star"), wxT(" 2 stars"), wxT(" 3 stars"), wxT(" 4 stars"), wxT(" 5 stars") }; - int m_choiceStarsNChoices = sizeof(m_choiceStarsChoices) / sizeof(wxString); - m_choiceStars = new wxChoice(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, m_choiceStarsNChoices, m_choiceStarsChoices, 0); - m_choiceStars->SetSelection(0); - bSizer112->Add(m_choiceStars, 0, wxALL, 5); - - m_staticText43 = new wxStaticText(this, wxID_ANY, wxT("Review"), wxDefaultPosition, wxDefaultSize, 0); - m_staticText43->Wrap(-1); - bSizer112->Add(m_staticText43, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - m_textCtrlReview = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE); - bSizer112->Add(m_textCtrlReview, 1, wxALL|wxEXPAND, 5); - - wxBoxSizer* bSizer113; - bSizer113 = new wxBoxSizer(wxHORIZONTAL); - - m_buttonSubmit = new wxButton(this, wxID_SUBMIT, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonSubmit->SetMinSize(wxSize(85,25)); - - bSizer113->Add(m_buttonSubmit, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); - - bSizer113->Add(m_buttonCancel, 0, wxALL, 5); - - bSizer112->Add(bSizer113, 0, wxALIGN_RIGHT, 5); - - this->SetSizer(bSizer112); - this->Layout(); - - // Connect Events - m_textCtrlReview->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditReviewDialogBase::OnKeyDown), NULL, this); - m_buttonSubmit->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonSubmit), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonCancel), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); + + wxBoxSizer* bSizer112; + bSizer112 = new wxBoxSizer( wxVERTICAL ); + + + bSizer112->Add( 0, 3, 0, 0, 5 ); + + m_staticTextSeller = new wxStaticText( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextSeller->Wrap( -1 ); + bSizer112->Add( m_staticTextSeller, 0, wxALL|wxEXPAND, 5 ); + + + bSizer112->Add( 0, 3, 0, 0, 5 ); + + m_staticText110 = new wxStaticText( this, wxID_ANY, wxT("Rating"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText110->Wrap( -1 ); + bSizer112->Add( m_staticText110, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + wxString m_choiceStarsChoices[] = { wxT(" 1 star"), wxT(" 2 stars"), wxT(" 3 stars"), wxT(" 4 stars"), wxT(" 5 stars") }; + int m_choiceStarsNChoices = sizeof( m_choiceStarsChoices ) / sizeof( wxString ); + m_choiceStars = new wxChoice( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, m_choiceStarsNChoices, m_choiceStarsChoices, 0 ); + m_choiceStars->SetSelection( 0 ); + bSizer112->Add( m_choiceStars, 0, wxALL, 5 ); + + m_staticText43 = new wxStaticText( this, wxID_ANY, wxT("Review"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText43->Wrap( -1 ); + bSizer112->Add( m_staticText43, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_textCtrlReview = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE ); + bSizer112->Add( m_textCtrlReview, 1, wxALL|wxEXPAND, 5 ); + + wxBoxSizer* bSizer113; + bSizer113 = new wxBoxSizer( wxHORIZONTAL ); + + m_buttonSubmit = new wxButton( this, wxID_SUBMIT, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonSubmit->SetMinSize( wxSize( 85,25 ) ); + + bSizer113->Add( m_buttonSubmit, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer113->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer112->Add( bSizer113, 0, wxALIGN_RIGHT, 5 ); + + this->SetSizer( bSizer112 ); + this->Layout(); + + // Connect Events + m_textCtrlReview->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditReviewDialogBase::OnKeyDown ), NULL, this ); + m_buttonSubmit->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonSubmit ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonCancel ), NULL, this ); } CEditReviewDialogBase::~CEditReviewDialogBase() { - // Disconnect Events - m_textCtrlReview->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CEditReviewDialogBase::OnKeyDown), NULL, this); - m_buttonSubmit->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonSubmit), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CEditReviewDialogBase::OnButtonCancel), NULL, this); + // Disconnect Events + m_textCtrlReview->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditReviewDialogBase::OnKeyDown ), NULL, this ); + m_buttonSubmit->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonSubmit ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonCancel ), NULL, this ); } -CPokerLobbyDialogBase::CPokerLobbyDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +CPokerLobbyDialogBase::CPokerLobbyDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - this->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE)); - - wxBoxSizer* bSizer156; - bSizer156 = new wxBoxSizer(wxHORIZONTAL); - - m_treeCtrl = new wxTreeCtrl(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT); - m_treeCtrl->SetMinSize(wxSize(130,-1)); - - bSizer156->Add(m_treeCtrl, 0, wxEXPAND|wxTOP|wxBOTTOM|wxLEFT, 5); - - wxBoxSizer* bSizer172; - bSizer172 = new wxBoxSizer(wxVERTICAL); - - m_listCtrl = new wxListCtrl(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT); - bSizer172->Add(m_listCtrl, 1, wxEXPAND|wxALL, 5); - - m_buttonNewTable = new wxButton(this, wxID_OPENNEWTABLE, wxT("&Open New Table"), wxDefaultPosition, wxDefaultSize, 0); - bSizer172->Add(m_buttonNewTable, 0, wxALL, 5); - - bSizer156->Add(bSizer172, 1, wxEXPAND, 5); - - this->SetSizer(bSizer156); - this->Layout(); - - // Connect Events - m_treeCtrl->Connect(wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler(CPokerLobbyDialogBase::OnTreeSelChanged), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemActivated), NULL, this); - m_listCtrl->Connect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemSelected), NULL, this); - m_buttonNewTable->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerLobbyDialogBase::OnButtonNewTable), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE ) ); + + wxBoxSizer* bSizer156; + bSizer156 = new wxBoxSizer( wxHORIZONTAL ); + + m_treeCtrl = new wxTreeCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT ); + m_treeCtrl->SetMinSize( wxSize( 130,-1 ) ); + + bSizer156->Add( m_treeCtrl, 0, wxEXPAND|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + wxBoxSizer* bSizer172; + bSizer172 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrl = new wxListCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); + bSizer172->Add( m_listCtrl, 1, wxEXPAND|wxALL, 5 ); + + m_buttonNewTable = new wxButton( this, wxID_OPENNEWTABLE, wxT("&Open New Table"), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer172->Add( m_buttonNewTable, 0, wxALL, 5 ); + + bSizer156->Add( bSizer172, 1, wxEXPAND, 5 ); + + this->SetSizer( bSizer156 ); + this->Layout(); + + // Connect Events + m_treeCtrl->Connect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( CPokerLobbyDialogBase::OnTreeSelChanged ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemSelected ), NULL, this ); + m_buttonNewTable->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerLobbyDialogBase::OnButtonNewTable ), NULL, this ); } CPokerLobbyDialogBase::~CPokerLobbyDialogBase() { - // Disconnect Events - m_treeCtrl->Disconnect(wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler(CPokerLobbyDialogBase::OnTreeSelChanged), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemActivated), NULL, this); - m_listCtrl->Disconnect(wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler(CPokerLobbyDialogBase::OnListItemSelected), NULL, this); - m_buttonNewTable->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerLobbyDialogBase::OnButtonNewTable), NULL, this); + // Disconnect Events + m_treeCtrl->Disconnect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( CPokerLobbyDialogBase::OnTreeSelChanged ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemSelected ), NULL, this ); + m_buttonNewTable->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerLobbyDialogBase::OnButtonNewTable ), NULL, this ); } -CPokerDialogBase::CPokerDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) +CPokerDialogBase::CPokerDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer174; - bSizer174 = new wxBoxSizer(wxVERTICAL); - - m_checkSitOut = new wxCheckBox(this, wxID_ANY, wxT("Deal Me Out"), wxDefaultPosition, wxDefaultSize, 0); - - bSizer174->Add(m_checkSitOut, 0, wxALL, 5); - - m_buttonDealHand = new wxButton(this, wxID_DEALHAND, wxT("&Deal Hand"), wxDefaultPosition, wxSize(150,25), 0); - bSizer174->Add(m_buttonDealHand, 0, wxALL, 5); - - m_buttonFold = new wxButton(this, wxID_FOLD, wxT("&Fold"), wxDefaultPosition, wxSize(80,25), 0); - bSizer174->Add(m_buttonFold, 0, wxALL, 5); - - m_buttonCall = new wxButton(this, wxID_CALL, wxT("&Call"), wxDefaultPosition, wxSize(80,25), 0); - bSizer174->Add(m_buttonCall, 0, wxALL, 5); - - m_buttonRaise = new wxButton(this, wxID_RAISE, wxT("&Raise"), wxDefaultPosition, wxSize(80,25), 0); - bSizer174->Add(m_buttonRaise, 0, wxALL, 5); - - m_buttonLeaveTable = new wxButton(this, wxID_LEAVETABLE, wxT("&Leave Table"), wxDefaultPosition, wxSize(90,25), 0); - bSizer174->Add(m_buttonLeaveTable, 0, wxALL, 5); - - m_textDitchPlayer = new wxTextCtrl(this, wxID_DITCHPLAYER, wxEmptyString, wxDefaultPosition, wxSize(45,-1), wxTE_PROCESS_ENTER); - bSizer174->Add(m_textDitchPlayer, 0, wxALL, 5); - - m_checkPreFold = new wxCheckBox(this, wxID_ANY, wxT("FOLD"), wxDefaultPosition, wxSize(100,-1), 0); - - bSizer174->Add(m_checkPreFold, 0, wxALL, 5); - - m_checkPreCall = new wxCheckBox(this, wxID_ANY, wxT("CALL"), wxDefaultPosition, wxSize(100,-1), 0); - - bSizer174->Add(m_checkPreCall, 0, wxALL, 5); - - m_checkPreCallAny = new wxCheckBox(this, wxID_ANY, wxT("CALL ANY"), wxDefaultPosition, wxSize(100,-1), 0); - - bSizer174->Add(m_checkPreCallAny, 0, wxALL, 5); - - m_checkPreRaise = new wxCheckBox(this, wxID_ANY, wxT("RAISE"), wxDefaultPosition, wxSize(100,-1), 0); - - bSizer174->Add(m_checkPreRaise, 0, wxALL, 5); - - m_checkPreRaiseAny = new wxCheckBox(this, wxID_ANY, wxT("RAISE ANY"), wxDefaultPosition, wxSize(100,-1), 0); - - bSizer174->Add(m_checkPreRaiseAny, 0, wxALL, 5); - - this->SetSizer(bSizer174); - this->Layout(); - m_statusBar = this->CreateStatusBar(1, wxST_SIZEGRIP, wxID_ANY); - - // Connect Events - this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CPokerDialogBase::OnClose)); - this->Connect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_LEFT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_RIGHT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_MOTION, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Connect(wxEVT_PAINT, wxPaintEventHandler(CPokerDialogBase::OnPaint)); - this->Connect(wxEVT_SIZE, wxSizeEventHandler(CPokerDialogBase::OnSize)); - m_checkSitOut->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckSitOut), NULL, this); - m_buttonDealHand->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonDealHand), NULL, this); - m_buttonFold->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonFold), NULL, this); - m_buttonCall->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonCall), NULL, this); - m_buttonRaise->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonRaise), NULL, this); - m_buttonLeaveTable->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonLeaveTable), NULL, this); - m_textDitchPlayer->Connect(wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler(CPokerDialogBase::OnDitchPlayer), NULL, this); - m_checkPreFold->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreFold), NULL, this); - m_checkPreCall->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCall), NULL, this); - m_checkPreCallAny->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCallAny), NULL, this); - m_checkPreRaise->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaise), NULL, this); - m_checkPreRaiseAny->Connect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaiseAny), NULL, this); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer174; + bSizer174 = new wxBoxSizer( wxVERTICAL ); + + m_checkSitOut = new wxCheckBox( this, wxID_ANY, wxT("Deal Me Out"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer174->Add( m_checkSitOut, 0, wxALL, 5 ); + + m_buttonDealHand = new wxButton( this, wxID_DEALHAND, wxT("&Deal Hand"), wxDefaultPosition, wxSize( 150,25 ), 0 ); + bSizer174->Add( m_buttonDealHand, 0, wxALL, 5 ); + + m_buttonFold = new wxButton( this, wxID_FOLD, wxT("&Fold"), wxDefaultPosition, wxSize( 80,25 ), 0 ); + bSizer174->Add( m_buttonFold, 0, wxALL, 5 ); + + m_buttonCall = new wxButton( this, wxID_CALL, wxT("&Call"), wxDefaultPosition, wxSize( 80,25 ), 0 ); + bSizer174->Add( m_buttonCall, 0, wxALL, 5 ); + + m_buttonRaise = new wxButton( this, wxID_RAISE, wxT("&Raise"), wxDefaultPosition, wxSize( 80,25 ), 0 ); + bSizer174->Add( m_buttonRaise, 0, wxALL, 5 ); + + m_buttonLeaveTable = new wxButton( this, wxID_LEAVETABLE, wxT("&Leave Table"), wxDefaultPosition, wxSize( 90,25 ), 0 ); + bSizer174->Add( m_buttonLeaveTable, 0, wxALL, 5 ); + + m_textDitchPlayer = new wxTextCtrl( this, wxID_DITCHPLAYER, wxEmptyString, wxDefaultPosition, wxSize( 45,-1 ), wxTE_PROCESS_ENTER ); + bSizer174->Add( m_textDitchPlayer, 0, wxALL, 5 ); + + m_checkPreFold = new wxCheckBox( this, wxID_ANY, wxT("FOLD"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); + + bSizer174->Add( m_checkPreFold, 0, wxALL, 5 ); + + m_checkPreCall = new wxCheckBox( this, wxID_ANY, wxT("CALL"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); + + bSizer174->Add( m_checkPreCall, 0, wxALL, 5 ); + + m_checkPreCallAny = new wxCheckBox( this, wxID_ANY, wxT("CALL ANY"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); + + bSizer174->Add( m_checkPreCallAny, 0, wxALL, 5 ); + + m_checkPreRaise = new wxCheckBox( this, wxID_ANY, wxT("RAISE"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); + + bSizer174->Add( m_checkPreRaise, 0, wxALL, 5 ); + + m_checkPreRaiseAny = new wxCheckBox( this, wxID_ANY, wxT("RAISE ANY"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); + + bSizer174->Add( m_checkPreRaiseAny, 0, wxALL, 5 ); + + this->SetSizer( bSizer174 ); + this->Layout(); + m_statusBar = this->CreateStatusBar( 1, wxST_SIZEGRIP, wxID_ANY ); + + // Connect Events + this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CPokerDialogBase::OnClose ) ); + this->Connect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_LEFT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_RIGHT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_MOTION, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Connect( wxEVT_PAINT, wxPaintEventHandler( CPokerDialogBase::OnPaint ) ); + this->Connect( wxEVT_SIZE, wxSizeEventHandler( CPokerDialogBase::OnSize ) ); + m_checkSitOut->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckSitOut ), NULL, this ); + m_buttonDealHand->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonDealHand ), NULL, this ); + m_buttonFold->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonFold ), NULL, this ); + m_buttonCall->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonCall ), NULL, this ); + m_buttonRaise->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonRaise ), NULL, this ); + m_buttonLeaveTable->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonLeaveTable ), NULL, this ); + m_textDitchPlayer->Connect( wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler( CPokerDialogBase::OnDitchPlayer ), NULL, this ); + m_checkPreFold->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreFold ), NULL, this ); + m_checkPreCall->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCall ), NULL, this ); + m_checkPreCallAny->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCallAny ), NULL, this ); + m_checkPreRaise->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaise ), NULL, this ); + m_checkPreRaiseAny->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaiseAny ), NULL, this ); } CPokerDialogBase::~CPokerDialogBase() { - // Disconnect Events - this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CPokerDialogBase::OnClose)); - this->Disconnect(wxEVT_LEFT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_LEFT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_MIDDLE_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_MIDDLE_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_RIGHT_DOWN, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_RIGHT_UP, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_MOTION, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_LEFT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_MIDDLE_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_RIGHT_DCLICK, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_LEAVE_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_ENTER_WINDOW, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_MOUSEWHEEL, wxMouseEventHandler(CPokerDialogBase::OnMouseEvents)); - this->Disconnect(wxEVT_PAINT, wxPaintEventHandler(CPokerDialogBase::OnPaint)); - this->Disconnect(wxEVT_SIZE, wxSizeEventHandler(CPokerDialogBase::OnSize)); - m_checkSitOut->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckSitOut), NULL, this); - m_buttonDealHand->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonDealHand), NULL, this); - m_buttonFold->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonFold), NULL, this); - m_buttonCall->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonCall), NULL, this); - m_buttonRaise->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonRaise), NULL, this); - m_buttonLeaveTable->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnButtonLeaveTable), NULL, this); - m_textDitchPlayer->Disconnect(wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler(CPokerDialogBase::OnDitchPlayer), NULL, this); - m_checkPreFold->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreFold), NULL, this); - m_checkPreCall->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCall), NULL, this); - m_checkPreCallAny->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreCallAny), NULL, this); - m_checkPreRaise->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaise), NULL, this); - m_checkPreRaiseAny->Disconnect(wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler(CPokerDialogBase::OnCheckPreRaiseAny), NULL, this); + // Disconnect Events + this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CPokerDialogBase::OnClose ) ); + this->Disconnect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_LEFT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_RIGHT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MOTION, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); + this->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CPokerDialogBase::OnPaint ) ); + this->Disconnect( wxEVT_SIZE, wxSizeEventHandler( CPokerDialogBase::OnSize ) ); + m_checkSitOut->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckSitOut ), NULL, this ); + m_buttonDealHand->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonDealHand ), NULL, this ); + m_buttonFold->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonFold ), NULL, this ); + m_buttonCall->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonCall ), NULL, this ); + m_buttonRaise->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonRaise ), NULL, this ); + m_buttonLeaveTable->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonLeaveTable ), NULL, this ); + m_textDitchPlayer->Disconnect( wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler( CPokerDialogBase::OnDitchPlayer ), NULL, this ); + m_checkPreFold->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreFold ), NULL, this ); + m_checkPreCall->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCall ), NULL, this ); + m_checkPreCallAny->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCallAny ), NULL, this ); + m_checkPreRaise->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaise ), NULL, this ); + m_checkPreRaiseAny->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaiseAny ), NULL, this ); } -CGetTextFromUserDialogBase::CGetTextFromUserDialogBase(wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxDialog(parent, id, title, pos, size, style) +CGetTextFromUserDialogBase::CGetTextFromUserDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { - this->SetSizeHints(wxDefaultSize, wxDefaultSize); - - wxBoxSizer* bSizer79; - bSizer79 = new wxBoxSizer(wxVERTICAL); - - wxBoxSizer* bSizer81; - bSizer81 = new wxBoxSizer(wxVERTICAL); - - - bSizer81->Add(0, 0, 1, wxEXPAND, 5); - - m_staticTextMessage1 = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_staticTextMessage1->Wrap(-1); - bSizer81->Add(m_staticTextMessage1, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - m_textCtrl1 = new wxTextCtrl(this, wxID_TEXTCTRL, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER); - bSizer81->Add(m_textCtrl1, 0, wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL, 5); - - m_staticTextMessage2 = new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - m_staticTextMessage2->Wrap(-1); - m_staticTextMessage2->Hide(); - - bSizer81->Add(m_staticTextMessage2, 0, wxTOP|wxRIGHT|wxLEFT, 5); - - m_textCtrl2 = new wxTextCtrl(this, wxID_TEXTCTRL, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER); - m_textCtrl2->Hide(); - - bSizer81->Add(m_textCtrl2, 0, wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL, 5); - - - bSizer81->Add(0, 0, 1, wxEXPAND, 5); - - bSizer79->Add(bSizer81, 1, wxEXPAND|wxALL, 10); - - wxBoxSizer* bSizer80; - bSizer80 = new wxBoxSizer(wxHORIZONTAL); - - - bSizer80->Add(0, 0, 1, wxEXPAND, 5); - - m_buttonOK = new wxButton(this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize(-1,-1), 0); - m_buttonOK->SetMinSize(wxSize(85,25)); - - bSizer80->Add(m_buttonOK, 0, wxALL, 5); - - m_buttonCancel = new wxButton(this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0); - m_buttonCancel->SetMinSize(wxSize(85,25)); + this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + + wxBoxSizer* bSizer79; + bSizer79 = new wxBoxSizer( wxVERTICAL ); + + wxBoxSizer* bSizer81; + bSizer81 = new wxBoxSizer( wxVERTICAL ); + + + bSizer81->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_staticTextMessage1 = new wxStaticText( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMessage1->Wrap( -1 ); + bSizer81->Add( m_staticTextMessage1, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_textCtrl1 = new wxTextCtrl( this, wxID_TEXTCTRL, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER ); + bSizer81->Add( m_textCtrl1, 0, wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL, 5 ); + + m_staticTextMessage2 = new wxStaticText( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMessage2->Wrap( -1 ); + m_staticTextMessage2->Hide(); + + bSizer81->Add( m_staticTextMessage2, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + + m_textCtrl2 = new wxTextCtrl( this, wxID_TEXTCTRL, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER ); + m_textCtrl2->Hide(); + + bSizer81->Add( m_textCtrl2, 0, wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL, 5 ); + + + bSizer81->Add( 0, 0, 1, wxEXPAND, 5 ); + + bSizer79->Add( bSizer81, 1, wxEXPAND|wxALL, 10 ); + + wxBoxSizer* bSizer80; + bSizer80 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer80->Add( 0, 0, 1, wxEXPAND, 5 ); + + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonOK->SetMinSize( wxSize( 85,25 ) ); + + bSizer80->Add( m_buttonOK, 0, wxALL, 5 ); + + m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); + + bSizer80->Add( m_buttonCancel, 0, wxALL, 5 ); + + bSizer79->Add( bSizer80, 0, wxEXPAND, 5 ); + + this->SetSizer( bSizer79 ); + this->Layout(); + + // Connect Events + this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CGetTextFromUserDialogBase::OnClose ) ); + m_textCtrl1->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CGetTextFromUserDialogBase::OnKeyDown ), NULL, this ); + m_textCtrl2->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CGetTextFromUserDialogBase::OnKeyDown ), NULL, this ); + m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CGetTextFromUserDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CGetTextFromUserDialogBase::OnButtonCancel ), NULL, this ); +} - bSizer80->Add(m_buttonCancel, 0, wxALL, 5); +CGetTextFromUserDialogBase::~CGetTextFromUserDialogBase() +{ + // Disconnect Events + this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CGetTextFromUserDialogBase::OnClose ) ); + m_textCtrl1->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CGetTextFromUserDialogBase::OnKeyDown ), NULL, this ); + m_textCtrl2->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CGetTextFromUserDialogBase::OnKeyDown ), NULL, this ); + m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CGetTextFromUserDialogBase::OnButtonOK ), NULL, this ); + m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CGetTextFromUserDialogBase::OnButtonCancel ), NULL, this ); +} - bSizer79->Add(bSizer80, 0, wxEXPAND, 5); +COptionsPanelBitcoinBase::COptionsPanelBitcoinBase( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style ) : wxPanel( parent, id, pos, size, style ) +{ + wxBoxSizer* bSizer62; + bSizer62 = new wxBoxSizer( wxVERTICAL ); + + + bSizer62->Add( 0, 20, 0, wxEXPAND, 5 ); + + m_staticText32 = new wxStaticText( this, wxID_ANY, wxT("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText32->Wrap( -1 ); + bSizer62->Add( m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + wxBoxSizer* bSizer56; + bSizer56 = new wxBoxSizer( wxHORIZONTAL ); + + m_staticText31 = new wxStaticText( this, wxID_ANY, wxT("Transaction fee:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText31->Wrap( -1 ); + bSizer56->Add( m_staticText31, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_textCtrlTransactionFee = new wxTextCtrl( this, wxID_TRANSACTIONFEE, wxEmptyString, wxDefaultPosition, wxSize( 70,-1 ), 0 ); + bSizer56->Add( m_textCtrlTransactionFee, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + bSizer62->Add( bSizer56, 0, wxEXPAND, 5 ); + + this->SetSizer( bSizer62 ); + this->Layout(); + bSizer62->Fit( this ); + + // Connect Events + m_textCtrlTransactionFee->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsPanelBitcoinBase::OnKillFocusTransactionFee ), NULL, this ); +} - this->SetSizer(bSizer79); - this->Layout(); +COptionsPanelBitcoinBase::~COptionsPanelBitcoinBase() +{ + // Disconnect Events + m_textCtrlTransactionFee->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsPanelBitcoinBase::OnKillFocusTransactionFee ), NULL, this ); +} - // Connect Events - this->Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CGetTextFromUserDialogBase::OnClose)); - m_textCtrl1->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); - m_textCtrl2->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); - m_buttonOK->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonCancel), NULL, this); +COptionsPanelUIBase::COptionsPanelUIBase( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style ) : wxPanel( parent, id, pos, size, style ) +{ + wxBoxSizer* bSizer57; + bSizer57 = new wxBoxSizer( wxVERTICAL ); + + + bSizer57->Add( 0, 20, 1, wxEXPAND, 5 ); + + m_checkMinToTray = new wxCheckBox( this, wxID_MINTOTRAY, wxT("Minimize to tray"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer57->Add( m_checkMinToTray, 0, wxALL, 5 ); + + m_checkCloseToTray = new wxCheckBox( this, wxID_ANY, wxT("Close to tray"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer57->Add( m_checkCloseToTray, 0, wxALL, 5 ); + + m_checkStartOnSysBoot = new wxCheckBox( this, wxID_ANY, wxT("Start on system boot"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer57->Add( m_checkStartOnSysBoot, 0, wxALL, 5 ); + + m_checkAskBeforeClosing = new wxCheckBox( this, wxID_ANY, wxT("Ask before closing"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer57->Add( m_checkAskBeforeClosing, 0, wxALL, 5 ); + + m_checkAlwaysShowTray = new wxCheckBox( this, wxID_ANY, wxT("Always show tray icon"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer57->Add( m_checkAlwaysShowTray, 0, wxALL, 5 ); + + this->SetSizer( bSizer57 ); + this->Layout(); + bSizer57->Fit( this ); } -CGetTextFromUserDialogBase::~CGetTextFromUserDialogBase() +COptionsPanelUIBase::~COptionsPanelUIBase() { - // Disconnect Events - this->Disconnect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(CGetTextFromUserDialogBase::OnClose)); - m_textCtrl1->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); - m_textCtrl2->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(CGetTextFromUserDialogBase::OnKeyDown), NULL, this); - m_buttonOK->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonOK), NULL, this); - m_buttonCancel->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(CGetTextFromUserDialogBase::OnButtonCancel), NULL, this); } diff --git a/uibase.h b/uibase.h index bfcd8ecc..61e05522 100644 --- a/uibase.h +++ b/uibase.h @@ -1,7 +1,3 @@ -// Copyright (c) 2009 Satoshi Nakamoto -// Distributed under the MIT/X11 software license, see the accompanying -// file license.txt or http://www.opensource.org/licenses/mit-license.php. - /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -34,11 +30,12 @@ #include #include #include +#include +#include #include #include #include #include -#include #include /////////////////////////////////////////////////////////////////////////// @@ -50,674 +47,721 @@ #define wxID_TEXTCTRLADDRESS 1004 #define wxID_BUTTONCOPY 1005 #define wxID_BUTTONCHANGE 1006 -#define wxID_TRANSACTIONFEE 1007 -#define wxID_TEXTCTRLPAYTO 1008 -#define wxID_BUTTONPASTE 1009 -#define wxID_BUTTONADDRESSBOOK 1010 -#define wxID_TEXTCTRLAMOUNT 1011 -#define wxID_CHOICETRANSFERTYPE 1012 -#define wxID_LISTCTRL 1013 -#define wxID_BUTTONRENAME 1014 -#define wxID_BUTTONNEW 1015 -#define wxID_BUTTONEDIT 1016 -#define wxID_BUTTONDELETE 1017 -#define wxID_DEL0 1018 -#define wxID_DEL1 1019 -#define wxID_DEL2 1020 -#define wxID_DEL3 1021 -#define wxID_DEL4 1022 -#define wxID_DEL5 1023 -#define wxID_DEL6 1024 -#define wxID_DEL7 1025 -#define wxID_DEL8 1026 -#define wxID_DEL9 1027 -#define wxID_DEL10 1028 -#define wxID_DEL11 1029 -#define wxID_DEL12 1030 -#define wxID_DEL13 1031 -#define wxID_DEL14 1032 -#define wxID_DEL15 1033 -#define wxID_DEL16 1034 -#define wxID_DEL17 1035 -#define wxID_DEL18 1036 -#define wxID_DEL19 1037 -#define wxID_BUTTONPREVIEW 1038 -#define wxID_BUTTONSAMPLE 1039 -#define wxID_CANCEL2 1040 -#define wxID_BUTTONBACK 1041 -#define wxID_BUTTONNEXT 1042 -#define wxID_SUBMIT 1043 -#define wxID_OPENNEWTABLE 1044 -#define wxID_DEALHAND 1045 -#define wxID_FOLD 1046 -#define wxID_CALL 1047 -#define wxID_RAISE 1048 -#define wxID_LEAVETABLE 1049 -#define wxID_DITCHPLAYER 1050 -#define wxID_TEXTCTRL 1051 +#define wxID_TEXTCTRLPAYTO 1007 +#define wxID_BUTTONPASTE 1008 +#define wxID_BUTTONADDRESSBOOK 1009 +#define wxID_TEXTCTRLAMOUNT 1010 +#define wxID_CHOICETRANSFERTYPE 1011 +#define wxID_LISTCTRL 1012 +#define wxID_BUTTONRENAME 1013 +#define wxID_BUTTONNEW 1014 +#define wxID_BUTTONEDIT 1015 +#define wxID_BUTTONDELETE 1016 +#define wxID_DEL0 1017 +#define wxID_DEL1 1018 +#define wxID_DEL2 1019 +#define wxID_DEL3 1020 +#define wxID_DEL4 1021 +#define wxID_DEL5 1022 +#define wxID_DEL6 1023 +#define wxID_DEL7 1024 +#define wxID_DEL8 1025 +#define wxID_DEL9 1026 +#define wxID_DEL10 1027 +#define wxID_DEL11 1028 +#define wxID_DEL12 1029 +#define wxID_DEL13 1030 +#define wxID_DEL14 1031 +#define wxID_DEL15 1032 +#define wxID_DEL16 1033 +#define wxID_DEL17 1034 +#define wxID_DEL18 1035 +#define wxID_DEL19 1036 +#define wxID_BUTTONPREVIEW 1037 +#define wxID_BUTTONSAMPLE 1038 +#define wxID_CANCEL2 1039 +#define wxID_BUTTONBACK 1040 +#define wxID_BUTTONNEXT 1041 +#define wxID_SUBMIT 1042 +#define wxID_OPENNEWTABLE 1043 +#define wxID_DEALHAND 1044 +#define wxID_FOLD 1045 +#define wxID_CALL 1046 +#define wxID_RAISE 1047 +#define wxID_LEAVETABLE 1048 +#define wxID_DITCHPLAYER 1049 +#define wxID_TEXTCTRL 1050 +#define wxID_TRANSACTIONFEE 1051 +#define wxID_MINTOTRAY 1052 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase /////////////////////////////////////////////////////////////////////////////// -class CMainFrameBase : public wxFrame +class CMainFrameBase : public wxFrame { -private: - -protected: - wxMenuBar* m_menubar; - wxMenu* m_menuFile; - wxMenu* m_menuHelp; - wxToolBar* m_toolBar; - wxStatusBar* m_statusBar; - - wxStaticText* m_staticText32; - wxTextCtrl* m_textCtrlAddress; - wxButton* m_buttonCopy; - wxButton* m_button91; - - wxPanel* m_panel14; - wxStaticText* m_staticText41; - wxStaticText* m_staticTextBalance; - - wxChoice* m_choiceFilter; - wxNotebook* m_notebook; - wxPanel* m_panel7; - wxPanel* m_panel9; - wxPanel* m_panel8; - wxPanel* m_panel10; - wxPanel* m_panel11; - - // Virtual event handlers, overide them in your derived class - virtual void OnClose(wxCloseEvent& event){ event.Skip(); } - virtual void OnIdle(wxIdleEvent& event){ event.Skip(); } - virtual void OnMouseEvents(wxMouseEvent& event){ event.Skip(); } - virtual void OnPaint(wxPaintEvent& event){ event.Skip(); } - virtual void OnMenuFileExit(wxCommandEvent& event){ event.Skip(); } - virtual void OnMenuOptionsGenerate(wxCommandEvent& event){ event.Skip(); } - virtual void OnMenuOptionsChangeYourAddress(wxCommandEvent& event){ event.Skip(); } - virtual void OnMenuOptionsOptions(wxCommandEvent& event){ event.Skip(); } - virtual void OnMenuHelpAbout(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonSend(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonAddressBook(wxCommandEvent& event){ event.Skip(); } - virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } - virtual void OnMouseEventsAddress(wxMouseEvent& event){ event.Skip(); } - virtual void OnSetFocusAddress(wxFocusEvent& event){ event.Skip(); } - virtual void OnButtonCopy(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonChange(wxCommandEvent& event){ event.Skip(); } - virtual void OnListColBeginDrag(wxListEvent& event){ event.Skip(); } - virtual void OnListItemActivatedAllTransactions(wxListEvent& event){ event.Skip(); } - virtual void OnPaintListCtrl(wxPaintEvent& event){ event.Skip(); } - virtual void OnListItemActivatedOrdersSent(wxListEvent& event){ event.Skip(); } - virtual void OnListItemActivatedProductsSent(wxListEvent& event){ event.Skip(); } - virtual void OnListItemActivatedOrdersReceived(wxListEvent& event){ event.Skip(); } - - -public: - wxMenu* m_menuOptions; - wxListCtrl* m_listCtrl; - wxListCtrl* m_listCtrlEscrows; - wxListCtrl* m_listCtrlOrdersSent; - wxListCtrl* m_listCtrlProductsSent; - wxListCtrl* m_listCtrlOrdersReceived; - CMainFrameBase(wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(705,484), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); - ~CMainFrameBase(); - + private: + + protected: + wxMenuBar* m_menubar; + wxMenu* m_menuFile; + wxMenu* m_menuHelp; + wxToolBar* m_toolBar; + wxStatusBar* m_statusBar; + + wxStaticText* m_staticText32; + wxTextCtrl* m_textCtrlAddress; + wxButton* m_buttonCopy; + wxButton* m_button91; + + wxPanel* m_panel14; + wxStaticText* m_staticText41; + wxStaticText* m_staticTextBalance; + + wxChoice* m_choiceFilter; + wxNotebook* m_notebook; + wxPanel* m_panel7; + wxPanel* m_panel9; + wxPanel* m_panel8; + wxPanel* m_panel10; + wxPanel* m_panel11; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } + virtual void OnIconize( wxIconizeEvent& event ){ event.Skip(); } + virtual void OnIdle( wxIdleEvent& event ){ event.Skip(); } + virtual void OnMouseEvents( wxMouseEvent& event ){ event.Skip(); } + virtual void OnPaint( wxPaintEvent& event ){ event.Skip(); } + virtual void OnMenuFileExit( wxCommandEvent& event ){ event.Skip(); } + virtual void OnMenuOptionsGenerate( wxCommandEvent& event ){ event.Skip(); } + virtual void OnUpdateMenuGenerate( wxUpdateUIEvent& event ){ event.Skip(); } + virtual void OnMenuOptionsChangeYourAddress( wxCommandEvent& event ){ event.Skip(); } + virtual void OnMenuOptionsOptions( wxCommandEvent& event ){ event.Skip(); } + virtual void OnMenuHelpAbout( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonSend( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonAddressBook( wxCommandEvent& event ){ event.Skip(); } + virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } + virtual void OnMouseEventsAddress( wxMouseEvent& event ){ event.Skip(); } + virtual void OnSetFocusAddress( wxFocusEvent& event ){ event.Skip(); } + virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonChange( wxCommandEvent& event ){ event.Skip(); } + virtual void OnListColBeginDrag( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemActivatedAllTransactions( wxListEvent& event ){ event.Skip(); } + virtual void OnPaintListCtrl( wxPaintEvent& event ){ event.Skip(); } + virtual void OnListItemActivatedOrdersSent( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemActivatedProductsSent( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemActivatedOrdersReceived( wxListEvent& event ){ event.Skip(); } + + + public: + wxMenu* m_menuOptions; + wxListCtrl* m_listCtrl; + wxListCtrl* m_listCtrlEscrows; + wxListCtrl* m_listCtrlOrdersSent; + wxListCtrl* m_listCtrlProductsSent; + wxListCtrl* m_listCtrlOrdersReceived; + CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 705,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + ~CMainFrameBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CTxDetailsDialogBase /////////////////////////////////////////////////////////////////////////////// -class CTxDetailsDialogBase : public wxDialog +class CTxDetailsDialogBase : public wxDialog { -private: - -protected: - wxHtmlWindow* m_htmlWin; - wxButton* m_buttonOK; - - // Virtual event handlers, overide them in your derived class - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - - -public: - CTxDetailsDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Transaction Details"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(620,450), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); - ~CTxDetailsDialogBase(); - + private: + + protected: + wxHtmlWindow* m_htmlWin; + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + + + public: + CTxDetailsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Transaction Details"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 620,450 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + ~CTxDetailsDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class COptionsDialogBase /////////////////////////////////////////////////////////////////////////////// -class COptionsDialogBase : public wxDialog +class COptionsDialogBase : public wxDialog { -private: - -protected: - - wxStaticText* m_staticText32; - wxStaticText* m_staticText31; - wxTextCtrl* m_textCtrlTransactionFee; - wxButton* m_buttonOK; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnKillFocusTransactionFee(wxFocusEvent& event){ event.Skip(); } - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - COptionsDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Options"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(500,261), long style = wxDEFAULT_DIALOG_STYLE); - ~COptionsDialogBase(); - + private: + + protected: + wxBoxSizer* panelSizer; + wxTreeCtrl* m_treeCtrl; + wxStaticLine* m_staticline1; + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + wxButton* m_buttonApply; + + // Virtual event handlers, overide them in your derived class + virtual void MenuSelChanged( wxTreeEvent& event ){ event.Skip(); } + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + COptionsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Options"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 555,377 ), long style = wxDEFAULT_DIALOG_STYLE ); + ~COptionsDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CAboutDialogBase /////////////////////////////////////////////////////////////////////////////// -class CAboutDialogBase : public wxDialog +class CAboutDialogBase : public wxDialog { -private: - -protected: - - - wxStaticText* m_staticText40; - - wxStaticText* m_staticTextMain; - - - wxButton* m_buttonOK; - - // Virtual event handlers, overide them in your derived class - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - - -public: - wxStaticText* m_staticTextVersion; - CAboutDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("About Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(507,298), long style = wxDEFAULT_DIALOG_STYLE); - ~CAboutDialogBase(); - + private: + + protected: + + + wxStaticText* m_staticText40; + + wxStaticText* m_staticTextMain; + + + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + + + public: + wxStaticText* m_staticTextVersion; + CAboutDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("About Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 507,298 ), long style = wxDEFAULT_DIALOG_STYLE ); + ~CAboutDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CSendDialogBase /////////////////////////////////////////////////////////////////////////////// -class CSendDialogBase : public wxDialog +class CSendDialogBase : public wxDialog { -private: - -protected: - - - wxStaticText* m_staticText14; - - wxStaticBitmap* m_bitmapCheckMark; - wxStaticText* m_staticText36; - wxTextCtrl* m_textCtrlAddress; - wxButton* m_buttonPaste; - wxButton* m_buttonAddress; - wxStaticText* m_staticText19; - wxTextCtrl* m_textCtrlAmount; - wxStaticText* m_staticText20; - wxChoice* m_choiceTransferType; - - - wxStaticText* m_staticTextFrom; - wxTextCtrl* m_textCtrlFrom; - wxStaticText* m_staticTextMessage; - wxTextCtrl* m_textCtrlMessage; - - wxButton* m_buttonSend; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } - virtual void OnTextAddress(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonPaste(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonAddressBook(wxCommandEvent& event){ event.Skip(); } - virtual void OnKillFocusAmount(wxFocusEvent& event){ event.Skip(); } - virtual void OnButtonSend(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - CSendDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Send Coins"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(675,312), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); - ~CSendDialogBase(); - + private: + + protected: + + + wxStaticText* m_staticText14; + + wxStaticBitmap* m_bitmapCheckMark; + wxStaticText* m_staticText36; + wxTextCtrl* m_textCtrlAddress; + wxButton* m_buttonPaste; + wxButton* m_buttonAddress; + wxStaticText* m_staticText19; + wxTextCtrl* m_textCtrlAmount; + wxStaticText* m_staticText20; + wxChoice* m_choiceTransferType; + + + wxStaticText* m_staticTextFrom; + wxTextCtrl* m_textCtrlFrom; + wxStaticText* m_staticTextMessage; + wxTextCtrl* m_textCtrlMessage; + + wxButton* m_buttonSend; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } + virtual void OnTextAddress( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonPaste( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonAddressBook( wxCommandEvent& event ){ event.Skip(); } + virtual void OnKillFocusAmount( wxFocusEvent& event ){ event.Skip(); } + virtual void OnButtonSend( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + CSendDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Send Coins"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 675,312 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + ~CSendDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CSendingDialogBase /////////////////////////////////////////////////////////////////////////////// -class CSendingDialogBase : public wxDialog +class CSendingDialogBase : public wxDialog { -private: - -protected: - wxStaticText* m_staticTextSending; - wxTextCtrl* m_textCtrlStatus; - - wxButton* m_buttonOK; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnClose(wxCloseEvent& event){ event.Skip(); } - virtual void OnPaint(wxPaintEvent& event){ event.Skip(); } - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - CSendingDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Sending..."), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(442,151), long style = wxDEFAULT_DIALOG_STYLE); - ~CSendingDialogBase(); - + private: + + protected: + wxStaticText* m_staticTextSending; + wxTextCtrl* m_textCtrlStatus; + + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } + virtual void OnPaint( wxPaintEvent& event ){ event.Skip(); } + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + CSendingDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Sending..."), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 442,151 ), long style = wxDEFAULT_DIALOG_STYLE ); + ~CSendingDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CYourAddressDialogBase /////////////////////////////////////////////////////////////////////////////// -class CYourAddressDialogBase : public wxDialog +class CYourAddressDialogBase : public wxDialog { -private: - -protected: - - wxStaticText* m_staticText45; - wxListCtrl* m_listCtrl; - - wxButton* m_buttonRename; - wxButton* m_buttonNew; - wxButton* m_buttonCopy; - wxButton* m_buttonOK; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnClose(wxCloseEvent& event){ event.Skip(); } - virtual void OnListEndLabelEdit(wxListEvent& event){ event.Skip(); } - virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } - virtual void OnListItemSelected(wxListEvent& event){ event.Skip(); } - virtual void OnButtonRename(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonNew(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCopy(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - CYourAddressDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Your Bitcoin Addresses"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(610,390), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); - ~CYourAddressDialogBase(); - + private: + + protected: + + wxStaticText* m_staticText45; + wxListCtrl* m_listCtrl; + + wxButton* m_buttonRename; + wxButton* m_buttonNew; + wxButton* m_buttonCopy; + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } + virtual void OnListEndLabelEdit( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemSelected( wxListEvent& event ){ event.Skip(); } + virtual void OnButtonRename( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonNew( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + CYourAddressDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Your Bitcoin Addresses"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 610,390 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + ~CYourAddressDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CAddressBookDialogBase /////////////////////////////////////////////////////////////////////////////// -class CAddressBookDialogBase : public wxDialog +class CAddressBookDialogBase : public wxDialog { -private: - -protected: - - wxStaticText* m_staticText55; - wxListCtrl* m_listCtrl; - - wxButton* m_buttonEdit; - wxButton* m_buttonNew; - wxButton* m_buttonDelete; - wxButton* m_buttonOK; - - // Virtual event handlers, overide them in your derived class - virtual void OnClose(wxCloseEvent& event){ event.Skip(); } - virtual void OnListEndLabelEdit(wxListEvent& event){ event.Skip(); } - virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } - virtual void OnListItemSelected(wxListEvent& event){ event.Skip(); } - virtual void OnButtonEdit(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonNew(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDelete(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - wxButton* m_buttonCancel; - CAddressBookDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Address Book"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(610,390), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); - ~CAddressBookDialogBase(); - + private: + + protected: + + wxStaticText* m_staticText55; + wxListCtrl* m_listCtrl; + + wxButton* m_buttonEdit; + wxButton* m_buttonNew; + wxButton* m_buttonDelete; + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } + virtual void OnListEndLabelEdit( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemSelected( wxListEvent& event ){ event.Skip(); } + virtual void OnButtonEdit( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonNew( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDelete( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + wxButton* m_buttonCancel; + CAddressBookDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Address Book"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 610,390 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + ~CAddressBookDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CProductsDialogBase /////////////////////////////////////////////////////////////////////////////// -class CProductsDialogBase : public wxDialog +class CProductsDialogBase : public wxDialog { -private: - -protected: - wxComboBox* m_comboBoxCategory; - wxTextCtrl* m_textCtrlSearch; - wxButton* m_buttonSearch; - wxListCtrl* m_listCtrl; - - // Virtual event handlers, overide them in your derived class - virtual void OnCombobox(wxCommandEvent& event){ event.Skip(); } - virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } - virtual void OnButtonSearch(wxCommandEvent& event){ event.Skip(); } - virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } - - -public: - CProductsDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Marketplace"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(708,535), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER); - ~CProductsDialogBase(); - + private: + + protected: + wxComboBox* m_comboBoxCategory; + wxTextCtrl* m_textCtrlSearch; + wxButton* m_buttonSearch; + wxListCtrl* m_listCtrl; + + // Virtual event handlers, overide them in your derived class + virtual void OnCombobox( wxCommandEvent& event ){ event.Skip(); } + virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } + virtual void OnButtonSearch( wxCommandEvent& event ){ event.Skip(); } + virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } + + + public: + CProductsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Marketplace"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 708,535 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + ~CProductsDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CEditProductDialogBase /////////////////////////////////////////////////////////////////////////////// -class CEditProductDialogBase : public wxFrame +class CEditProductDialogBase : public wxFrame { -private: - -protected: - wxScrolledWindow* m_scrolledWindow; - wxStaticText* m_staticText106; - wxComboBox* m_comboBoxCategory; - wxStaticText* m_staticText108; - wxTextCtrl* m_textCtrlTitle; - wxStaticText* m_staticText107; - wxTextCtrl* m_textCtrlPrice; - wxStaticText* m_staticText22; - wxTextCtrl* m_textCtrlDescription; - wxStaticText* m_staticText23; - wxTextCtrl* m_textCtrlInstructions; - wxStaticText* m_staticText24; - wxStaticText* m_staticText25; - - wxTextCtrl* m_textCtrlLabel0; - wxTextCtrl* m_textCtrlField0; - wxButton* m_buttonDel0; - wxTextCtrl* m_textCtrlLabel1; - wxTextCtrl* m_textCtrlField1; - wxButton* m_buttonDel1; - wxTextCtrl* m_textCtrlLabel2; - wxTextCtrl* m_textCtrlField2; - wxButton* m_buttonDel2; - wxTextCtrl* m_textCtrlLabel3; - wxTextCtrl* m_textCtrlField3; - wxButton* m_buttonDel3; - wxTextCtrl* m_textCtrlLabel4; - wxTextCtrl* m_textCtrlField4; - wxButton* m_buttonDel4; - wxTextCtrl* m_textCtrlLabel5; - wxTextCtrl* m_textCtrlField5; - wxButton* m_buttonDel5; - wxTextCtrl* m_textCtrlLabel6; - wxTextCtrl* m_textCtrlField6; - wxButton* m_buttonDel6; - wxTextCtrl* m_textCtrlLabel7; - wxTextCtrl* m_textCtrlField7; - wxButton* m_buttonDel7; - wxTextCtrl* m_textCtrlLabel8; - wxTextCtrl* m_textCtrlField8; - wxButton* m_buttonDel8; - wxTextCtrl* m_textCtrlLabel9; - wxTextCtrl* m_textCtrlField9; - wxButton* m_buttonDel9; - wxTextCtrl* m_textCtrlLabel10; - wxTextCtrl* m_textCtrlField10; - wxButton* m_buttonDel10; - wxTextCtrl* m_textCtrlLabel11; - wxTextCtrl* m_textCtrlField11; - wxButton* m_buttonDel11; - wxTextCtrl* m_textCtrlLabel12; - wxTextCtrl* m_textCtrlField12; - wxButton* m_buttonDel12; - wxTextCtrl* m_textCtrlLabel13; - wxTextCtrl* m_textCtrlField13; - wxButton* m_buttonDel13; - wxTextCtrl* m_textCtrlLabel14; - wxTextCtrl* m_textCtrlField14; - wxButton* m_buttonDel14; - wxTextCtrl* m_textCtrlLabel15; - wxTextCtrl* m_textCtrlField15; - wxButton* m_buttonDel15; - wxTextCtrl* m_textCtrlLabel16; - wxTextCtrl* m_textCtrlField16; - wxButton* m_buttonDel16; - wxTextCtrl* m_textCtrlLabel17; - wxTextCtrl* m_textCtrlField17; - wxButton* m_buttonDel17; - wxTextCtrl* m_textCtrlLabel18; - wxTextCtrl* m_textCtrlField18; - wxButton* m_buttonDel18; - wxTextCtrl* m_textCtrlLabel19; - wxTextCtrl* m_textCtrlField19; - wxButton* m_buttonDel19; - wxButton* m_buttonAddField; - wxButton* m_buttonOK; - wxButton* m_buttonPreview; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } - virtual void OnButtonDel0(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel1(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel2(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel3(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel4(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel5(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel6(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel7(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel8(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel9(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel10(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel11(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel12(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel13(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel14(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel15(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel16(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel17(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel18(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDel19(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonAddField(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonSend(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonPreview(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - wxFlexGridSizer* fgSizer5; - CEditProductDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Edit Product"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(660,640), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); - ~CEditProductDialogBase(); - + private: + + protected: + wxScrolledWindow* m_scrolledWindow; + wxStaticText* m_staticText106; + wxComboBox* m_comboBoxCategory; + wxStaticText* m_staticText108; + wxTextCtrl* m_textCtrlTitle; + wxStaticText* m_staticText107; + wxTextCtrl* m_textCtrlPrice; + wxStaticText* m_staticText22; + wxTextCtrl* m_textCtrlDescription; + wxStaticText* m_staticText23; + wxTextCtrl* m_textCtrlInstructions; + wxStaticText* m_staticText24; + wxStaticText* m_staticText25; + + wxTextCtrl* m_textCtrlLabel0; + wxTextCtrl* m_textCtrlField0; + wxButton* m_buttonDel0; + wxTextCtrl* m_textCtrlLabel1; + wxTextCtrl* m_textCtrlField1; + wxButton* m_buttonDel1; + wxTextCtrl* m_textCtrlLabel2; + wxTextCtrl* m_textCtrlField2; + wxButton* m_buttonDel2; + wxTextCtrl* m_textCtrlLabel3; + wxTextCtrl* m_textCtrlField3; + wxButton* m_buttonDel3; + wxTextCtrl* m_textCtrlLabel4; + wxTextCtrl* m_textCtrlField4; + wxButton* m_buttonDel4; + wxTextCtrl* m_textCtrlLabel5; + wxTextCtrl* m_textCtrlField5; + wxButton* m_buttonDel5; + wxTextCtrl* m_textCtrlLabel6; + wxTextCtrl* m_textCtrlField6; + wxButton* m_buttonDel6; + wxTextCtrl* m_textCtrlLabel7; + wxTextCtrl* m_textCtrlField7; + wxButton* m_buttonDel7; + wxTextCtrl* m_textCtrlLabel8; + wxTextCtrl* m_textCtrlField8; + wxButton* m_buttonDel8; + wxTextCtrl* m_textCtrlLabel9; + wxTextCtrl* m_textCtrlField9; + wxButton* m_buttonDel9; + wxTextCtrl* m_textCtrlLabel10; + wxTextCtrl* m_textCtrlField10; + wxButton* m_buttonDel10; + wxTextCtrl* m_textCtrlLabel11; + wxTextCtrl* m_textCtrlField11; + wxButton* m_buttonDel11; + wxTextCtrl* m_textCtrlLabel12; + wxTextCtrl* m_textCtrlField12; + wxButton* m_buttonDel12; + wxTextCtrl* m_textCtrlLabel13; + wxTextCtrl* m_textCtrlField13; + wxButton* m_buttonDel13; + wxTextCtrl* m_textCtrlLabel14; + wxTextCtrl* m_textCtrlField14; + wxButton* m_buttonDel14; + wxTextCtrl* m_textCtrlLabel15; + wxTextCtrl* m_textCtrlField15; + wxButton* m_buttonDel15; + wxTextCtrl* m_textCtrlLabel16; + wxTextCtrl* m_textCtrlField16; + wxButton* m_buttonDel16; + wxTextCtrl* m_textCtrlLabel17; + wxTextCtrl* m_textCtrlField17; + wxButton* m_buttonDel17; + wxTextCtrl* m_textCtrlLabel18; + wxTextCtrl* m_textCtrlField18; + wxButton* m_buttonDel18; + wxTextCtrl* m_textCtrlLabel19; + wxTextCtrl* m_textCtrlField19; + wxButton* m_buttonDel19; + wxButton* m_buttonAddField; + wxButton* m_buttonOK; + wxButton* m_buttonPreview; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } + virtual void OnButtonDel0( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel1( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel2( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel3( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel4( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel5( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel6( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel7( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel8( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel9( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel10( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel11( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel12( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel13( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel14( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel15( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel16( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel17( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel18( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDel19( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonAddField( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonSend( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonPreview( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + wxFlexGridSizer* fgSizer5; + CEditProductDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Edit Product"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 660,640 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + ~CEditProductDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CViewProductDialogBase /////////////////////////////////////////////////////////////////////////////// -class CViewProductDialogBase : public wxFrame +class CViewProductDialogBase : public wxFrame { -private: - -protected: - wxHtmlWindow* m_htmlWinReviews; - wxScrolledWindow* m_scrolledWindow; - wxRichTextCtrl* m_richTextHeading; - wxStaticText* m_staticTextInstructions; - wxButton* m_buttonSubmitForm; - wxButton* m_buttonCancelForm; - wxButton* m_buttonBack; - wxButton* m_buttonNext; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnButtonSubmitForm(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancelForm(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonBack(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonNext(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - CViewProductDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Order Form"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(630,520), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); - ~CViewProductDialogBase(); - + private: + + protected: + wxHtmlWindow* m_htmlWinReviews; + wxScrolledWindow* m_scrolledWindow; + wxRichTextCtrl* m_richTextHeading; + wxStaticText* m_staticTextInstructions; + wxButton* m_buttonSubmitForm; + wxButton* m_buttonCancelForm; + wxButton* m_buttonBack; + wxButton* m_buttonNext; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonSubmitForm( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancelForm( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonBack( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonNext( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + CViewProductDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Order Form"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 630,520 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + ~CViewProductDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CViewOrderDialogBase /////////////////////////////////////////////////////////////////////////////// -class CViewOrderDialogBase : public wxFrame +class CViewOrderDialogBase : public wxFrame { -private: - -protected: - wxHtmlWindow* m_htmlWin; - wxButton* m_buttonOK; - - // Virtual event handlers, overide them in your derived class - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - - -public: - CViewOrderDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("View Order"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(630,520), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); - ~CViewOrderDialogBase(); - + private: + + protected: + wxHtmlWindow* m_htmlWin; + wxButton* m_buttonOK; + + // Virtual event handlers, overide them in your derived class + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + + + public: + CViewOrderDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("View Order"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 630,520 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + ~CViewOrderDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CEditReviewDialogBase /////////////////////////////////////////////////////////////////////////////// -class CEditReviewDialogBase : public wxFrame +class CEditReviewDialogBase : public wxFrame { -private: - -protected: - - wxStaticText* m_staticTextSeller; - - wxStaticText* m_staticText110; - wxChoice* m_choiceStars; - wxStaticText* m_staticText43; - wxTextCtrl* m_textCtrlReview; - wxButton* m_buttonSubmit; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } - virtual void OnButtonSubmit(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - - -public: - CEditReviewDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Enter Review"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(630,440), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL); - ~CEditReviewDialogBase(); - + private: + + protected: + + wxStaticText* m_staticTextSeller; + + wxStaticText* m_staticText110; + wxChoice* m_choiceStars; + wxStaticText* m_staticText43; + wxTextCtrl* m_textCtrlReview; + wxButton* m_buttonSubmit; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } + virtual void OnButtonSubmit( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + CEditReviewDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Enter Review"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 630,440 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + ~CEditReviewDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CPokerLobbyDialogBase /////////////////////////////////////////////////////////////////////////////// -class CPokerLobbyDialogBase : public wxFrame +class CPokerLobbyDialogBase : public wxFrame { -private: - -protected: - wxTreeCtrl* m_treeCtrl; - wxListCtrl* m_listCtrl; - wxButton* m_buttonNewTable; - - // Virtual event handlers, overide them in your derived class - virtual void OnTreeSelChanged(wxTreeEvent& event){ event.Skip(); } - virtual void OnListItemActivated(wxListEvent& event){ event.Skip(); } - virtual void OnListItemSelected(wxListEvent& event){ event.Skip(); } - virtual void OnButtonNewTable(wxCommandEvent& event){ event.Skip(); } - - -public: - CPokerLobbyDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker Lobby"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(586,457), long style = wxDEFAULT_FRAME_STYLE|wxTAB_TRAVERSAL); - ~CPokerLobbyDialogBase(); - + private: + + protected: + wxTreeCtrl* m_treeCtrl; + wxListCtrl* m_listCtrl; + wxButton* m_buttonNewTable; + + // Virtual event handlers, overide them in your derived class + virtual void OnTreeSelChanged( wxTreeEvent& event ){ event.Skip(); } + virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemSelected( wxListEvent& event ){ event.Skip(); } + virtual void OnButtonNewTable( wxCommandEvent& event ){ event.Skip(); } + + + public: + CPokerLobbyDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker Lobby"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 586,457 ), long style = wxDEFAULT_FRAME_STYLE|wxTAB_TRAVERSAL ); + ~CPokerLobbyDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CPokerDialogBase /////////////////////////////////////////////////////////////////////////////// -class CPokerDialogBase : public wxFrame +class CPokerDialogBase : public wxFrame { -private: - -protected: - wxButton* m_buttonDealHand; - wxButton* m_buttonFold; - wxButton* m_buttonCall; - wxButton* m_buttonRaise; - wxButton* m_buttonLeaveTable; - wxTextCtrl* m_textDitchPlayer; - - // Virtual event handlers, overide them in your derived class - virtual void OnClose(wxCloseEvent& event){ event.Skip(); } - virtual void OnMouseEvents(wxMouseEvent& event){ event.Skip(); } - virtual void OnPaint(wxPaintEvent& event){ event.Skip(); } - virtual void OnSize(wxSizeEvent& event){ event.Skip(); } - virtual void OnCheckSitOut(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonDealHand(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonFold(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCall(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonRaise(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonLeaveTable(wxCommandEvent& event){ event.Skip(); } - virtual void OnDitchPlayer(wxCommandEvent& event){ event.Skip(); } - virtual void OnCheckPreFold(wxCommandEvent& event){ event.Skip(); } - virtual void OnCheckPreCall(wxCommandEvent& event){ event.Skip(); } - virtual void OnCheckPreCallAny(wxCommandEvent& event){ event.Skip(); } - virtual void OnCheckPreRaise(wxCommandEvent& event){ event.Skip(); } - virtual void OnCheckPreRaiseAny(wxCommandEvent& event){ event.Skip(); } - - -public: - wxCheckBox* m_checkSitOut; - wxCheckBox* m_checkPreFold; - wxCheckBox* m_checkPreCall; - wxCheckBox* m_checkPreCallAny; - wxCheckBox* m_checkPreRaise; - wxCheckBox* m_checkPreRaiseAny; - wxStatusBar* m_statusBar; - CPokerDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(806,550), long style = wxDEFAULT_FRAME_STYLE|wxFRAME_NO_TASKBAR|wxFULL_REPAINT_ON_RESIZE|wxTAB_TRAVERSAL); - ~CPokerDialogBase(); - + private: + + protected: + wxButton* m_buttonDealHand; + wxButton* m_buttonFold; + wxButton* m_buttonCall; + wxButton* m_buttonRaise; + wxButton* m_buttonLeaveTable; + wxTextCtrl* m_textDitchPlayer; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } + virtual void OnMouseEvents( wxMouseEvent& event ){ event.Skip(); } + virtual void OnPaint( wxPaintEvent& event ){ event.Skip(); } + virtual void OnSize( wxSizeEvent& event ){ event.Skip(); } + virtual void OnCheckSitOut( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonDealHand( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonFold( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCall( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonRaise( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonLeaveTable( wxCommandEvent& event ){ event.Skip(); } + virtual void OnDitchPlayer( wxCommandEvent& event ){ event.Skip(); } + virtual void OnCheckPreFold( wxCommandEvent& event ){ event.Skip(); } + virtual void OnCheckPreCall( wxCommandEvent& event ){ event.Skip(); } + virtual void OnCheckPreCallAny( wxCommandEvent& event ){ event.Skip(); } + virtual void OnCheckPreRaise( wxCommandEvent& event ){ event.Skip(); } + virtual void OnCheckPreRaiseAny( wxCommandEvent& event ){ event.Skip(); } + + + public: + wxCheckBox* m_checkSitOut; + wxCheckBox* m_checkPreFold; + wxCheckBox* m_checkPreCall; + wxCheckBox* m_checkPreCallAny; + wxCheckBox* m_checkPreRaise; + wxCheckBox* m_checkPreRaiseAny; + wxStatusBar* m_statusBar; + CPokerDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 806,550 ), long style = wxDEFAULT_FRAME_STYLE|wxFRAME_NO_TASKBAR|wxFULL_REPAINT_ON_RESIZE|wxTAB_TRAVERSAL ); + ~CPokerDialogBase(); + }; /////////////////////////////////////////////////////////////////////////////// /// Class CGetTextFromUserDialogBase /////////////////////////////////////////////////////////////////////////////// -class CGetTextFromUserDialogBase : public wxDialog +class CGetTextFromUserDialogBase : public wxDialog { -private: - -protected: - - wxStaticText* m_staticTextMessage1; - wxTextCtrl* m_textCtrl1; - wxStaticText* m_staticTextMessage2; - wxTextCtrl* m_textCtrl2; - - - wxButton* m_buttonOK; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnClose(wxCloseEvent& event){ event.Skip(); } - virtual void OnKeyDown(wxKeyEvent& event){ event.Skip(); } - virtual void OnButtonOK(wxCommandEvent& event){ event.Skip(); } - virtual void OnButtonCancel(wxCommandEvent& event){ event.Skip(); } - + private: + + protected: + + wxStaticText* m_staticTextMessage1; + wxTextCtrl* m_textCtrl1; + wxStaticText* m_staticTextMessage2; + wxTextCtrl* m_textCtrl2; + + + wxButton* m_buttonOK; + wxButton* m_buttonCancel; + + // Virtual event handlers, overide them in your derived class + virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } + virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } + virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + + + public: + CGetTextFromUserDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxEmptyString, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 403,138 ), long style = wxDEFAULT_DIALOG_STYLE ); + ~CGetTextFromUserDialogBase(); + +}; -public: - CGetTextFromUserDialogBase(wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxEmptyString, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize(403,138), long style = wxDEFAULT_DIALOG_STYLE); - ~CGetTextFromUserDialogBase(); +/////////////////////////////////////////////////////////////////////////////// +/// Class COptionsPanelBitcoinBase +/////////////////////////////////////////////////////////////////////////////// +class COptionsPanelBitcoinBase : public wxPanel +{ + private: + + protected: + + wxStaticText* m_staticText32; + wxStaticText* m_staticText31; + + // Virtual event handlers, overide them in your derived class + virtual void OnKillFocusTransactionFee( wxFocusEvent& event ){ event.Skip(); } + + + public: + wxTextCtrl* m_textCtrlTransactionFee; + COptionsPanelBitcoinBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( -1,-1 ), long style = wxTAB_TRAVERSAL ); + ~COptionsPanelBitcoinBase(); + +}; +/////////////////////////////////////////////////////////////////////////////// +/// Class COptionsPanelUIBase +/////////////////////////////////////////////////////////////////////////////// +class COptionsPanelUIBase : public wxPanel +{ + private: + + protected: + + + public: + wxCheckBox* m_checkMinToTray; + wxCheckBox* m_checkCloseToTray; + wxCheckBox* m_checkStartOnSysBoot; + wxCheckBox* m_checkAskBeforeClosing; + wxCheckBox* m_checkAlwaysShowTray; + COptionsPanelUIBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( -1,-1 ), long style = wxTAB_TRAVERSAL ); + ~COptionsPanelUIBase(); + }; #endif //__uibase__ diff --git a/uiproject.fbp b/uiproject.fbp index 49b1fefa..ccd8b222 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -18,7 +18,7 @@ 1 0 0 - + wxSYS_COLOUR_BTNFACE @@ -48,7 +48,7 @@ - + OnIconize OnIdle @@ -151,7 +151,7 @@ OnMenuOptionsGenerate - + OnUpdateMenuGenerate @@ -1665,7 +1665,7 @@ - + @@ -1679,7 +1679,7 @@ COptionsDialogBase - 500,261 + 555,377 wxDEFAULT_DIALOG_STYLE Options @@ -1724,28 +1724,18 @@ none 5 - wxEXPAND|wxLEFT + wxEXPAND 1 - bSizer57 - wxVERTICAL - none - - 5 - wxEXPAND - 0 - - 20 - protected - 0 - - + panelSizer + wxHORIZONTAL + protected 5 - wxALIGN_CENTER_VERTICAL|wxALL + wxALL|wxEXPAND 0 - + 1 @@ -1753,20 +1743,18 @@ 0 wxID_ANY - Optional transaction fee you give to the nodes that process your transactions. - m_staticText32 + m_treeCtrl protected - - + 100,-1 + wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT - -1 @@ -1789,126 +1777,79 @@ + + + + + + + + + + + + + + + + + + MenuSelChanged + + + - - 5 - wxEXPAND - 0 - - - bSizer56 - wxHORIZONTAL - none - - 5 - wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Transaction fee: - - - m_staticText31 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_TRANSACTIONFEE - - 0 - - m_textCtrlTransactionFee - protected - - 70,-1 - - - - - - - - - - - - - OnKillFocusTransactionFee - - - - - - - - - - - - - - - - - - - - - - - - - + + + + 5 + wxEXPAND | wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + + + m_staticline1 + protected + + + wxLI_HORIZONTAL + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -2024,6 +1965,58 @@ + + 5 + wxALL + 0 + + + + 0 + 0 + + + 1 + wxID_ANY + Apply + + + m_buttonApply + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -11856,5 +11849,551 @@ + + + + 1 + + + 0 + wxID_ANY + + + COptionsPanelBitcoinBase + + -1,-1 + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer62 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 20 + protected + 0 + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + Optional transaction fee you give to the nodes that process your transactions. + + + m_staticText32 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + bSizer56 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + Transaction fee: + + + m_staticText31 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_TRANSACTIONFEE + + 0 + + m_textCtrlTransactionFee + public + + 70,-1 + + + + + + + + + + + + + OnKillFocusTransactionFee + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + 0 + wxID_ANY + + + COptionsPanelUIBase + + -1,-1 + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer57 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + 20 + protected + 0 + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_MINTOTRAY + Minimize to tray + + + m_checkMinToTray + public + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + Close to tray + + + m_checkCloseToTray + public + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + Start on system boot + + + m_checkStartOnSysBoot + public + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + Ask before closing + + + m_checkAskBeforeClosing + public + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + Always show tray icon + + + m_checkAlwaysShowTray + public + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + From 6d97df0e992dc5015b5327935ab89923d2dc0ba9 Mon Sep 17 00:00:00 2001 From: sirius-m Date: Fri, 2 Oct 2009 10:14:05 +0000 Subject: [PATCH 005/133] startup shortcut works git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@11 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- bugs.txt | 1 - changelog.txt | 2 +- ui.cpp | 75 ++++++++++++++++++++++++++++++++++++--------------- ui.h | 1 + 4 files changed, 55 insertions(+), 24 deletions(-) diff --git a/bugs.txt b/bugs.txt index 348c359e..0f00111e 100644 --- a/bugs.txt +++ b/bugs.txt @@ -1,4 +1,3 @@ Known bugs: -- For some reason, CreateHardLink doesn't add a shortcut to the startup folder - When the program is minimized to tray, double clicking the icon only restores it to the task bar - Window flickers when blocks are added (problem with repainting?) \ No newline at end of file diff --git a/changelog.txt b/changelog.txt index 685eb629..fee4f661 100644 --- a/changelog.txt +++ b/changelog.txt @@ -2,5 +2,5 @@ Changes after 0.1.5: -------------------- + Options dialog layout changed - added the UI options panel + Minimize to tray feature -+ Startup on system boot feature ++ Startup on system boot feature (adds a shortcut to the Startup folder) + Ask before closing \ No newline at end of file diff --git a/ui.cpp b/ui.cpp index 751a50be..e69630d3 100644 --- a/ui.cpp +++ b/ui.cpp @@ -859,18 +859,6 @@ void CMainFrame::OnMenuFileExit(wxCommandEvent& event) Close(true); } -void GenerateBitcoins(bool flag) -{ - fGenerateBitcoins = flag; - nTransactionsUpdated++; - CWalletDB().WriteSetting("fGenerateBitcoins", fGenerateBitcoins); - if (fGenerateBitcoins) - if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) - printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); - - taskBarIcon->UpdateTooltip(); -} - void CMainFrame::OnMenuOptionsGenerate(wxCommandEvent& event) { GenerateBitcoins(event.IsChecked()); @@ -3394,24 +3382,67 @@ void ApplyUISettings() { taskBarIcon->Hide(); // Autostart on system startup? - if (startOnSysBoot) { - // Get the startup folder path - char targetPath[ MAX_PATH ]; - SHGetSpecialFolderPath(0, targetPath, CSIDL_STARTUP, 0); - strcat(targetPath, "\\bitcoin.lnk"); + // Get the startup folder shortcut path + char linkPath[ MAX_PATH ]; + SHGetSpecialFolderPath(0, linkPath, CSIDL_STARTUP, 0); + strcat(linkPath, "\\Bitcoin.lnk"); - // And the current executable path - char currentPath[ MAX_PATH ]; - GetModuleFileName(NULL, currentPath, _MAX_PATH + 1); + // If the shortcut exists already, remove it for updating + remove(linkPath); - // Create the shortcut - CreateHardLink(targetPath, currentPath, NULL); + if (startOnSysBoot) { + CoInitialize(NULL); + // Get the current executable path + char exePath[ MAX_PATH ]; + GetModuleFileName(NULL, exePath, _MAX_PATH + 1); + + HRESULT hres = NULL; + IShellLink* psl = NULL; + // Get a pointer to the IShellLink interface. + hres = CoCreateInstance(CLSID_ShellLink, NULL, + CLSCTX_INPROC_SERVER, IID_IShellLink, + reinterpret_cast(&psl)); + + if (SUCCEEDED(hres)) + { + IPersistFile* ppf = NULL; + // Set the path to the shortcut target + psl->SetPath(exePath); + // Query IShellLink for the IPersistFile interface for + // saving the shortcut in persistent storage. + hres = psl->QueryInterface(IID_IPersistFile, + reinterpret_cast(&ppf)); + if (SUCCEEDED(hres)) + { + WCHAR wsz[MAX_PATH]; + // Ensure that the string is ANSI. + MultiByteToWideChar(CP_ACP, 0, linkPath, -1, + wsz, MAX_PATH); + // Save the link by calling IPersistFile::Save. + hres = ppf->Save(wsz, TRUE); + ppf->Release(); + } + psl->Release(); + } + CoUninitialize(); } } +void GenerateBitcoins(bool flag) +{ + fGenerateBitcoins = flag; + nTransactionsUpdated++; + CWalletDB().WriteSetting("fGenerateBitcoins", fGenerateBitcoins); + if (fGenerateBitcoins) + if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) + printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); + + taskBarIcon->UpdateTooltip(); +} + // randsendtest to bitcoin address diff --git a/ui.h b/ui.h index 5f3897c1..11c88147 100644 --- a/ui.h +++ b/ui.h @@ -28,6 +28,7 @@ extern void CrossThreadCall(int nID, void* pdata); extern void MainFrameRepaint(); extern void Shutdown(void* parg); void ApplyUISettings(); +void GenerateBitcoins(bool flag); // UI settings extern int minimizeToTray; From 429187c6a8816e6df73f69530d0641ac0dcc514e Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sat, 3 Oct 2009 11:52:21 +0000 Subject: [PATCH 006/133] Startup folder shortcut opens the program minimized. Restoring a minimized-to-tray window works correctly. git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@12 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- bugs.txt | 1 - ui.cpp | 5 +++++ 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/bugs.txt b/bugs.txt index 0f00111e..f990b67a 100644 --- a/bugs.txt +++ b/bugs.txt @@ -1,3 +1,2 @@ Known bugs: -- When the program is minimized to tray, double clicking the icon only restores it to the task bar - Window flickers when blocks are added (problem with repainting?) \ No newline at end of file diff --git a/ui.cpp b/ui.cpp index e69630d3..e7be55da 100644 --- a/ui.cpp +++ b/ui.cpp @@ -3002,6 +3002,7 @@ void CBitcoinTBIcon::OnMenuRestore(wxCommandEvent&) { void CBitcoinTBIcon::Restore() { pframeMain->Show(); + pframeMain->Iconize(false); pframeMain->Raise(); if (!alwaysShowTrayIcon) Hide(); @@ -3296,6 +3297,9 @@ bool CMyApp::OnInit2() taskBarIcon = new CBitcoinTBIcon(); ApplyUISettings(); + if (mapArgs.count("/min") && minimizeToTray) { + pframeMain->Iconize(true); + } return true; } @@ -3408,6 +3412,7 @@ void ApplyUISettings() { IPersistFile* ppf = NULL; // Set the path to the shortcut target psl->SetPath(exePath); + psl->SetArguments("/min"); // Query IShellLink for the IPersistFile interface for // saving the shortcut in persistent storage. hres = psl->QueryInterface(IID_IPersistFile, From 661f878002d789f7e50b19d63da246414015c44b Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sat, 3 Oct 2009 17:02:59 +0000 Subject: [PATCH 007/133] Added NSIS installer generation script git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@13 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- setup.nsi | 138 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 138 insertions(+) create mode 100644 setup.nsi diff --git a/setup.nsi b/setup.nsi new file mode 100644 index 00000000..f49fe310 --- /dev/null +++ b/setup.nsi @@ -0,0 +1,138 @@ +# Auto-generated by EclipseNSIS Script Wizard +# 3.10.2009 19:00:28 + +Name Bitcoin + +# General Symbol Definitions +!define REGKEY "SOFTWARE\$(^Name)" +!define VERSION 0.1.6 +!define COMPANY "Bitcoin project" +!define URL http://bitcoin.sourceforge.net/ + +# MUI Symbol Definitions +!define MUI_ICON "rc\bitcoin.ico" +!define MUI_FINISHPAGE_NOAUTOCLOSE +!define MUI_STARTMENUPAGE_REGISTRY_ROOT HKLM +!define MUI_STARTMENUPAGE_REGISTRY_KEY ${REGKEY} +!define MUI_STARTMENUPAGE_REGISTRY_VALUENAME StartMenuGroup +!define MUI_STARTMENUPAGE_DEFAULTFOLDER Bitcoin +!define MUI_FINISHPAGE_RUN $INSTDIR\bitcoin.exe +!define MUI_UNICON "${NSISDIR}\Contrib\Graphics\Icons\modern-uninstall.ico" +!define MUI_UNFINISHPAGE_NOAUTOCLOSE + +# Included files +!include Sections.nsh +!include MUI2.nsh + +# Variables +Var StartMenuGroup + +# Installer pages +!insertmacro MUI_PAGE_WELCOME +!insertmacro MUI_PAGE_DIRECTORY +!insertmacro MUI_PAGE_STARTMENU Application $StartMenuGroup +!insertmacro MUI_PAGE_INSTFILES +!insertmacro MUI_PAGE_FINISH +!insertmacro MUI_UNPAGE_CONFIRM +!insertmacro MUI_UNPAGE_INSTFILES + +# Installer languages +!insertmacro MUI_LANGUAGE English + +# Installer attributes +OutFile Bitcoin_0.1.6_setup.exe +InstallDir $PROGRAMFILES\Bitcoin +CRCCheck on +XPStyle on +ShowInstDetails show +VIProductVersion 0.1.6.0 +VIAddVersionKey ProductName Bitcoin +VIAddVersionKey ProductVersion "${VERSION}" +VIAddVersionKey CompanyName "${COMPANY}" +VIAddVersionKey CompanyWebsite "${URL}" +VIAddVersionKey FileVersion "${VERSION}" +VIAddVersionKey FileDescription "" +VIAddVersionKey LegalCopyright "" +InstallDirRegKey HKLM "${REGKEY}" Path +ShowUninstDetails show + +# Installer sections +Section -Main SEC0000 + SetOutPath $INSTDIR + SetOverwrite on + File bitcoin.exe + File libeay32.dll + File mingwm10.dll + WriteRegStr HKLM "${REGKEY}\Components" Main 1 +SectionEnd + +Section -post SEC0001 + WriteRegStr HKLM "${REGKEY}" Path $INSTDIR + SetOutPath $INSTDIR + WriteUninstaller $INSTDIR\uninstall.exe + !insertmacro MUI_STARTMENU_WRITE_BEGIN Application + SetOutPath $SMPROGRAMS\$StartMenuGroup + CreateShortcut "$SMPROGRAMS\$StartMenuGroup\Bitcoin.lnk" $INSTDIR\bitcoin.exe + CreateShortcut "$SMPROGRAMS\$StartMenuGroup\Uninstall Bitcoin.lnk" $INSTDIR\uninstall.exe + !insertmacro MUI_STARTMENU_WRITE_END + WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayName "$(^Name)" + WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayVersion "${VERSION}" + WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" Publisher "${COMPANY}" + WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" URLInfoAbout "${URL}" + WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayIcon $INSTDIR\uninstall.exe + WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" UninstallString $INSTDIR\uninstall.exe + WriteRegDWORD HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" NoModify 1 + WriteRegDWORD HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" NoRepair 1 +SectionEnd + +# Macro for selecting uninstaller sections +!macro SELECT_UNSECTION SECTION_NAME UNSECTION_ID + Push $R0 + ReadRegStr $R0 HKLM "${REGKEY}\Components" "${SECTION_NAME}" + StrCmp $R0 1 0 next${UNSECTION_ID} + !insertmacro SelectSection "${UNSECTION_ID}" + GoTo done${UNSECTION_ID} +next${UNSECTION_ID}: + !insertmacro UnselectSection "${UNSECTION_ID}" +done${UNSECTION_ID}: + Pop $R0 +!macroend + +# Uninstaller sections +Section /o -un.Main UNSEC0000 + Delete /REBOOTOK $INSTDIR\mingwm10.dll + Delete /REBOOTOK $INSTDIR\libeay32.dll + Delete /REBOOTOK $INSTDIR\bitcoin.exe + DeleteRegValue HKLM "${REGKEY}\Components" Main +SectionEnd + +Section -un.post UNSEC0001 + DeleteRegKey HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" + Delete /REBOOTOK "$SMPROGRAMS\$StartMenuGroup\Uninstall Bitcoin.lnk" + Delete /REBOOTOK "$SMPROGRAMS\$StartMenuGroup\Bitcoin.lnk" + Delete /REBOOTOK $INSTDIR\uninstall.exe + DeleteRegValue HKLM "${REGKEY}" StartMenuGroup + DeleteRegValue HKLM "${REGKEY}" Path + DeleteRegKey /IfEmpty HKLM "${REGKEY}\Components" + DeleteRegKey /IfEmpty HKLM "${REGKEY}" + RmDir /REBOOTOK $SMPROGRAMS\$StartMenuGroup + RmDir /REBOOTOK $INSTDIR + Push $R0 + StrCpy $R0 $StartMenuGroup 1 + StrCmp $R0 ">" no_smgroup +no_smgroup: + Pop $R0 +SectionEnd + +# Installer functions +Function .onInit + InitPluginsDir +FunctionEnd + +# Uninstaller functions +Function un.onInit + ReadRegStr $INSTDIR HKLM "${REGKEY}" Path + !insertmacro MUI_STARTMENU_GETFOLDER Application $StartMenuGroup + !insertmacro SELECT_UNSECTION Main ${UNSEC0000} +FunctionEnd + From 0cc05617d17947d139d09f4ddcca7aeca755f00a Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sun, 4 Oct 2009 11:38:29 +0000 Subject: [PATCH 008/133] Fixed the installer script and made the autostart registry based. git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@14 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- changelog.txt | 5 ++-- headers.h | 1 - setup.nsi | 43 +++++++++++++++------------- ui.cpp | 77 ++++++++++++++++++++++----------------------------- uibase.cpp | 2 +- uiproject.fbp | 6 ++-- 6 files changed, 63 insertions(+), 71 deletions(-) diff --git a/changelog.txt b/changelog.txt index fee4f661..d2380b8a 100644 --- a/changelog.txt +++ b/changelog.txt @@ -2,5 +2,6 @@ Changes after 0.1.5: -------------------- + Options dialog layout changed - added the UI options panel + Minimize to tray feature -+ Startup on system boot feature (adds a shortcut to the Startup folder) -+ Ask before closing \ No newline at end of file ++ Startup on system boot feature ++ Ask before closing ++ NSIS installer \ No newline at end of file diff --git a/headers.h b/headers.h index 92911505..d0393c54 100644 --- a/headers.h +++ b/headers.h @@ -37,7 +37,6 @@ #include #include #include -#include #include #define BOUNDSCHECK 1 #include diff --git a/setup.nsi b/setup.nsi index f49fe310..56a46ea3 100644 --- a/setup.nsi +++ b/setup.nsi @@ -53,7 +53,7 @@ VIAddVersionKey CompanyWebsite "${URL}" VIAddVersionKey FileVersion "${VERSION}" VIAddVersionKey FileDescription "" VIAddVersionKey LegalCopyright "" -InstallDirRegKey HKLM "${REGKEY}" Path +InstallDirRegKey HKCU "${REGKEY}" Path ShowUninstDetails show # Installer sections @@ -63,32 +63,33 @@ Section -Main SEC0000 File bitcoin.exe File libeay32.dll File mingwm10.dll - WriteRegStr HKLM "${REGKEY}\Components" Main 1 + WriteRegStr HKCU "${REGKEY}\Components" Main 1 + WriteRegStr HKCU SOFTWARE\Microsoft\Windows\CurrentVersion\Run Bitcoin "$INSTDIR\bitcoin.exe /min" SectionEnd Section -post SEC0001 - WriteRegStr HKLM "${REGKEY}" Path $INSTDIR + WriteRegStr HKCU "${REGKEY}" Path $INSTDIR SetOutPath $INSTDIR WriteUninstaller $INSTDIR\uninstall.exe !insertmacro MUI_STARTMENU_WRITE_BEGIN Application - SetOutPath $SMPROGRAMS\$StartMenuGroup + CreateDirectory $SMPROGRAMS\$StartMenuGroup CreateShortcut "$SMPROGRAMS\$StartMenuGroup\Bitcoin.lnk" $INSTDIR\bitcoin.exe CreateShortcut "$SMPROGRAMS\$StartMenuGroup\Uninstall Bitcoin.lnk" $INSTDIR\uninstall.exe !insertmacro MUI_STARTMENU_WRITE_END - WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayName "$(^Name)" - WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayVersion "${VERSION}" - WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" Publisher "${COMPANY}" - WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" URLInfoAbout "${URL}" - WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayIcon $INSTDIR\uninstall.exe - WriteRegStr HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" UninstallString $INSTDIR\uninstall.exe - WriteRegDWORD HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" NoModify 1 - WriteRegDWORD HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" NoRepair 1 + WriteRegStr HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayName "$(^Name)" + WriteRegStr HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayVersion "${VERSION}" + WriteRegStr HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" Publisher "${COMPANY}" + WriteRegStr HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" URLInfoAbout "${URL}" + WriteRegStr HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" DisplayIcon $INSTDIR\uninstall.exe + WriteRegStr HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" UninstallString $INSTDIR\uninstall.exe + WriteRegDWORD HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" NoModify 1 + WriteRegDWORD HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" NoRepair 1 SectionEnd # Macro for selecting uninstaller sections !macro SELECT_UNSECTION SECTION_NAME UNSECTION_ID Push $R0 - ReadRegStr $R0 HKLM "${REGKEY}\Components" "${SECTION_NAME}" + ReadRegStr $R0 HKCU "${REGKEY}\Components" "${SECTION_NAME}" StrCmp $R0 1 0 next${UNSECTION_ID} !insertmacro SelectSection "${UNSECTION_ID}" GoTo done${UNSECTION_ID} @@ -103,18 +104,20 @@ Section /o -un.Main UNSEC0000 Delete /REBOOTOK $INSTDIR\mingwm10.dll Delete /REBOOTOK $INSTDIR\libeay32.dll Delete /REBOOTOK $INSTDIR\bitcoin.exe - DeleteRegValue HKLM "${REGKEY}\Components" Main + DeleteRegValue HKCU "${REGKEY}\Components" Main + DeleteRegValue HKCU SOFTWARE\Microsoft\Windows\CurrentVersion\Run Bitcoin SectionEnd Section -un.post UNSEC0001 - DeleteRegKey HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" + DeleteRegKey HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" Delete /REBOOTOK "$SMPROGRAMS\$StartMenuGroup\Uninstall Bitcoin.lnk" Delete /REBOOTOK "$SMPROGRAMS\$StartMenuGroup\Bitcoin.lnk" Delete /REBOOTOK $INSTDIR\uninstall.exe - DeleteRegValue HKLM "${REGKEY}" StartMenuGroup - DeleteRegValue HKLM "${REGKEY}" Path - DeleteRegKey /IfEmpty HKLM "${REGKEY}\Components" - DeleteRegKey /IfEmpty HKLM "${REGKEY}" + Delete /REBOOTOK $INSTDIR\db.log + DeleteRegValue HKCU "${REGKEY}" StartMenuGroup + DeleteRegValue HKCU "${REGKEY}" Path + DeleteRegKey /IfEmpty HKCU "${REGKEY}\Components" + DeleteRegKey /IfEmpty HKCU "${REGKEY}" RmDir /REBOOTOK $SMPROGRAMS\$StartMenuGroup RmDir /REBOOTOK $INSTDIR Push $R0 @@ -131,7 +134,7 @@ FunctionEnd # Uninstaller functions Function un.onInit - ReadRegStr $INSTDIR HKLM "${REGKEY}" Path + ReadRegStr $INSTDIR HKCU "${REGKEY}" Path !insertmacro MUI_STARTMENU_GETFOLDER Application $StartMenuGroup !insertmacro SELECT_UNSECTION Main ${UNSEC0000} FunctionEnd diff --git a/ui.cpp b/ui.cpp index e7be55da..a1e582ab 100644 --- a/ui.cpp +++ b/ui.cpp @@ -3016,8 +3016,9 @@ void CBitcoinTBIcon::UpdateTooltip() { wxMenu *CBitcoinTBIcon::CreatePopupMenu() { wxMenu *menu = new wxMenu; - wxMenuItem* generateCheck = menu->AppendCheckItem(PU_GENERATE, _T("Generate Coins")); menu->Append(PU_RESTORE, _T("Open Bitcoin")); + wxMenuItem* generateCheck = menu->AppendCheckItem(PU_GENERATE, _T("Generate Coins")); + menu->InsertSeparator(2); menu->Append(PU_EXIT, _T("Exit")); generateCheck->Check(fGenerateBitcoins); @@ -3386,51 +3387,39 @@ void ApplyUISettings() { taskBarIcon->Hide(); // Autostart on system startup? - // Get the startup folder shortcut path - char linkPath[ MAX_PATH ]; - SHGetSpecialFolderPath(0, linkPath, CSIDL_STARTUP, 0); - strcat(linkPath, "\\Bitcoin.lnk"); - - // If the shortcut exists already, remove it for updating - remove(linkPath); - - if (startOnSysBoot) { - CoInitialize(NULL); - // Get the current executable path - char exePath[ MAX_PATH ]; - GetModuleFileName(NULL, exePath, _MAX_PATH + 1); - - HRESULT hres = NULL; - IShellLink* psl = NULL; - // Get a pointer to the IShellLink interface. - hres = CoCreateInstance(CLSID_ShellLink, NULL, - CLSCTX_INPROC_SERVER, IID_IShellLink, - reinterpret_cast(&psl)); - - if (SUCCEEDED(hres)) - { - IPersistFile* ppf = NULL; - // Set the path to the shortcut target - psl->SetPath(exePath); - psl->SetArguments("/min"); - // Query IShellLink for the IPersistFile interface for - // saving the shortcut in persistent storage. - hres = psl->QueryInterface(IID_IPersistFile, - reinterpret_cast(&ppf)); - if (SUCCEEDED(hres)) - { - WCHAR wsz[MAX_PATH]; - // Ensure that the string is ANSI. - MultiByteToWideChar(CP_ACP, 0, linkPath, -1, - wsz, MAX_PATH); - // Save the link by calling IPersistFile::Save. - hres = ppf->Save(wsz, TRUE); - ppf->Release(); - } - psl->Release(); + // Open the startup registry key + HKEY hKey; + LONG lnRes = RegOpenKeyEx( + HKEY_CURRENT_USER, + "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run", + 0, + KEY_ALL_ACCESS, + &hKey + ); + + if ( ERROR_SUCCESS == lnRes ) + { + if (startOnSysBoot) { + // Get the current executable path + char exePath[ MAX_PATH ]; + GetModuleFileName(NULL, exePath, _MAX_PATH + 1); + char runCmd[ MAX_PATH + 5 ]; + strcat(runCmd, exePath); + strcat(runCmd," /min"); + + RegSetValueEx(hKey, + "Bitcoin", + 0, + REG_SZ, + (BYTE*)runCmd, + sizeof(runCmd) + ); + } + else { + RegDeleteValue(hKey, "Bitcoin"); } - CoUninitialize(); } + RegCloseKey(hKey); } diff --git a/uibase.cpp b/uibase.cpp index dcaa6975..a88c67eb 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -1870,7 +1870,7 @@ COptionsPanelUIBase::COptionsPanelUIBase( wxWindow* parent, wxWindowID id, const bSizer57->Add( m_checkCloseToTray, 0, wxALL, 5 ); - m_checkStartOnSysBoot = new wxCheckBox( this, wxID_ANY, wxT("Start on system boot"), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkStartOnSysBoot = new wxCheckBox( this, wxID_ANY, wxT("Start with Windows"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer57->Add( m_checkStartOnSysBoot, 0, wxALL, 5 ); diff --git a/uiproject.fbp b/uiproject.fbp index ccd8b222..315cc725 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -11849,7 +11849,7 @@ - + 1 @@ -12076,7 +12076,7 @@ - + 1 @@ -12250,7 +12250,7 @@ 0 wxID_ANY - Start on system boot + Start with Windows m_checkStartOnSysBoot From 99cef996c788755af95a0a614d7154a30928d4b9 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 21 Oct 2009 01:08:05 +0000 Subject: [PATCH 009/133] flush wallet.dat, multi-proc, reorg options, revert to startup folder shortcut git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@15 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- readme.txt => build.txt | 23 +- db.cpp | 57 +- db.h | 3 +- headers.h | 6 +- irc.cpp | 14 +- key.h | 12 + main.cpp | 152 +++- main.h | 14 +- makefile | 4 +- makefile.vc | 4 +- net.cpp | 111 +-- net.h | 17 +- serialize.h | 2 +- ui.cpp | 663 +++++++++------- ui.h | 108 +-- uibase.cpp | 233 +++--- uibase.h | 193 +++-- uiproject.fbp | 1581 +++++++++++++++++++++++---------------- util.cpp | 4 +- 19 files changed, 1898 insertions(+), 1303 deletions(-) rename readme.txt => build.txt (76%) diff --git a/readme.txt b/build.txt similarity index 76% rename from readme.txt rename to build.txt index 7615e6fd..b6b526f1 100644 --- a/readme.txt +++ b/build.txt @@ -1,4 +1,4 @@ -BitCoin v0.1.5 ALPHA +BitCoin v0.1.6 ALPHA Copyright (c) 2009 Satoshi Nakamoto Distributed under the MIT/X11 software license, see the accompanying @@ -10,7 +10,7 @@ cryptographic software written by Eric Young (eay@cryptsoft.com). Compilers Supported ------------------- -MinGW GCC (v3.4.5) +MinGW GCC (currently v3.4.5) Microsoft Visual C++ 6.0 SP6 @@ -31,6 +31,16 @@ Berkeley DB New BSD license with additional requirement that linked software Boost MIT-like license +Notes +----- +The UI layout is edited with wxFormBuilder. Open the project file +uiproject.fbp. It generates uibase.cpp and uibase.h, which define base +classes that do the rote work of constructing all the UI elements. + +The release is built with GCC and then "strip bitcoin.exe" to strip the debug +symbols, which reduces the executable size by about 90%. + + OpenSSL ------- Bitcoin does not use any encryption. If you want to do a no-everything @@ -47,7 +57,7 @@ Add this to crypto\err\err_all.c before the ERR_load_crypto_strings line: Edit ms\mingw32.bat and replace the Configure line's parameters with this no-everything list. You have to put this in the batch file because batch -files can't handle more than 9 parameters. +files can't take more than 9 command line parameters. perl Configure mingw threads no-rc2 no-rc4 no-rc5 no-idea no-des no-bf no-cast no-aes no-camellia no-seed no-rsa no-dh Also REM out the following line in ms\mingw32.bat. The build fails after it's @@ -64,7 +74,7 @@ If you want to use it with MSVC, generate the .lib file Berkeley DB ----------- -MinGW with MSYS: +Using MinGW and MSYS: cd \DB\build_unix sh ../dist/configure --enable-mingw --enable-cxx make @@ -72,5 +82,6 @@ make Boost ----- -You may need Boost version 1.35 to build with MSVC 6.0. I couldn't get -version 1.37 to compile with MSVC 6.0. +If you have trouble compiling Boost with Microsoft Visual C++ 6.0, try going +back to Boost version 1.35. It looks like they may be starting to reduce +support for MSVC 6.0. diff --git a/db.cpp b/db.cpp index 6607ff59..a9f5b795 100644 --- a/db.cpp +++ b/db.cpp @@ -453,11 +453,7 @@ bool CAddrDB::LoadAddresses() } } - //// debug print - printf("mapAddresses:\n"); - foreach(const PAIRTYPE(vector, CAddress)& item, mapAddresses) - item.second.print(); - printf("-----\n"); + printf("Loaded %d addresses\n", mapAddresses.size()); // Fix for possible bug that manifests in mapAddresses.count in irc.cpp, // just need to call count here and it doesn't happen there. The bug was the @@ -501,6 +497,26 @@ bool CReviewDB::WriteReviews(uint256 hash, const vector& vReviews) // CWalletDB // +CWalletDB::~CWalletDB() +{ + // Flush whenever all handles to wallet.dat are closed + Close(); + CRITICAL_BLOCK(cs_db) + { + map::iterator mi = mapFileUseCount.find(strFile); + if (mi != mapFileUseCount.end()) + { + int nRefCount = (*mi).second; + if (nRefCount == 0) + { + dbenv.txn_checkpoint(0, 0, 0); + dbenv.lsn_reset(strFile.c_str(), 0); + mapFileUseCount.erase(mi++); + } + } + } +} + bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) { vchDefaultKeyRet.clear(); @@ -568,34 +584,51 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) { ssValue >> vchDefaultKeyRet; } - else if (strType == "setting") /// or settings or option or options or config? + else if (strType == "setting") { string strKey; ssKey >> strKey; + + // Menu state + if (strKey == "fShowGenerated") ssValue >> fShowGenerated; if (strKey == "fGenerateBitcoins") ssValue >> fGenerateBitcoins; + + // Options if (strKey == "nTransactionFee") ssValue >> nTransactionFee; if (strKey == "addrIncoming") ssValue >> addrIncoming; - if (strKey == "minimizeToTray") ssValue >> minimizeToTray; - if (strKey == "closeToTray") ssValue >> closeToTray; - if (strKey == "startOnSysBoot") ssValue >> startOnSysBoot; - if (strKey == "askBeforeClosing") ssValue >> askBeforeClosing; - if (strKey == "alwaysShowTrayIcon") ssValue >> alwaysShowTrayIcon; + if (strKey == "fLimitProcessors") ssValue >> fLimitProcessors; + if (strKey == "nLimitProcessors") ssValue >> nLimitProcessors; + if (strKey == "fMinimizeToTray") ssValue >> fMinimizeToTray; + if (strKey == "fMinimizeOnClose") ssValue >> fMinimizeOnClose; } } } + printf("fShowGenerated = %d\n", fShowGenerated); printf("fGenerateBitcoins = %d\n", fGenerateBitcoins); printf("nTransactionFee = %I64d\n", nTransactionFee); printf("addrIncoming = %s\n", addrIncoming.ToString().c_str()); + printf("fMinimizeToTray = %d\n", fMinimizeToTray); + printf("fMinimizeOnClose = %d\n", fMinimizeOnClose); + + // The transaction fee setting won't be needed for many years to come. + // Setting it to zero here in case they set it to something in an earlier version. + if (nTransactionFee != 0) + { + nTransactionFee = 0; + WriteSetting("nTransactionFee", nTransactionFee); + } return true; } -bool LoadWallet() +bool LoadWallet(bool& fFirstRunRet) { + fFirstRunRet = false; vector vchDefaultKey; if (!CWalletDB("cr").LoadWallet(vchDefaultKey)) return false; + fFirstRunRet = vchDefaultKey.empty(); if (mapKeys.count(vchDefaultKey)) { diff --git a/db.h b/db.h index 1139f768..49614687 100644 --- a/db.h +++ b/db.h @@ -338,6 +338,7 @@ class CWalletDB : public CDB { public: CWalletDB(const char* pszMode="r+", bool fTxn=false) : CDB("wallet.dat", pszMode, fTxn) { } + ~CWalletDB(); private: CWalletDB(const CWalletDB&); void operator=(const CWalletDB&); @@ -412,7 +413,7 @@ public: bool LoadWallet(vector& vchDefaultKeyRet); }; -bool LoadWallet(); +bool LoadWallet(bool& fFirstRunRet); inline bool SetAddressBookName(const string& strAddress, const string& strName) { diff --git a/headers.h b/headers.h index d0393c54..f7e88e0c 100644 --- a/headers.h +++ b/headers.h @@ -10,11 +10,11 @@ #ifdef _WIN32_WINNT #undef _WIN32_WINNT #endif -#define _WIN32_WINNT 0x0500 +#define _WIN32_WINNT 0x0400 #ifdef _WIN32_IE #undef _WIN32_IE #endif -#define _WIN32_IE 0x0500 +#define _WIN32_IE 0x0400 #define WIN32_LEAN_AND_MEAN 1 #include #include @@ -28,6 +28,8 @@ #include #include #include +#include +#include #include #include #include diff --git a/irc.cpp b/irc.cpp index 3518df24..cfc9464a 100644 --- a/irc.cpp +++ b/irc.cpp @@ -52,7 +52,7 @@ bool DecodeAddress(string str, CAddress& addr) static bool Send(SOCKET hSocket, const char* pszSend) { if (strstr(pszSend, "PONG") != pszSend) - printf("SENDING: %s\n", pszSend); + printf("IRC SENDING: %s\n", pszSend); const char* psz = pszSend; const char* pszEnd = psz + strlen(psz); while (psz < pszEnd) @@ -145,7 +145,7 @@ bool Wait(int nSeconds) { if (fShutdown) return false; - printf("Waiting %d seconds to reconnect to IRC\n", nSeconds); + printf("IRC waiting %d seconds to reconnect\n", nSeconds); for (int i = 0; i < nSeconds; i++) { if (fShutdown) @@ -220,7 +220,6 @@ void ThreadIRCSeed(void* parg) { if (strLine.empty() || strLine.size() > 900 || strLine[0] != ':') continue; - printf("IRC %s\n", strLine.c_str()); vector vWords; ParseString(strLine, ' ', vWords); @@ -235,7 +234,7 @@ void ThreadIRCSeed(void* parg) // index 7 is limited to 16 characters // could get full length name at index 10, but would be different from join messages strcpy(pszName, vWords[7].c_str()); - printf("GOT WHO: [%s] ", pszName); + printf("IRC got who\n"); } if (vWords[1] == "JOIN" && vWords[0].size() > 1) @@ -244,7 +243,7 @@ void ThreadIRCSeed(void* parg) strcpy(pszName, vWords[0].c_str() + 1); if (strchr(pszName, '!')) *strchr(pszName, '!') = '\0'; - printf("GOT JOIN: [%s] ", pszName); + printf("IRC got join\n"); } if (pszName[0] == 'u') @@ -254,7 +253,7 @@ void ThreadIRCSeed(void* parg) { CAddrDB addrdb; if (AddAddress(addrdb, addr)) - printf("new "); + printf("IRC got new address\n"); else { // make it try connecting again @@ -262,14 +261,13 @@ void ThreadIRCSeed(void* parg) if (mapAddresses.count(addr.GetKey())) mapAddresses[addr.GetKey()].nLastFailed = 0; } - addr.print(); CRITICAL_BLOCK(cs_mapIRCAddresses) mapIRCAddresses.insert(make_pair(addr.GetKey(), addr)); } else { - printf("decode failed\n"); + printf("IRC decode failed\n"); } } } diff --git a/key.h b/key.h index 8b0b54e4..39ca86d3 100644 --- a/key.h +++ b/key.h @@ -44,6 +44,7 @@ class CKey { protected: EC_KEY* pkey; + bool fSet; public: CKey() @@ -51,6 +52,7 @@ public: pkey = EC_KEY_new_by_curve_name(NID_secp256k1); if (pkey == NULL) throw key_error("CKey::CKey() : EC_KEY_new_by_curve_name failed"); + fSet = false; } CKey(const CKey& b) @@ -58,12 +60,14 @@ public: pkey = EC_KEY_dup(b.pkey); if (pkey == NULL) throw key_error("CKey::CKey(const CKey&) : EC_KEY_dup failed"); + fSet = b.fSet; } CKey& operator=(const CKey& b) { if (!EC_KEY_copy(pkey, b.pkey)) throw key_error("CKey::operator=(const CKey&) : EC_KEY_copy failed"); + fSet = b.fSet; return (*this); } @@ -72,10 +76,16 @@ public: EC_KEY_free(pkey); } + bool IsNull() const + { + return !fSet; + } + void MakeNewKey() { if (!EC_KEY_generate_key(pkey)) throw key_error("CKey::MakeNewKey() : EC_KEY_generate_key failed"); + fSet = true; } bool SetPrivKey(const CPrivKey& vchPrivKey) @@ -83,6 +93,7 @@ public: const unsigned char* pbegin = &vchPrivKey[0]; if (!d2i_ECPrivateKey(&pkey, &pbegin, vchPrivKey.size())) return false; + fSet = true; return true; } @@ -103,6 +114,7 @@ public: const unsigned char* pbegin = &vchPubKey[0]; if (!o2i_ECPublicKey(&pkey, &pbegin, vchPubKey.size())) return false; + fSet = true; return true; } diff --git a/main.cpp b/main.cpp index ebf9d727..de5a5939 100644 --- a/main.cpp +++ b/main.cpp @@ -34,7 +34,7 @@ map mapOrphanTransactions; multimap mapOrphanTransactionsByPrev; map mapWallet; -vector > vWalletUpdated; +vector vWalletUpdated; CCriticalSection cs_mapWallet; map, CPrivKey> mapKeys; @@ -46,9 +46,12 @@ string strSetDataDir; int nDropMessagesTest = 0; // Settings -int fGenerateBitcoins; +int fGenerateBitcoins = false; int64 nTransactionFee = 0; CAddress addrIncoming; +int fLimitProcessors = false; +int nLimitProcessors = 1; + @@ -135,7 +138,7 @@ bool AddToWallet(const CWalletTx& wtxIn) return false; // Notify UI - vWalletUpdated.push_back(make_pair(hash, fInsertedNew)); + vWalletUpdated.push_back(hash); } // Refresh UI @@ -1126,6 +1129,9 @@ bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) } } + // Notify UI to update prev block coinbase if it was ours + vWalletUpdated.push_back(hashBestChain); + // New best link hashBestChain = hash; pindexBest = pindexNew; @@ -1702,7 +1708,7 @@ bool ProcessMessages(CNode* pfrom) } CATCH_PRINT_EXCEPTION("ProcessMessage()") if (!fRet) - printf("ProcessMessage(%s, %d bytes) from %s to %s FAILED\n", strCommand.c_str(), nMessageSize, pfrom->addr.ToString().c_str(), addrLocalHost.ToString().c_str()); + printf("ProcessMessage(%s, %d bytes) FAILED\n", strCommand.c_str(), nMessageSize); } vRecv.Compact(); @@ -1715,10 +1721,7 @@ bool ProcessMessages(CNode* pfrom) bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) { static map > mapReuseKey; - printf("received: %-12s (%d bytes) ", strCommand.c_str(), vRecv.size()); - for (int i = 0; i < min(vRecv.size(), (unsigned int)20); i++) - printf("%02x ", vRecv[i] & 0xff); - printf("\n"); + printf("received: %-12s (%d bytes)\n", strCommand.c_str(), vRecv.size()); if (nDropMessagesTest > 0 && GetRand(nDropMessagesTest) == 0) { printf("dropmessages DROPPING RECV MESSAGE\n"); @@ -1759,7 +1762,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), uint256(0)); } - printf("version message: %s has version %d, addrMe=%s\n", pfrom->addr.ToString().c_str(), pfrom->nVersion, addrMe.ToString().c_str()); + printf("version message: version %d\n", pfrom->nVersion); } @@ -1775,13 +1778,24 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vector vAddr; vRecv >> vAddr; + // Clear addrknown lists periodically to allow refresh broadcasts + static int64 nLastClearedAddrKnown; + if (nLastClearedAddrKnown < GetAdjustedTime() - 24 * 60 * 60) + { + nLastClearedAddrKnown = GetAdjustedTime(); + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + pnode->setAddrKnown.clear(); + } + // Store the new addresses CAddrDB addrdb; foreach(const CAddress& addr, vAddr) { if (fShutdown) return true; - if (AddAddress(addrdb, addr)) + AddAddress(addrdb, addr); + if (addr.IsRoutable() && addr.ip != addrLocalHost.ip) { // Put on lists to send to other nodes pfrom->setAddrKnown.insert(addr); @@ -1989,8 +2003,6 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) if (fShutdown) return true; const CAddress& addr = item.second; - //// will need this if we lose IRC - //if (addr.nTime > nSince || (rand() % nSize) < 500) if (addr.nTime > nSince) pfrom->vAddrToSend.push_back(addr); } @@ -2132,9 +2144,11 @@ bool SendMessages(CNode* pto) while (!pto->mapAskFor.empty() && (*pto->mapAskFor.begin()).first <= nNow) { const CInv& inv = (*pto->mapAskFor.begin()).second; - printf("sending getdata: %s\n", inv.ToString().c_str()); if (!AlreadyHave(txdb, inv)) + { + printf("sending getdata: %s\n", inv.ToString().c_str()); vAskFor.push_back(inv); + } pto->mapAskFor.erase(pto->mapAskFor.begin()); } if (!vAskFor.empty()) @@ -2162,6 +2176,49 @@ bool SendMessages(CNode* pto) // BitcoinMiner // +void GenerateBitcoins(bool fGenerate) +{ + if (fGenerateBitcoins != fGenerate) + { + fGenerateBitcoins = fGenerate; + CWalletDB().WriteSetting("fGenerateBitcoins", fGenerateBitcoins); + MainFrameRepaint(); + } + if (fGenerateBitcoins) + { + int nProcessors = atoi(getenv("NUMBER_OF_PROCESSORS")); + printf("%d processors\n", nProcessors); + if (nProcessors < 1) + nProcessors = 1; + if (fLimitProcessors && nProcessors > nLimitProcessors) + nProcessors = nLimitProcessors; + int nAddThreads = nProcessors - vnThreadsRunning[3]; + printf("starting %d bitcoinminer threads\n", nAddThreads); + for (int i = 0; i < nAddThreads; i++) + if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) + printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); + } +} + +void ThreadBitcoinMiner(void* parg) +{ + vnThreadsRunning[3]++; + CheckForShutdown(3); + try + { + bool fRet = BitcoinMiner(); + printf("BitcoinMiner returned %s\n\n\n", fRet ? "true" : "false"); + vnThreadsRunning[3]--; + } + catch (std::exception& e) { + vnThreadsRunning[3]--; + PrintException(&e, "ThreadBitcoinMiner()"); + } catch (...) { + vnThreadsRunning[3]--; + PrintException(NULL, "ThreadBitcoinMiner()"); + } +} + int FormatHashBlocks(void* pbuffer, unsigned int len) { unsigned char* pdata = (unsigned char*)pbuffer; @@ -2210,13 +2267,13 @@ void BlockSHA256(const void* pin, unsigned int nBlocks, void* pout) bool BitcoinMiner() { printf("BitcoinMiner started\n"); - SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); CKey key; key.MakeNewKey(); CBigNum bnExtraNonce = 0; while (fGenerateBitcoins) { + SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); Sleep(50); CheckForShutdown(3); while (vNodes.empty()) @@ -2338,7 +2395,6 @@ bool BitcoinMiner() BlockSHA256(&tmp.block, nBlocks0, &tmp.hash1); BlockSHA256(&tmp.hash1, nBlocks1, &hash); - if (hash <= hashTarget) { pblock->nNonce = tmp.block.nNonce; @@ -2352,6 +2408,12 @@ bool BitcoinMiner() SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL); CRITICAL_BLOCK(cs_main) { + if (pindexPrev != pindexBest) + { + SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); + break; + } + // Save key if (!AddKey(key)) return false; @@ -2368,7 +2430,7 @@ bool BitcoinMiner() } // Update nTime every few seconds - if ((++tmp.block.nNonce & 0x3ffff) == 0) + if ((++tmp.block.nNonce & 0xffff) == 0) { CheckForShutdown(3); if (tmp.block.nNonce == 0) @@ -2379,6 +2441,8 @@ bool BitcoinMiner() break; if (!fGenerateBitcoins) break; + if (fLimitProcessors && vnThreadsRunning[3] > nLimitProcessors) + return true; tmp.block.nTime = pblock->nTime = max(pindexPrev->GetMedianTimePast()+1, GetAdjustedTime()); } } @@ -2538,7 +2602,7 @@ bool SelectCoins(int64 nTargetValue, set& setCoinsRet) -bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, int64& nFeeRequiredRet) +bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CKey& keyRet, int64& nFeeRequiredRet) { nFeeRequiredRet = 0; CRITICAL_BLOCK(cs_main) @@ -2565,30 +2629,28 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, in foreach(CWalletTx* pcoin, setCoins) nValueIn += pcoin->GetCredit(); - // Fill vout[0] to the payee - wtxNew.vout.push_back(CTxOut(nValueOut, scriptPubKey)); + // Fill a vout to the payee + bool fChangeFirst = GetRand(2); + if (!fChangeFirst) + wtxNew.vout.push_back(CTxOut(nValueOut, scriptPubKey)); - // Fill vout[1] back to self with any change + // Fill a vout back to self with any change if (nValueIn > nValue) { - /// todo: for privacy, should randomize the order of outputs, - // would also have to use a new key for the change. - // Use the same key as one of the coins - vector vchPubKey; - CTransaction& txFirst = *(*setCoins.begin()); - foreach(const CTxOut& txout, txFirst.vout) - if (txout.IsMine()) - if (ExtractPubKey(txout.scriptPubKey, true, vchPubKey)) - break; - if (vchPubKey.empty()) - return false; + // New private key + if (keyRet.IsNull()) + keyRet.MakeNewKey(); - // Fill vout[1] to ourself + // Fill a vout to ourself CScript scriptPubKey; - scriptPubKey << vchPubKey << OP_CHECKSIG; + scriptPubKey << keyRet.GetPubKey() << OP_CHECKSIG; wtxNew.vout.push_back(CTxOut(nValueIn - nValue, scriptPubKey)); } + // Fill a vout to the payee + if (fChangeFirst) + wtxNew.vout.push_back(CTxOut(nValueOut, scriptPubKey)); + // Fill vin foreach(CWalletTx* pcoin, setCoins) for (int nOut = 0; nOut < pcoin->vout.size(); nOut++) @@ -2621,13 +2683,24 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, in } // Call after CreateTransaction unless you want to abort -bool CommitTransactionSpent(const CWalletTx& wtxNew) +bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key) { CRITICAL_BLOCK(cs_main) CRITICAL_BLOCK(cs_mapWallet) { - //// todo: make this transactional, never want to add a transaction - //// without marking spent transactions + //// todo: eventually should make this transactional, never want to add a + //// transaction without marking spent transactions, although the risk of + //// interruption during this step is remote. + + // This is only to keep the database open to defeat the auto-flush for the + // duration of this scope. This is the only place where this optimization + // maybe makes sense; please don't do it anywhere else. Keeping databases + // open longer than necessary can create deadlocks. + CWalletDB walletdb("r"); + + // Add the change's private key to wallet + if (!key.IsNull() && !AddKey(key)) + throw runtime_error("CommitTransactionSpent() : AddKey failed\n"); // Add tx to wallet, because if it has change it's also ours, // otherwise just for transaction history. @@ -2641,7 +2714,7 @@ bool CommitTransactionSpent(const CWalletTx& wtxNew) { pcoin->fSpent = true; pcoin->WriteToDisk(); - vWalletUpdated.push_back(make_pair(pcoin->GetHash(), false)); + vWalletUpdated.push_back(pcoin->GetHash()); } } MainFrameRepaint(); @@ -2655,8 +2728,9 @@ bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) { CRITICAL_BLOCK(cs_main) { + CKey key; int64 nFeeRequired; - if (!CreateTransaction(scriptPubKey, nValue, wtxNew, nFeeRequired)) + if (!CreateTransaction(scriptPubKey, nValue, wtxNew, key, nFeeRequired)) { string strError; if (nValue + nFeeRequired > GetBalance()) @@ -2666,7 +2740,7 @@ bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) wxMessageBox(strError, "Sending..."); return error("SendMoney() : %s\n", strError.c_str()); } - if (!CommitTransactionSpent(wtxNew)) + if (!CommitTransactionSpent(wtxNew, key)) { wxMessageBox("Error finalizing transaction ", "Sending..."); return error("SendMoney() : Error finalizing transaction"); diff --git a/main.h b/main.h index 9dd29bb6..6b11285e 100644 --- a/main.h +++ b/main.h @@ -41,6 +41,9 @@ extern int nDropMessagesTest; extern int fGenerateBitcoins; extern int64 nTransactionFee; extern CAddress addrIncoming; +extern int fLimitProcessors; +extern int nLimitProcessors; + @@ -58,14 +61,17 @@ void ReacceptWalletTransactions(); void RelayWalletTransactions(); bool LoadBlockIndex(bool fAllowNew=true); void PrintBlockTree(); -bool BitcoinMiner(); bool ProcessMessages(CNode* pfrom); bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv); bool SendMessages(CNode* pto); int64 GetBalance(); -bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& txNew, int64& nFeeRequiredRet); -bool CommitTransactionSpent(const CWalletTx& wtxNew); +bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CKey& keyRet, int64& nFeeRequiredRet); +bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key); bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew); +void GenerateBitcoins(bool fGenerate); +void ThreadBitcoinMiner(void* parg); +bool BitcoinMiner(); + @@ -1320,7 +1326,7 @@ public: extern map mapTransactions; extern map mapWallet; -extern vector > vWalletUpdated; +extern vector vWalletUpdated; extern CCriticalSection cs_mapWallet; extern map, CPrivKey> mapKeys; extern map > mapPubKeys; diff --git a/makefile b/makefile index 221684e8..64bb7730 100644 --- a/makefile +++ b/makefile @@ -12,7 +12,7 @@ ifeq "$(BUILD)" "debug" D=d # note: gcc 3.x profile doesn't work #DEBUGFLAGS=-O0 -g -pg -D__WXDEBUG__ -DEBUGFLAGS=-g -D__WXDEBUG__ -Wall -Wextra +DEBUGFLAGS=-g -D__WXDEBUG__ endif @@ -23,7 +23,7 @@ LIBS= \ -l db_cxx \ -l eay32 \ -l wxmsw28$(D)_richtext -l wxmsw28$(D)_html -l wxmsw28$(D)_core -l wxmsw28$(D)_adv -l wxbase28$(D) -l wxtiff$(D) -l wxjpeg$(D) -l wxpng$(D) -l wxzlib$(D) -l wxregex$(D) -l wxexpat$(D) \ - -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 + -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 -l shlwapi WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h diff --git a/makefile.vc b/makefile.vc index bf7e9bc5..c3bd0c4b 100644 --- a/makefile.vc +++ b/makefile.vc @@ -18,8 +18,8 @@ LIBPATHS=/LIBPATH:"/DB/build_windows/$(BUILD)" /LIBPATH:"/OpenSSL/out" /LIBPATH: LIBS= \ libdb47s$(D).lib \ libeay32.lib \ - wxmsw28$(D)_richtext.lib wxmsw28$(D)_html.lib wxmsw28$(D)_core.lib wxbase28$(D).lib wxtiff$(D).lib wxjpeg$(D).lib wxpng$(D).lib wxzlib$(D).lib wxregex$(D).lib wxexpat$(D).lib \ - kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib + wxmsw28$(D)_richtext.lib wxmsw28$(D)_html.lib wxmsw28$(D)_core.lib wxmsw28$(D)_adv.lib wxbase28$(D).lib wxtiff$(D).lib wxjpeg$(D).lib wxpng$(D).lib wxzlib$(D).lib wxregex$(D).lib wxexpat$(D).lib \ + kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib shlwapi.lib WXDEFS=/DWIN32 /D__WXMSW__ /D_WINDOWS /DNOPCH CFLAGS=/c /nologo /Ob0 /MD$(D) /EHsc /GR /Zm300 /YX /Fpobj/headers.pch $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h diff --git a/net.cpp b/net.cpp index b8d96006..db138e13 100644 --- a/net.cpp +++ b/net.cpp @@ -23,7 +23,8 @@ CAddress addrLocalHost(0, DEFAULT_PORT, nLocalServices); CNode nodeLocalHost(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); CNode* pnodeLocalHost = &nodeLocalHost; bool fShutdown = false; -array vfThreadRunning; +array vnThreadsRunning; + vector vNodes; CCriticalSection cs_vNodes; map, CAddress> mapAddresses; @@ -57,7 +58,7 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) if (fProxy) { - printf("Proxy connecting to %s\n", addrConnect.ToString().c_str()); + printf("Proxy connecting %s\n", addrConnect.ToStringLog().c_str()); char pszSocks4IP[] = "\4\1\0\0\0\0\0\0user"; memcpy(pszSocks4IP + 2, &addrConnect.port, 2); memcpy(pszSocks4IP + 4, &addrConnect.ip, 4); @@ -81,7 +82,7 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) closesocket(hSocket); return error("Proxy returned error %d\n", pchRet[1]); } - printf("Proxy connection established %s\n", addrConnect.ToString().c_str()); + printf("Proxy connection established %s\n", addrConnect.ToStringLog().c_str()); } hSocketRet = hSocket; @@ -219,6 +220,13 @@ bool AddAddress(CAddrDB& addrdb, const CAddress& addr) addrdb.WriteAddress(addrFound); return true; } + else if (addrFound.nTime < GetAdjustedTime() - 24 * 60 * 60) + { + // Periodically update most recently seen time + addrFound.nTime = GetAdjustedTime(); + addrdb.WriteAddress(addrFound); + return false; + } } } return false; @@ -373,14 +381,14 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) } /// debug print - printf("trying %s\n", addrConnect.ToString().c_str()); + printf("trying connection %s\n", addrConnect.ToStringLog().c_str()); // Connect SOCKET hSocket; if (ConnectSocket(addrConnect, hSocket)) { /// debug print - printf("connected %s\n", addrConnect.ToString().c_str()); + printf("connected %s\n", addrConnect.ToStringLog().c_str()); // Set to nonblocking u_long nOne = 1; @@ -410,7 +418,7 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) void CNode::Disconnect() { - printf("disconnecting node %s\n", addr.ToString().c_str()); + printf("disconnecting node %s\n", addr.ToStringLog().c_str()); closesocket(hSocket); @@ -450,14 +458,20 @@ void ThreadSocketHandler(void* parg) loop { - vfThreadRunning[0] = true; + vnThreadsRunning[0] = true; CheckForShutdown(0); try { ThreadSocketHandler2(parg); + vnThreadsRunning[0] = false; + } + catch (std::exception& e) { + vnThreadsRunning[0] = false; + PrintException(&e, "ThreadSocketHandler()"); + } catch (...) { + vnThreadsRunning[0] = false; + PrintException(NULL, "ThreadSocketHandler()"); } - CATCH_PRINT_EXCEPTION("ThreadSocketHandler()") - vfThreadRunning[0] = false; Sleep(5000); } } @@ -548,9 +562,9 @@ void ThreadSocketHandler2(void* parg) } } - vfThreadRunning[0] = false; + vnThreadsRunning[0] = false; int nSelect = select(hSocketMax + 1, &fdsetRecv, &fdsetSend, NULL, &timeout); - vfThreadRunning[0] = true; + vnThreadsRunning[0] = true; CheckForShutdown(0); if (nSelect == SOCKET_ERROR) { @@ -590,7 +604,7 @@ void ThreadSocketHandler2(void* parg) } else { - printf("accepted connection from %s\n", addr.ToString().c_str()); + printf("accepted connection %s\n", addr.ToStringLog().c_str()); CNode* pnode = new CNode(hSocket, addr, true); pnode->AddRef(); CRITICAL_BLOCK(cs_vNodes) @@ -697,14 +711,20 @@ void ThreadOpenConnections(void* parg) loop { - vfThreadRunning[1] = true; + vnThreadsRunning[1] = true; CheckForShutdown(1); try { ThreadOpenConnections2(parg); + vnThreadsRunning[1] = false; + } + catch (std::exception& e) { + vnThreadsRunning[1] = false; + PrintException(&e, "ThreadOpenConnections()"); + } catch (...) { + vnThreadsRunning[1] = false; + PrintException(NULL, "ThreadOpenConnections()"); } - CATCH_PRINT_EXCEPTION("ThreadOpenConnections()") - vfThreadRunning[1] = false; Sleep(5000); } } @@ -720,14 +740,14 @@ void ThreadOpenConnections2(void* parg) loop { // Wait - vfThreadRunning[1] = false; + vnThreadsRunning[1] = false; Sleep(500); while (vNodes.size() >= nMaxConnections || vNodes.size() >= mapAddresses.size()) { CheckForShutdown(1); Sleep(2000); } - vfThreadRunning[1] = true; + vnThreadsRunning[1] = true; CheckForShutdown(1); @@ -823,9 +843,9 @@ void ThreadOpenConnections2(void* parg) if (addrConnect.ip == addrLocalHost.ip || !addrConnect.IsIPv4() || FindNode(addrConnect.ip)) continue; - vfThreadRunning[1] = false; + vnThreadsRunning[1] = false; CNode* pnode = ConnectNode(addrConnect); - vfThreadRunning[1] = true; + vnThreadsRunning[1] = true; CheckForShutdown(1); if (!pnode) continue; @@ -867,14 +887,20 @@ void ThreadMessageHandler(void* parg) loop { - vfThreadRunning[2] = true; + vnThreadsRunning[2] = true; CheckForShutdown(2); try { ThreadMessageHandler2(parg); + vnThreadsRunning[2] = false; + } + catch (std::exception& e) { + vnThreadsRunning[2] = false; + PrintException(&e, "ThreadMessageHandler()"); + } catch (...) { + vnThreadsRunning[2] = false; + PrintException(NULL, "ThreadMessageHandler()"); } - CATCH_PRINT_EXCEPTION("ThreadMessageHandler()") - vfThreadRunning[2] = false; Sleep(5000); } } @@ -905,9 +931,9 @@ void ThreadMessageHandler2(void* parg) } // Wait and allow messages to bunch up - vfThreadRunning[2] = false; + vnThreadsRunning[2] = false; Sleep(100); - vfThreadRunning[2] = true; + vnThreadsRunning[2] = true; CheckForShutdown(2); } } @@ -920,29 +946,6 @@ void ThreadMessageHandler2(void* parg) -//// todo: start one thread per processor, use getenv("NUMBER_OF_PROCESSORS") -void ThreadBitcoinMiner(void* parg) -{ - vfThreadRunning[3] = true; - CheckForShutdown(3); - try - { - bool fRet = BitcoinMiner(); - printf("BitcoinMiner returned %s\n\n\n", fRet ? "true" : "false"); - } - CATCH_PRINT_EXCEPTION("BitcoinMiner()") - vfThreadRunning[3] = false; -} - - - - - - - - - - bool StartNode(string& strError) { @@ -1067,17 +1070,17 @@ bool StopNode() fShutdown = true; nTransactionsUpdated++; int64 nStart = GetTime(); - while (vfThreadRunning[0] || vfThreadRunning[2] || vfThreadRunning[3]) + while (vnThreadsRunning[0] || vnThreadsRunning[2] || vnThreadsRunning[3]) { if (GetTime() - nStart > 15) break; Sleep(20); } - if (vfThreadRunning[0]) printf("ThreadSocketHandler still running\n"); - if (vfThreadRunning[1]) printf("ThreadOpenConnections still running\n"); - if (vfThreadRunning[2]) printf("ThreadMessageHandler still running\n"); - if (vfThreadRunning[3]) printf("ThreadBitcoinMiner still running\n"); - while (vfThreadRunning[2]) + if (vnThreadsRunning[0]) printf("ThreadSocketHandler still running\n"); + if (vnThreadsRunning[1]) printf("ThreadOpenConnections still running\n"); + if (vnThreadsRunning[2]) printf("ThreadMessageHandler still running\n"); + if (vnThreadsRunning[3]) printf("ThreadBitcoinMiner still running\n"); + while (vnThreadsRunning[2]) Sleep(20); Sleep(50); @@ -1091,7 +1094,7 @@ void CheckForShutdown(int n) if (fShutdown) { if (n != -1) - vfThreadRunning[n] = false; + vnThreadsRunning[n] = false; if (n == 0) foreach(CNode* pnode, vNodes) closesocket(pnode->hSocket); diff --git a/net.h b/net.h index cd311fc5..33410042 100644 --- a/net.h +++ b/net.h @@ -20,8 +20,6 @@ enum - - bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet); bool GetMyExternalIP(unsigned int& ipRet); bool AddAddress(CAddrDB& addrdb, const CAddress& addr); @@ -29,7 +27,6 @@ CNode* FindNode(unsigned int ip); CNode* ConnectNode(CAddress addrConnect, int64 nTimeout=0); void AbandonRequests(void (*fn)(void*, CDataStream&), void* param1); bool AnySubscribed(unsigned int nChannel); -void ThreadBitcoinMiner(void* parg); bool StartNode(string& strError=REF(string())); bool StopNode(); void CheckForShutdown(int n); @@ -206,7 +203,7 @@ public: READWRITE(nTime); } READWRITE(nServices); - READWRITE(FLATDATA(pchReserved)); + READWRITE(FLATDATA(pchReserved)); // for IPv6 READWRITE(ip); READWRITE(port); ) @@ -280,10 +277,14 @@ public: return strprintf("%u.%u.%u.%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0)); } + string ToStringLog() const + { + return ""; + } + string ToString() const { return strprintf("%u.%u.%u.%u:%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0), ntohs(port)); - //return strprintf("%u.%u.%u.%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0)); } void print() const @@ -416,7 +417,7 @@ extern uint64 nLocalServices; extern CAddress addrLocalHost; extern CNode* pnodeLocalHost; extern bool fShutdown; -extern array vfThreadRunning; +extern array vnThreadsRunning; extern vector vNodes; extern CCriticalSection cs_vNodes; extern map, CAddress> mapAddresses; @@ -599,9 +600,7 @@ public: unsigned int nSize = vSend.size() - nPushPos - sizeof(CMessageHeader); memcpy((char*)&vSend[nPushPos] + offsetof(CMessageHeader, nMessageSize), &nSize, sizeof(nSize)); - printf("(%d bytes) ", nSize); - //for (int i = nPushPos+sizeof(CMessageHeader); i < min(vSend.size(), nPushPos+sizeof(CMessageHeader)+20U); i++) - // printf("%02x ", vSend[i] & 0xff); + printf("(%d bytes) ", nSize); printf("\n"); nPushPos = -1; diff --git a/serialize.h b/serialize.h index b7ab86d2..9b20e2a0 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 105; +static const int VERSION = 106; diff --git a/ui.cpp b/ui.cpp index a1e582ab..ce43a786 100644 --- a/ui.cpp +++ b/ui.cpp @@ -7,6 +7,13 @@ #include #endif +void ThreadRequestProductDetails(void* parg); +void ThreadRandSendTest(void* parg); +bool GetStartOnSystemStartup(); +void SetStartOnSystemStartup(bool fAutoStart); + + + DEFINE_EVENT_TYPE(wxEVT_CROSSTHREADCALL) DEFINE_EVENT_TYPE(wxEVT_REPLY1) DEFINE_EVENT_TYPE(wxEVT_REPLY2) @@ -16,22 +23,21 @@ DEFINE_EVENT_TYPE(wxEVT_TABLEUPDATED) DEFINE_EVENT_TYPE(wxEVT_TABLEDELETED) CMainFrame* pframeMain = NULL; +CMyTaskBarIcon* ptaskbaricon = NULL; map mapAddressBook; -CBitcoinTBIcon* taskBarIcon = NULL; // Tray icon - - -void ThreadRequestProductDetails(void* parg); -void ThreadRandSendTest(void* parg); +map mapArgs; bool fRandSendTest = false; void RandSend(); extern int g_isPainting; -// UI settings and their default values -int minimizeToTray = 1; -int closeToTray = 1; -int startOnSysBoot = 1; -int askBeforeClosing = 1; -int alwaysShowTrayIcon = 1; +// Settings +int fShowGenerated = true; +int fMinimizeToTray = true; +int fMinimizeOnClose = true; + + + + @@ -282,7 +288,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); SetIcon(wxICON(bitcoin)); - m_menuOptions->Check(wxID_OPTIONSGENERATEBITCOINS, fGenerateBitcoins); + ptaskbaricon = new CMyTaskBarIcon(); // Init toolbar with transparency masked bitmaps m_toolBar->ClearTools(); @@ -327,7 +333,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) //m_listCtrlOrdersReceived->InsertColumn(4, "", wxLIST_FORMAT_LEFT, 100); // Init status bar - int pnWidths[3] = { -100, 81, 286 }; + int pnWidths[3] = { -100, 88, 290 }; m_statusBar->SetFieldsCount(3, pnWidths); // Fill your address text box @@ -342,6 +348,8 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) CMainFrame::~CMainFrame() { pframeMain = NULL; + delete ptaskbaricon; + ptaskbaricon = NULL; } void Shutdown(void* parg) @@ -362,28 +370,24 @@ void Shutdown(void* parg) void CMainFrame::OnClose(wxCloseEvent& event) { - if (closeToTray && event.CanVeto()) { - event.Veto(); - SendToTray(); - } - else if (!event.CanVeto() || !askBeforeClosing || wxMessageBox("Quit program?", "Confirm", wxYES_NO, this) == wxYES) { - delete taskBarIcon; - Destroy(); - _beginthread(Shutdown, 0, NULL); - } + if (fMinimizeToTray && fMinimizeOnClose && event.CanVeto() && !IsIconized()) + { + // Divert close to minimize + event.Veto(); + Iconize(true); + } + else + { + Destroy(); + _beginthread(Shutdown, 0, NULL); + } } void CMainFrame::OnIconize(wxIconizeEvent& event) { - if (minimizeToTray) { - SendToTray(); - } -} - -void CMainFrame::SendToTray() -{ - Hide(); - taskBarIcon->Show(); + // Hide the task bar button when minimized. + // Event is sent when the frame is minimized or restored. + Show(!fMinimizeToTray || !event.Iconized()); } void CMainFrame::OnMouseEvents(wxMouseEvent& event) @@ -405,25 +409,21 @@ void CMainFrame::InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSo string str0 = strSort; long nData = *(long*)&hashKey; - if (fNew) + // Find item + if (!fNew && nIndex == -1) + { + while ((nIndex = m_listCtrl->FindItem(nIndex, nData)) != -1) + if (GetItemText(m_listCtrl, nIndex, 1) == hashKey.ToString()) + break; + } + + // fNew is for blind insert, only use if you're sure it's new + if (fNew || nIndex == -1) { nIndex = m_listCtrl->InsertItem(0, str0); } else { - if (nIndex == -1) - { - // Find item - while ((nIndex = m_listCtrl->FindItem(nIndex, nData)) != -1) - if (GetItemText(m_listCtrl, nIndex, 1) == hashKey.ToString()) - break; - if (nIndex == -1) - { - printf("CMainFrame::InsertLine : Couldn't find item to be updated\n"); - return; - } - } - // If sort key changed, must delete and reinsert to make it relocate if (GetItemText(m_listCtrl, nIndex, 0) != str0) { @@ -484,6 +484,28 @@ void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) string strStatus = FormatTxStatus(wtx); map mapValue = wtx.mapValue; + // Filter + if (wtx.IsCoinBase()) + { + // View->Show Generated + if (!fShowGenerated) + return; + + // Don't show generated coin until confirmed by at least one block after it + // so we don't get the user's hopes up until it looks like it's probably accepted. + // + // It is not an error when generated blocks are not accepted. By design, + // some percentage of blocks, like 10% or more, will end up not accepted. + // This is the normal mechanism by which the network copes with latency. + // + // We display regular transactions right away before any confirmation + // because they can always get into some block eventually. Generated coins + // are special because if their block is not accepted, they are not valid. + // + if (wtx.GetDepthInMainChain() < 2) + return; + } + // Find the block the tx is in CBlockIndex* pindex = NULL; map::iterator mi = mapBlockIndex.find(wtx.hashBlock); @@ -785,16 +807,11 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) { TRY_CRITICAL_BLOCK(cs_mapWallet) { - pair item; - foreach(item, vWalletUpdated) + foreach(uint256 hash, vWalletUpdated) { - bool fNew = item.second; - map::iterator mi = mapWallet.find(item.first); + map::iterator mi = mapWallet.find(hash); if (mi != mapWallet.end()) - { - printf("vWalletUpdated: %s %s\n", (*mi).second.GetHash().ToString().substr(0,6).c_str(), fNew ? "new" : ""); - InsertTransaction((*mi).second, fNew); - } + InsertTransaction((*mi).second, false); } m_listCtrl->ScrollList(0, INT_MAX); vWalletUpdated.clear(); @@ -807,7 +824,7 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) // Update status bar string strGen = ""; if (fGenerateBitcoins) - strGen = " Generating"; + strGen = " Generating"; if (fGenerateBitcoins && vNodes.empty()) strGen = "(not connected)"; m_statusBar->SetStatusText(strGen, 1); @@ -856,39 +873,54 @@ void CMainFrame::OnCrossThreadCall(wxCommandEvent& event) void CMainFrame::OnMenuFileExit(wxCommandEvent& event) { + // File->Exit Close(true); } +void CMainFrame::OnMenuViewShowGenerated(wxCommandEvent& event) +{ + // View->Show Generated + fShowGenerated = event.IsChecked(); + CWalletDB().WriteSetting("fShowGenerated", fShowGenerated); + RefreshListCtrl(); +} + +void CMainFrame::OnUpdateUIViewShowGenerated(wxUpdateUIEvent& event) +{ + event.Check(fShowGenerated); +} + void CMainFrame::OnMenuOptionsGenerate(wxCommandEvent& event) { + // Options->Generate Coins GenerateBitcoins(event.IsChecked()); +} - Refresh(); - wxPaintEvent eventPaint; - AddPendingEvent(eventPaint); +void CMainFrame::OnUpdateUIOptionsGenerate(wxUpdateUIEvent& event) +{ + event.Check(fGenerateBitcoins); } void CMainFrame::OnMenuOptionsChangeYourAddress(wxCommandEvent& event) { + // Options->Change Your Address OnButtonChange(event); } void CMainFrame::OnMenuOptionsOptions(wxCommandEvent& event) { + // Options->Options COptionsDialog dialog(this); dialog.ShowModal(); } void CMainFrame::OnMenuHelpAbout(wxCommandEvent& event) { + // Help->About CAboutDialog dialog(this); dialog.ShowModal(); } -void CMainFrame::OnUpdateMenuGenerate( wxUpdateUIEvent& event ) { - event.Check(fGenerateBitcoins); -} - void CMainFrame::OnButtonSend(wxCommandEvent& event) { /// debug test @@ -1252,58 +1284,74 @@ void CTxDetailsDialog::OnButtonOK(wxCommandEvent& event) COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) { + // Set up list box of page choices + m_listBox->Append("Main"); + //m_listBox->Append("Test 2"); + m_listBox->SetSelection(0); + SelectPage(0); + + // Init values + m_textCtrlTransactionFee->SetValue(FormatMoney(nTransactionFee)); + m_checkBoxLimitProcessors->SetValue(fLimitProcessors); + m_spinCtrlLimitProcessors->Enable(fLimitProcessors); + m_spinCtrlLimitProcessors->SetValue(nLimitProcessors); + int nProcessors = atoi(getenv("NUMBER_OF_PROCESSORS")); + if (nProcessors < 1) + nProcessors = 999; + m_spinCtrlLimitProcessors->SetRange(1, nProcessors); + m_checkBoxStartOnSystemStartup->SetValue(fTmpStartOnSystemStartup = GetStartOnSystemStartup()); + m_checkBoxMinimizeToTray->SetValue(fMinimizeToTray); + m_checkBoxMinimizeOnClose->Enable(fMinimizeToTray); + m_checkBoxMinimizeOnClose->SetValue(fMinimizeToTray && fMinimizeOnClose); + fTmpMinimizeOnClose = fMinimizeOnClose; m_buttonOK->SetFocus(); - m_treeCtrl->AddRoot(wxT("Settings")); - m_treeCtrl->AppendItem(m_treeCtrl->GetRootItem(), wxT("Bitcoin")); - m_treeCtrl->AppendItem(m_treeCtrl->GetRootItem(), wxT("UI")); +} - panelUI = new COptionsPanelUI(this); - panelBitcoin = new COptionsPanelBitcoin(this); - currentPanel = panelBitcoin; +void COptionsDialog::SelectPage(int nPage) +{ + m_panelMain->Show(nPage == 0); + m_panelTest2->Show(nPage == 1); - panelSizer->Add(panelUI); - panelSizer->Hide(panelUI); - panelSizer->Add(panelBitcoin); - panelSizer->Layout(); + m_scrolledWindow->Layout(); + m_scrolledWindow->SetScrollbars(0, 0, 0, 0, 0, 0); +} +void COptionsDialog::OnListBox(wxCommandEvent& event) +{ + SelectPage(event.GetSelection()); } -void COptionsDialog::MenuSelChanged( wxTreeEvent& event ) +void COptionsDialog::OnKillFocusTransactionFee(wxFocusEvent& event) { - panelSizer->Hide(currentPanel); - wxString text = m_treeCtrl->GetItemText(event.GetItem()); - if (text == "Bitcoin") { - panelSizer->Show(panelBitcoin); - currentPanel = panelBitcoin; - } - else { - panelSizer->Show(panelUI); - currentPanel = panelUI; - } - panelSizer->Layout(); + int64 nTmp = nTransactionFee; + ParseMoney(m_textCtrlTransactionFee->GetValue(), nTmp); + m_textCtrlTransactionFee->SetValue(FormatMoney(nTmp)); } -void COptionsDialog::OnButtonOK(wxCommandEvent& event) +void COptionsDialog::OnCheckBoxLimitProcessors(wxCommandEvent& event) { - // nTransactionFee - int64 nPrevTransactionFee = nTransactionFee; - if (ParseMoney(panelBitcoin->m_textCtrlTransactionFee->GetValue(), nTransactionFee) && nTransactionFee != nPrevTransactionFee) - CWalletDB().WriteSetting("transactionFee", nTransactionFee); + m_spinCtrlLimitProcessors->Enable(event.IsChecked()); +} - minimizeToTray = panelUI->m_checkMinToTray->IsChecked(); - closeToTray = panelUI->m_checkCloseToTray->IsChecked(); - startOnSysBoot = panelUI->m_checkStartOnSysBoot->IsChecked(); - askBeforeClosing = panelUI->m_checkAskBeforeClosing->IsChecked(); - alwaysShowTrayIcon = panelUI->m_checkAlwaysShowTray->IsChecked(); +void COptionsDialog::OnCheckBoxMinimizeToTray(wxCommandEvent& event) +{ + m_checkBoxMinimizeOnClose->Enable(event.IsChecked()); - CWalletDB().WriteSetting("minimizeToTray", minimizeToTray); - CWalletDB().WriteSetting("closeToTray", closeToTray); - CWalletDB().WriteSetting("startOnSysBoot", startOnSysBoot); - CWalletDB().WriteSetting("askBeforeClosing", askBeforeClosing); - CWalletDB().WriteSetting("alwaysShowTrayIcon", alwaysShowTrayIcon); + // Save the value in fTmpMinimizeOnClose so we can + // show the checkbox unchecked when its parent is unchecked + if (event.IsChecked()) + m_checkBoxMinimizeOnClose->SetValue(fTmpMinimizeOnClose); + else + { + fTmpMinimizeOnClose = m_checkBoxMinimizeOnClose->GetValue(); + m_checkBoxMinimizeOnClose->SetValue(false); + } - ApplyUISettings(); +} +void COptionsDialog::OnButtonOK(wxCommandEvent& event) +{ + OnButtonApply(event); Close(); } @@ -1312,44 +1360,53 @@ void COptionsDialog::OnButtonCancel(wxCommandEvent& event) Close(); } - - -////////////////////////////////////////////////////////////////////////////// -// -// COptionsPanelBitcoin -// - -COptionsPanelBitcoin::COptionsPanelBitcoin(wxWindow* parent) : COptionsPanelBitcoinBase(parent) +void COptionsDialog::OnButtonApply(wxCommandEvent& event) { - m_textCtrlTransactionFee->SetValue(FormatMoney(nTransactionFee)); -} + CWalletDB walletdb; -void COptionsPanelBitcoin::OnKillFocusTransactionFee(wxFocusEvent& event) -{ - int64 nTmp = nTransactionFee; - ParseMoney(m_textCtrlTransactionFee->GetValue(), nTmp); - m_textCtrlTransactionFee->SetValue(FormatMoney(nTmp)); -} + int64 nPrevTransactionFee = nTransactionFee; + if (ParseMoney(m_textCtrlTransactionFee->GetValue(), nTransactionFee) && nTransactionFee != nPrevTransactionFee) + walletdb.WriteSetting("nTransactionFee", nTransactionFee); + int nPrevMaxProc = (fLimitProcessors ? nLimitProcessors : INT_MAX); + if (fLimitProcessors != m_checkBoxLimitProcessors->GetValue()) + { + fLimitProcessors = m_checkBoxLimitProcessors->GetValue(); + walletdb.WriteSetting("fLimitProcessors", fLimitProcessors); + } + if (nLimitProcessors != m_spinCtrlLimitProcessors->GetValue()) + { + nLimitProcessors = m_spinCtrlLimitProcessors->GetValue(); + walletdb.WriteSetting("nLimitProcessors", nLimitProcessors); + } + if (fGenerateBitcoins && (fLimitProcessors ? nLimitProcessors : INT_MAX) > nPrevMaxProc) + GenerateBitcoins(fGenerateBitcoins); -////////////////////////////////////////////////////////////////////////////// -// -// COptionsPanelUI -// + if (fTmpStartOnSystemStartup != m_checkBoxStartOnSystemStartup->GetValue()) + { + fTmpStartOnSystemStartup = m_checkBoxStartOnSystemStartup->GetValue(); + SetStartOnSystemStartup(fTmpStartOnSystemStartup); + } -COptionsPanelUI::COptionsPanelUI(wxWindow* parent) : COptionsPanelUIBase(parent) -{ - m_checkMinToTray->SetValue(minimizeToTray); - m_checkCloseToTray->SetValue(closeToTray); - m_checkStartOnSysBoot->SetValue(startOnSysBoot); - m_checkAskBeforeClosing->SetValue(askBeforeClosing); - m_checkAlwaysShowTray->SetValue(alwaysShowTrayIcon); + if (fMinimizeToTray != m_checkBoxMinimizeToTray->GetValue()) + { + fMinimizeToTray = m_checkBoxMinimizeToTray->GetValue(); + walletdb.WriteSetting("fMinimizeToTray", fMinimizeToTray); + ptaskbaricon->Show(fMinimizeToTray); + } + + if (fMinimizeOnClose != (fMinimizeToTray ? m_checkBoxMinimizeOnClose->GetValue() : fTmpMinimizeOnClose)) + { + fMinimizeOnClose = (fMinimizeToTray ? m_checkBoxMinimizeOnClose->GetValue() : fTmpMinimizeOnClose); + walletdb.WriteSetting("fMinimizeOnClose", fMinimizeOnClose); + } } + ////////////////////////////////////////////////////////////////////////////// // // CAboutDialog @@ -1387,6 +1444,7 @@ CSendDialog::CSendDialog(wxWindow* parent, const wxString& strAddress) : CSendDi m_textCtrlAddress->SetValue(strAddress); m_choiceTransferType->SetSelection(0); m_bitmapCheckMark->Show(false); + fEnabledPrev = true; //// todo: should add a display of your balance for convenience // Set Icon @@ -1418,6 +1476,19 @@ void CSendDialog::OnTextAddress(wxCommandEvent& event) m_staticTextMessage->Enable(fEnable); m_textCtrlMessage->Enable(fEnable); m_textCtrlMessage->SetBackgroundColour(wxSystemSettings::GetColour(fEnable ? wxSYS_COLOUR_WINDOW : wxSYS_COLOUR_BTNFACE)); + if (!fEnable && fEnabledPrev) + { + strFromSave = m_textCtrlFrom->GetValue(); + strMessageSave = m_textCtrlMessage->GetValue(); + m_textCtrlFrom->SetValue("Will appear as \"From: Unknown\""); + m_textCtrlMessage->SetValue("Can't include a message when sending to a Bitcoin address"); + } + else if (fEnable && !fEnabledPrev) + { + m_textCtrlFrom->SetValue(strFromSave); + m_textCtrlMessage->SetValue(strMessageSave); + } + fEnabledPrev = fEnable; } void CSendDialog::OnKillFocusAmount(wxFocusEvent& event) @@ -1773,8 +1844,9 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) Error("You don't have enough money"); return; } + CKey key; int64 nFeeRequired; - if (!CreateTransaction(scriptPubKey, nPrice, wtx, nFeeRequired)) + if (!CreateTransaction(scriptPubKey, nPrice, wtx, key, nFeeRequired)) { if (nPrice + nFeeRequired > GetBalance()) Error(strprintf("This is an oversized transaction that requires a transaction fee of %s", FormatMoney(nFeeRequired).c_str())); @@ -1799,7 +1871,7 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) return; // Commit - if (!CommitTransactionSpent(wtx)) + if (!CommitTransactionSpent(wtx, key)) { Error("Error finalizing payment"); return; @@ -2950,84 +3022,112 @@ void CEditReviewDialog::GetReview(CReview& review) + + ////////////////////////////////////////////////////////////////////////////// // -// BitcoinTBIcon +// CMyTaskBarIcon // -enum { - PU_RESTORE = 10001, - PU_GENERATE, - PU_EXIT, +enum +{ + ID_TASKBAR_RESTORE = 10001, + ID_TASKBAR_GENERATE, + ID_TASKBAR_EXIT, }; -BEGIN_EVENT_TABLE(CBitcoinTBIcon, wxTaskBarIcon) - EVT_TASKBAR_LEFT_DCLICK (CBitcoinTBIcon::OnLeftButtonDClick) - EVT_MENU(PU_RESTORE, CBitcoinTBIcon::OnMenuRestore) - EVT_MENU(PU_GENERATE, CBitcoinTBIcon::OnMenuGenerate) - EVT_MENU(PU_EXIT, CBitcoinTBIcon::OnMenuExit) +BEGIN_EVENT_TABLE(CMyTaskBarIcon, wxTaskBarIcon) + EVT_TASKBAR_LEFT_DCLICK(CMyTaskBarIcon::OnLeftButtonDClick) + EVT_MENU(ID_TASKBAR_RESTORE, CMyTaskBarIcon::OnMenuRestore) + EVT_MENU(ID_TASKBAR_GENERATE, CMyTaskBarIcon::OnMenuGenerate) + EVT_UPDATE_UI(ID_TASKBAR_GENERATE, CMyTaskBarIcon::OnUpdateUIGenerate) + EVT_MENU(ID_TASKBAR_EXIT, CMyTaskBarIcon::OnMenuExit) END_EVENT_TABLE() -void CBitcoinTBIcon::Show() +void CMyTaskBarIcon::Show(bool fShow) { - string tooltip = "Bitcoin"; - tooltip += fGenerateBitcoins ? " - Generating" : ""; - SetIcon(wxICON(bitcoin), tooltip); + if (fShow) + { + string strTooltip = "Bitcoin"; + if (fGenerateBitcoins) + strTooltip = "Bitcoin - Generating"; + if (fGenerateBitcoins && vNodes.empty()) + strTooltip = "Bitcoin - (not connected)"; + SetIcon(wxICON(bitcoin), strTooltip); + } + else + { + RemoveIcon(); + } } -void CBitcoinTBIcon::Hide() +void CMyTaskBarIcon::Hide() { - RemoveIcon(); + Show(false); } -void CBitcoinTBIcon::OnLeftButtonDClick(wxTaskBarIconEvent&) +void CMyTaskBarIcon::OnLeftButtonDClick(wxTaskBarIconEvent& event) { - Restore(); + Restore(); } -void CBitcoinTBIcon::OnMenuExit(wxCommandEvent&) +void CMyTaskBarIcon::OnMenuRestore(wxCommandEvent& event) { - pframeMain->Close(true); + Restore(); } -void CBitcoinTBIcon::OnMenuGenerate(wxCommandEvent& event) +void CMyTaskBarIcon::Restore() { - GenerateBitcoins(event.IsChecked()); - pframeMain->Refresh(); -} - -void CBitcoinTBIcon::OnMenuRestore(wxCommandEvent&) { - Restore(); -} - -void CBitcoinTBIcon::Restore() { pframeMain->Show(); pframeMain->Iconize(false); pframeMain->Raise(); - if (!alwaysShowTrayIcon) - Hide(); } -void CBitcoinTBIcon::UpdateTooltip() { - if (IsIconInstalled()) - Show(); +void CMyTaskBarIcon::OnMenuGenerate(wxCommandEvent& event) +{ + GenerateBitcoins(event.IsChecked()); +} + +void CMyTaskBarIcon::OnUpdateUIGenerate(wxUpdateUIEvent& event) +{ + event.Check(fGenerateBitcoins); } -wxMenu *CBitcoinTBIcon::CreatePopupMenu() +void CMyTaskBarIcon::OnMenuExit(wxCommandEvent& event) { - wxMenu *menu = new wxMenu; - menu->Append(PU_RESTORE, _T("Open Bitcoin")); - wxMenuItem* generateCheck = menu->AppendCheckItem(PU_GENERATE, _T("Generate Coins")); - menu->InsertSeparator(2); - menu->Append(PU_EXIT, _T("Exit")); + pframeMain->Close(true); +} - generateCheck->Check(fGenerateBitcoins); +void CMyTaskBarIcon::UpdateTooltip() +{ + if (IsIconInstalled()) + Show(true); +} - return menu; +wxMenu* CMyTaskBarIcon::CreatePopupMenu() +{ + wxMenu* pmenu = new wxMenu; + pmenu->Append(ID_TASKBAR_RESTORE, "&Open Bitcoin"); + pmenu->AppendCheckItem(ID_TASKBAR_GENERATE, "&Generate Coins")->Check(fGenerateBitcoins); +#ifndef __WXMAC_OSX__ // Mac has built-in quit menu + pmenu->AppendSeparator(); + pmenu->Append(ID_TASKBAR_EXIT, "E&xit"); +#endif + return pmenu; } + + + + + + + + + + ////////////////////////////////////////////////////////////////////////////// // // CMyApp @@ -3150,7 +3250,7 @@ bool CMyApp::OnInit2() // Parameters // wxImage::AddHandler(new wxPNGHandler); - map mapArgs = ParseParameters(argc, argv); + mapArgs = ParseParameters(argc, argv); if (mapArgs.count("/datadir")) strSetDataDir = mapArgs["/datadir"]; @@ -3179,6 +3279,7 @@ bool CMyApp::OnInit2() // // Load data files // + bool fFirstRun; string strErrors; int64 nStart, nEnd; @@ -3198,7 +3299,7 @@ bool CMyApp::OnInit2() printf("Loading wallet...\n"); QueryPerformanceCounter((LARGE_INTEGER*)&nStart); - if (!LoadWallet()) + if (!LoadWallet(fFirstRun)) strErrors += "Error loading wallet.dat \n"; QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); printf(" wallet %20I64d\n", nEnd - nStart); @@ -3244,62 +3345,60 @@ bool CMyApp::OnInit2() // // Create the main frame window // + pframeMain = new CMainFrame(NULL); + if (mapArgs.count("/min")) + pframeMain->Iconize(true); + pframeMain->Show(true); // have to show first to get taskbar button to hide + pframeMain->Show(!fMinimizeToTray || !pframeMain->IsIconized()); + ptaskbaricon->Show(fMinimizeToTray); + + if (!CheckDiskSpace()) { - pframeMain = new CMainFrame(NULL); - pframeMain->Show(); + OnExit(); + return false; + } - if (!CheckDiskSpace()) - { - OnExit(); - return false; - } + if (!StartNode(strErrors)) + wxMessageBox(strErrors, "Bitcoin"); - if (!StartNode(strErrors)) - wxMessageBox(strErrors, "Bitcoin"); + GenerateBitcoins(fGenerateBitcoins); - if (fGenerateBitcoins) - if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) - printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); + if (fFirstRun) + SetStartOnSystemStartup(true); - // - // Tests - // - if (argc >= 2 && stricmp(argv[1], "/send") == 0) - { - int64 nValue = 1; - if (argc >= 3) - ParseMoney(argv[2], nValue); - - string strAddress; - if (argc >= 4) - strAddress = argv[3]; - CAddress addr(strAddress.c_str()); - - CWalletTx wtx; - wtx.mapValue["to"] = strAddress; - wtx.mapValue["from"] = addrLocalHost.ToString(); - wtx.mapValue["message"] = "command line send"; - - // Send to IP address - CSendingDialog* pdialog = new CSendingDialog(pframeMain, addr, nValue, wtx); - if (!pdialog->ShowModal()) - return false; - } - if (mapArgs.count("/randsendtest")) - { - if (!mapArgs["/randsendtest"].empty()) - _beginthread(ThreadRandSendTest, 0, new string(mapArgs["/randsendtest"])); - else - fRandSendTest = true; - fDebug = true; - } + // + // Tests + // + if (argc >= 2 && stricmp(argv[1], "/send") == 0) + { + int64 nValue = 1; + if (argc >= 3) + ParseMoney(argv[2], nValue); + + string strAddress; + if (argc >= 4) + strAddress = argv[3]; + CAddress addr(strAddress.c_str()); + + CWalletTx wtx; + wtx.mapValue["to"] = strAddress; + wtx.mapValue["from"] = addrLocalHost.ToString(); + wtx.mapValue["message"] = "command line send"; + + // Send to IP address + CSendingDialog* pdialog = new CSendingDialog(pframeMain, addr, nValue, wtx); + if (!pdialog->ShowModal()) + return false; } - taskBarIcon = new CBitcoinTBIcon(); - ApplyUISettings(); - if (mapArgs.count("/min") && minimizeToTray) { - pframeMain->Iconize(true); + if (mapArgs.count("/randsendtest")) + { + if (!mapArgs["/randsendtest"].empty()) + _beginthread(ThreadRandSendTest, 0, new string(mapArgs["/randsendtest"])); + else + fRandSendTest = true; + fDebug = true; } return true; @@ -3320,14 +3419,14 @@ bool CMyApp::OnExceptionInMainLoop() catch (std::exception& e) { PrintException(&e, "CMyApp::OnExceptionInMainLoop()"); - wxLogWarning(_T("Exception %s %s"), typeid(e).name(), e.what()); + wxLogWarning("Exception %s %s", typeid(e).name(), e.what()); Sleep(1000); throw; } catch (...) { PrintException(NULL, "CMyApp::OnExceptionInMainLoop()"); - wxLogWarning(_T("Unknown exception")); + wxLogWarning("Unknown exception"); Sleep(1000); throw; } @@ -3345,14 +3444,14 @@ void CMyApp::OnUnhandledException() catch (std::exception& e) { PrintException(&e, "CMyApp::OnUnhandledException()"); - wxLogWarning(_T("Exception %s %s"), typeid(e).name(), e.what()); + wxLogWarning("Exception %s %s", typeid(e).name(), e.what()); Sleep(1000); throw; } catch (...) { PrintException(NULL, "CMyApp::OnUnhandledException()"); - wxLogWarning(_T("Unknown exception")); + wxLogWarning("Unknown exception"); Sleep(1000); throw; } @@ -3367,6 +3466,8 @@ void CMyApp::OnFatalException() void MainFrameRepaint() { + // This is called by network code that shouldn't access pframeMain and ptaskbaricon + // directly because it could still be running after the UI is closed. if (pframeMain) { printf("MainFrameRepaint()\n"); @@ -3374,68 +3475,84 @@ void MainFrameRepaint() pframeMain->Refresh(); pframeMain->AddPendingEvent(event); } + if (ptaskbaricon) + ptaskbaricon->UpdateTooltip(); } +string StartupShortcutPath() +{ + // Get the startup folder shortcut path + char pszLinkPath[MAX_PATH+100]; + pszLinkPath[0] = '\0'; + SHGetSpecialFolderPath(0, pszLinkPath, CSIDL_STARTUP, 0); + strcat(pszLinkPath, "\\Bitcoin.lnk"); + return pszLinkPath; +} + +bool GetStartOnSystemStartup() +{ + return FileExists(StartupShortcutPath().c_str()); +} -void ApplyUISettings() { - // Show the tray icon? - if (alwaysShowTrayIcon) - taskBarIcon->Show(); - else - taskBarIcon->Hide(); +void SetStartOnSystemStartup(bool fAutoStart) +{ + // If the shortcut exists already, remove it for updating + remove(StartupShortcutPath().c_str()); - // Autostart on system startup? - // Open the startup registry key - HKEY hKey; - LONG lnRes = RegOpenKeyEx( - HKEY_CURRENT_USER, - "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run", - 0, - KEY_ALL_ACCESS, - &hKey - ); + if (fAutoStart) + { + CoInitialize(NULL); - if ( ERROR_SUCCESS == lnRes ) - { - if (startOnSysBoot) { - // Get the current executable path - char exePath[ MAX_PATH ]; - GetModuleFileName(NULL, exePath, _MAX_PATH + 1); - char runCmd[ MAX_PATH + 5 ]; - strcat(runCmd, exePath); - strcat(runCmd," /min"); + // Get a pointer to the IShellLink interface. + HRESULT hres = NULL; + IShellLink* psl = NULL; + hres = CoCreateInstance(CLSID_ShellLink, NULL, + CLSCTX_INPROC_SERVER, IID_IShellLink, + reinterpret_cast(&psl)); - RegSetValueEx(hKey, - "Bitcoin", - 0, - REG_SZ, - (BYTE*)runCmd, - sizeof(runCmd) - ); - } - else { - RegDeleteValue(hKey, "Bitcoin"); - } - } - RegCloseKey(hKey); + if (SUCCEEDED(hres)) + { + // Get the current executable path + char pszExePath[MAX_PATH]; + GetModuleFileName(NULL, pszExePath, sizeof(pszExePath)); + _strlwr(pszExePath); + + // Set the path to the shortcut target + psl->SetPath(pszExePath); + PathRemoveFileSpec(pszExePath); + psl->SetWorkingDirectory(pszExePath); + psl->SetShowCmd(SW_SHOWMINNOACTIVE); + + // Query IShellLink for the IPersistFile interface for + // saving the shortcut in persistent storage. + IPersistFile* ppf = NULL; + hres = psl->QueryInterface(IID_IPersistFile, + reinterpret_cast(&ppf)); + if (SUCCEEDED(hres)) + { + WCHAR pwsz[MAX_PATH]; + // Ensure that the string is ANSI. + MultiByteToWideChar(CP_ACP, 0, StartupShortcutPath().c_str(), -1, pwsz, MAX_PATH); + // Save the link by calling IPersistFile::Save. + hres = ppf->Save(pwsz, TRUE); + ppf->Release(); + } + psl->Release(); + } + CoUninitialize(); + } } -void GenerateBitcoins(bool flag) -{ - fGenerateBitcoins = flag; - nTransactionsUpdated++; - CWalletDB().WriteSetting("fGenerateBitcoins", fGenerateBitcoins); - if (fGenerateBitcoins) - if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) - printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); - taskBarIcon->UpdateTooltip(); -} + + + + diff --git a/ui.h b/ui.h index 11c88147..f89456bf 100644 --- a/ui.h +++ b/ui.h @@ -21,21 +21,25 @@ enum + +extern map mapArgs; + +// Settings +extern int fShowGenerated; +extern int fMinimizeToTray; +extern int fMinimizeOnClose; + + + extern void HandleCtrlA(wxKeyEvent& event); extern string DateTimeStr(int64 nTime); extern string FormatTxStatus(const CWalletTx& wtx); extern void CrossThreadCall(int nID, void* pdata); extern void MainFrameRepaint(); extern void Shutdown(void* parg); -void ApplyUISettings(); -void GenerateBitcoins(bool flag); -// UI settings -extern int minimizeToTray; -extern int closeToTray; -extern int startOnSysBoot; -extern int askBeforeClosing; -extern int alwaysShowTrayIcon; + + @@ -44,14 +48,17 @@ class CMainFrame : public CMainFrameBase protected: // Event handlers void OnClose(wxCloseEvent& event); - void OnIconize( wxIconizeEvent& event ); + void OnIconize(wxIconizeEvent& event); void OnMouseEvents(wxMouseEvent& event); void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } void OnIdle(wxIdleEvent& event); void OnPaint(wxPaintEvent& event); void OnPaintListCtrl(wxPaintEvent& event); void OnMenuFileExit(wxCommandEvent& event); + void OnMenuViewShowGenerated(wxCommandEvent& event); + void OnUpdateUIViewShowGenerated(wxUpdateUIEvent& event); void OnMenuOptionsGenerate(wxCommandEvent& event); + void OnUpdateUIOptionsGenerate(wxUpdateUIEvent& event); void OnMenuOptionsChangeYourAddress(wxCommandEvent& event); void OnMenuOptionsOptions(wxCommandEvent& event); void OnMenuHelpAbout(wxCommandEvent& event); @@ -66,7 +73,6 @@ protected: void OnListItemActivatedProductsSent(wxListEvent& event); void OnListItemActivatedOrdersSent(wxListEvent& event); void OnListItemActivatedOrdersReceived(wxListEvent& event); - void OnUpdateMenuGenerate( wxUpdateUIEvent& event ); public: /** Constructor */ @@ -85,7 +91,6 @@ public: void InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); void RefreshListCtrl(); void RefreshStatus(); - void SendToTray(); }; @@ -107,47 +112,26 @@ public: -class COptionsPanelBitcoin : public COptionsPanelBitcoinBase -{ -protected: - // Event handlers - void OnKillFocusTransactionFee( wxFocusEvent& event ); - -public: - /** Constructor */ - COptionsPanelBitcoin(wxWindow* parent); -}; - - - -class COptionsPanelUI : public COptionsPanelUIBase -{ -protected: - // Event handlers - void OnOptionsChanged( wxCommandEvent& event ); - -public: - /** Constructor */ - COptionsPanelUI(wxWindow* parent); -}; - - - class COptionsDialog : public COptionsDialogBase { protected: // Event handlers - void MenuSelChanged( wxTreeEvent& event ); + void OnListBox(wxCommandEvent& event); + void OnKillFocusTransactionFee(wxFocusEvent& event); + void OnCheckBoxLimitProcessors(wxCommandEvent& event); + void OnCheckBoxMinimizeToTray(wxCommandEvent& event); void OnButtonOK(wxCommandEvent& event); void OnButtonCancel(wxCommandEvent& event); + void OnButtonApply(wxCommandEvent& event); - // Panels - COptionsPanelBitcoin* panelBitcoin; - COptionsPanelUI* panelUI; - wxPanel* currentPanel; public: /** Constructor */ COptionsDialog(wxWindow* parent); + + // Custom + bool fTmpStartOnSystemStartup; + bool fTmpMinimizeOnClose; + void SelectPage(int nPage); }; @@ -180,6 +164,11 @@ protected: public: /** Constructor */ CSendDialog(wxWindow* parent, const wxString& strAddress=""); + + // Custom + bool fEnabledPrev; + string strFromSave; + string strMessageSave; }; @@ -455,22 +444,33 @@ public: -class CBitcoinTBIcon : public wxTaskBarIcon +class CMyTaskBarIcon : public wxTaskBarIcon { protected: - void Restore(); - - // Event handlers - void OnLeftButtonDClick(wxTaskBarIconEvent&); - void OnMenuExit(wxCommandEvent&); - void OnMenuGenerate(wxCommandEvent&); - void OnMenuRestore(wxCommandEvent&); + // Event handlers + void OnLeftButtonDClick(wxTaskBarIconEvent& event); + void OnMenuRestore(wxCommandEvent& event); + void OnUpdateUIGenerate(wxUpdateUIEvent& event); + void OnMenuGenerate(wxCommandEvent& event); + void OnMenuExit(wxCommandEvent& event); public: - void Show(); - void Hide(); - void UpdateTooltip(); - virtual wxMenu *CreatePopupMenu(); + CMyTaskBarIcon() : wxTaskBarIcon() + { + Show(true); + } + + void Show(bool fShow=true); + void Hide(); + void Restore(); + void UpdateTooltip(); + virtual wxMenu* CreatePopupMenu(); DECLARE_EVENT_TABLE() }; + + + + + + diff --git a/uibase.cpp b/uibase.cpp index a88c67eb..0f9e22c5 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -24,14 +24,21 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_menubar->Append( m_menuFile, wxT("&File") ); + m_menuView = new wxMenu(); + wxMenuItem* m_menuViewShowGenerated; + m_menuViewShowGenerated = new wxMenuItem( m_menuView, wxID_VIEWSHOWGENERATED, wxString( wxT("&Show Generated Coins") ) , wxEmptyString, wxITEM_CHECK ); + m_menuView->Append( m_menuViewShowGenerated ); + + m_menubar->Append( m_menuView, wxT("&View") ); + m_menuOptions = new wxMenu(); wxMenuItem* m_menuOptionsGenerateBitcoins; m_menuOptionsGenerateBitcoins = new wxMenuItem( m_menuOptions, wxID_OPTIONSGENERATEBITCOINS, wxString( wxT("&Generate Coins") ) , wxEmptyString, wxITEM_CHECK ); m_menuOptions->Append( m_menuOptionsGenerateBitcoins ); - wxMenuItem* m_menuChangeYourAddress; - m_menuChangeYourAddress = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( wxT("&Change Your Address...") ) , wxEmptyString, wxITEM_NORMAL ); - m_menuOptions->Append( m_menuChangeYourAddress ); + wxMenuItem* m_menuOptionsChangeYourAddress; + m_menuOptionsChangeYourAddress = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( wxT("&Change Your Address...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptions->Append( m_menuOptionsChangeYourAddress ); wxMenuItem* m_menuOptionsOptions; m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( wxT("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); @@ -225,9 +232,11 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& this->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); this->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaint ) ); this->Connect( m_menuFileExit->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuFileExit ) ); + this->Connect( m_menuViewShowGenerated->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuViewShowGenerated ) ); + this->Connect( m_menuViewShowGenerated->GetId(), wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIViewShowGenerated ) ); this->Connect( m_menuOptionsGenerateBitcoins->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsGenerate ) ); - this->Connect( m_menuOptionsGenerateBitcoins->GetId(), wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateMenuGenerate ) ); - this->Connect( m_menuChangeYourAddress->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsChangeYourAddress ) ); + this->Connect( m_menuOptionsGenerateBitcoins->GetId(), wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIOptionsGenerate ) ); + this->Connect( m_menuOptionsChangeYourAddress->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsChangeYourAddress ) ); this->Connect( m_menuOptionsOptions->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsOptions ) ); this->Connect( m_menuHelpAbout->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuHelpAbout ) ); this->Connect( wxID_BUTTONSEND, wxEVT_COMMAND_TOOL_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonSend ) ); @@ -278,8 +287,10 @@ CMainFrameBase::~CMainFrameBase() this->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); this->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaint ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuFileExit ) ); + this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuViewShowGenerated ) ); + this->Disconnect( wxID_ANY, wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIViewShowGenerated ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsGenerate ) ); - this->Disconnect( wxID_ANY, wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateMenuGenerate ) ); + this->Disconnect( wxID_ANY, wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIOptionsGenerate ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsChangeYourAddress ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsOptions ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuHelpAbout ) ); @@ -353,15 +364,121 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w wxBoxSizer* bSizer55; bSizer55 = new wxBoxSizer( wxVERTICAL ); - panelSizer = new wxBoxSizer( wxHORIZONTAL ); + wxBoxSizer* bSizer66; + bSizer66 = new wxBoxSizer( wxHORIZONTAL ); + + m_listBox = new wxListBox( this, wxID_ANY, wxDefaultPosition, wxSize( 110,-1 ), 0, NULL, wxLB_NEEDED_SB|wxLB_SINGLE ); + bSizer66->Add( m_listBox, 0, wxEXPAND|wxRIGHT, 5 ); + + m_scrolledWindow = new wxScrolledWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 ); + m_scrolledWindow->SetScrollRate( 5, 5 ); + wxBoxSizer* bSizer63; + bSizer63 = new wxBoxSizer( wxVERTICAL ); + + m_panelMain = new wxPanel( m_scrolledWindow, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer69; + bSizer69 = new wxBoxSizer( wxVERTICAL ); + + + bSizer69->Add( 0, 14, 0, wxEXPAND, 5 ); + + m_staticText32 = new wxStaticText( m_panelMain, wxID_ANY, wxT("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText32->Wrap( -1 ); + m_staticText32->Hide(); + + bSizer69->Add( m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + wxBoxSizer* bSizer56; + bSizer56 = new wxBoxSizer( wxHORIZONTAL ); + + m_staticText31 = new wxStaticText( m_panelMain, wxID_ANY, wxT("Transaction fee:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText31->Wrap( -1 ); + m_staticText31->Hide(); + + bSizer56->Add( m_staticText31, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 ); + + m_textCtrlTransactionFee = new wxTextCtrl( m_panelMain, wxID_TRANSACTIONFEE, wxEmptyString, wxDefaultPosition, wxSize( 70,-1 ), 0 ); + m_textCtrlTransactionFee->Hide(); + + bSizer56->Add( m_textCtrlTransactionFee, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + bSizer69->Add( bSizer56, 0, wxEXPAND, 5 ); + + wxBoxSizer* bSizer71; + bSizer71 = new wxBoxSizer( wxHORIZONTAL ); + + m_checkBoxLimitProcessors = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Limit coin generation to"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer71->Add( m_checkBoxLimitProcessors, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + m_spinCtrlLimitProcessors = new wxSpinCtrl( m_panelMain, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( 48,-1 ), wxSP_ARROW_KEYS, 1, 999, 1 ); + bSizer71->Add( m_spinCtrlLimitProcessors, 0, wxTOP|wxBOTTOM, 5 ); + + m_staticText35 = new wxStaticText( m_panelMain, wxID_ANY, wxT("processors"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText35->Wrap( -1 ); + bSizer71->Add( m_staticText35, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + + bSizer69->Add( bSizer71, 0, 0, 5 ); + + m_checkBoxStartOnSystemStartup = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Start Bitcoin on system startup"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer69->Add( m_checkBoxStartOnSystemStartup, 0, wxALL, 5 ); + + m_checkBoxMinimizeToTray = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Minimize to the system tray instead of the taskbar"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer69->Add( m_checkBoxMinimizeToTray, 0, wxALL, 5 ); + + wxBoxSizer* bSizer101; + bSizer101 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer101->Add( 16, 0, 0, 0, 5 ); + + m_checkBoxMinimizeOnClose = new wxCheckBox( m_panelMain, wxID_ANY, wxT("M&inimize to system tray on close"), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer101->Add( m_checkBoxMinimizeOnClose, 0, wxALL, 5 ); + + bSizer69->Add( bSizer101, 1, wxEXPAND, 5 ); + + m_panelMain->SetSizer( bSizer69 ); + m_panelMain->Layout(); + bSizer69->Fit( m_panelMain ); + bSizer63->Add( m_panelMain, 0, wxEXPAND, 5 ); + + m_panelTest2 = new wxPanel( m_scrolledWindow, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer64; + bSizer64 = new wxBoxSizer( wxVERTICAL ); + + + bSizer64->Add( 0, 14, 0, wxEXPAND, 5 ); + + m_staticText321 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("Test panel 2 for future expansion"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText321->Wrap( -1 ); + bSizer64->Add( m_staticText321, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); - m_treeCtrl = new wxTreeCtrl( this, wxID_ANY, wxDefaultPosition, wxSize( 100,-1 ), wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT ); - panelSizer->Add( m_treeCtrl, 0, wxALL|wxEXPAND, 5 ); + m_staticText69 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText69->Wrap( -1 ); + bSizer64->Add( m_staticText69, 0, wxALL, 5 ); - bSizer55->Add( panelSizer, 1, wxEXPAND, 5 ); + m_staticText70 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText70->Wrap( -1 ); + bSizer64->Add( m_staticText70, 0, wxALL, 5 ); - m_staticline1 = new wxStaticLine( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLI_HORIZONTAL ); - bSizer55->Add( m_staticline1, 0, wxEXPAND | wxALL, 5 ); + m_staticText71 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText71->Wrap( -1 ); + bSizer64->Add( m_staticText71, 0, wxALL, 5 ); + + m_panelTest2->SetSizer( bSizer64 ); + m_panelTest2->Layout(); + bSizer64->Fit( m_panelTest2 ); + bSizer63->Add( m_panelTest2, 0, wxEXPAND, 5 ); + + m_scrolledWindow->SetSizer( bSizer63 ); + m_scrolledWindow->Layout(); + bSizer63->Fit( m_scrolledWindow ); + bSizer66->Add( m_scrolledWindow, 1, wxEXPAND|wxLEFT, 5 ); + + bSizer55->Add( bSizer66, 1, wxEXPAND|wxALL, 9 ); wxBoxSizer* bSizer58; bSizer58 = new wxBoxSizer( wxHORIZONTAL ); @@ -374,9 +491,8 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer58->Add( m_buttonCancel, 0, wxALL, 5 ); - m_buttonApply = new wxButton( this, wxID_ANY, wxT("Apply"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonApply->Enable( false ); - m_buttonApply->Hide(); + m_buttonApply = new wxButton( this, wxID_APPLY, wxT("&Apply"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonApply->SetMinSize( wxSize( 85,25 ) ); bSizer58->Add( m_buttonApply, 0, wxALL, 5 ); @@ -386,17 +502,25 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w this->Layout(); // Connect Events - m_treeCtrl->Connect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( COptionsDialogBase::MenuSelChanged ), NULL, this ); + m_listBox->Connect( wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler( COptionsDialogBase::OnListBox ), NULL, this ); + m_textCtrlTransactionFee->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusTransactionFee ), NULL, this ); + m_checkBoxLimitProcessors->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxLimitProcessors ), NULL, this ); + m_checkBoxMinimizeToTray->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxMinimizeToTray ), NULL, this ); m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonOK ), NULL, this ); m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonCancel ), NULL, this ); + m_buttonApply->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonApply ), NULL, this ); } COptionsDialogBase::~COptionsDialogBase() { // Disconnect Events - m_treeCtrl->Disconnect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( COptionsDialogBase::MenuSelChanged ), NULL, this ); + m_listBox->Disconnect( wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler( COptionsDialogBase::OnListBox ), NULL, this ); + m_textCtrlTransactionFee->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusTransactionFee ), NULL, this ); + m_checkBoxLimitProcessors->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxLimitProcessors ), NULL, this ); + m_checkBoxMinimizeToTray->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxMinimizeToTray ), NULL, this ); m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonOK ), NULL, this ); m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonCancel ), NULL, this ); + m_buttonApply->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonApply ), NULL, this ); } CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) @@ -1815,78 +1939,3 @@ CGetTextFromUserDialogBase::~CGetTextFromUserDialogBase() m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CGetTextFromUserDialogBase::OnButtonOK ), NULL, this ); m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CGetTextFromUserDialogBase::OnButtonCancel ), NULL, this ); } - -COptionsPanelBitcoinBase::COptionsPanelBitcoinBase( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style ) : wxPanel( parent, id, pos, size, style ) -{ - wxBoxSizer* bSizer62; - bSizer62 = new wxBoxSizer( wxVERTICAL ); - - - bSizer62->Add( 0, 20, 0, wxEXPAND, 5 ); - - m_staticText32 = new wxStaticText( this, wxID_ANY, wxT("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText32->Wrap( -1 ); - bSizer62->Add( m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); - - wxBoxSizer* bSizer56; - bSizer56 = new wxBoxSizer( wxHORIZONTAL ); - - m_staticText31 = new wxStaticText( this, wxID_ANY, wxT("Transaction fee:"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText31->Wrap( -1 ); - bSizer56->Add( m_staticText31, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 ); - - m_textCtrlTransactionFee = new wxTextCtrl( this, wxID_TRANSACTIONFEE, wxEmptyString, wxDefaultPosition, wxSize( 70,-1 ), 0 ); - bSizer56->Add( m_textCtrlTransactionFee, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - bSizer62->Add( bSizer56, 0, wxEXPAND, 5 ); - - this->SetSizer( bSizer62 ); - this->Layout(); - bSizer62->Fit( this ); - - // Connect Events - m_textCtrlTransactionFee->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsPanelBitcoinBase::OnKillFocusTransactionFee ), NULL, this ); -} - -COptionsPanelBitcoinBase::~COptionsPanelBitcoinBase() -{ - // Disconnect Events - m_textCtrlTransactionFee->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsPanelBitcoinBase::OnKillFocusTransactionFee ), NULL, this ); -} - -COptionsPanelUIBase::COptionsPanelUIBase( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style ) : wxPanel( parent, id, pos, size, style ) -{ - wxBoxSizer* bSizer57; - bSizer57 = new wxBoxSizer( wxVERTICAL ); - - - bSizer57->Add( 0, 20, 1, wxEXPAND, 5 ); - - m_checkMinToTray = new wxCheckBox( this, wxID_MINTOTRAY, wxT("Minimize to tray"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer57->Add( m_checkMinToTray, 0, wxALL, 5 ); - - m_checkCloseToTray = new wxCheckBox( this, wxID_ANY, wxT("Close to tray"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer57->Add( m_checkCloseToTray, 0, wxALL, 5 ); - - m_checkStartOnSysBoot = new wxCheckBox( this, wxID_ANY, wxT("Start with Windows"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer57->Add( m_checkStartOnSysBoot, 0, wxALL, 5 ); - - m_checkAskBeforeClosing = new wxCheckBox( this, wxID_ANY, wxT("Ask before closing"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer57->Add( m_checkAskBeforeClosing, 0, wxALL, 5 ); - - m_checkAlwaysShowTray = new wxCheckBox( this, wxID_ANY, wxT("Always show tray icon"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer57->Add( m_checkAlwaysShowTray, 0, wxALL, 5 ); - - this->SetSizer( bSizer57 ); - this->Layout(); - bSizer57->Fit( this ); -} - -COptionsPanelUIBase::~COptionsPanelUIBase() -{ -} diff --git a/uibase.h b/uibase.h index 61e05522..e262a3dc 100644 --- a/uibase.h +++ b/uibase.h @@ -30,69 +30,70 @@ #include #include #include -#include -#include +#include +#include +#include +#include #include #include -#include #include -#include +#include /////////////////////////////////////////////////////////////////////////// #define wxID_MAINFRAME 1000 -#define wxID_OPTIONSGENERATEBITCOINS 1001 -#define wxID_BUTTONSEND 1002 -#define wxID_BUTTONRECEIVE 1003 -#define wxID_TEXTCTRLADDRESS 1004 -#define wxID_BUTTONCOPY 1005 -#define wxID_BUTTONCHANGE 1006 -#define wxID_TEXTCTRLPAYTO 1007 -#define wxID_BUTTONPASTE 1008 -#define wxID_BUTTONADDRESSBOOK 1009 -#define wxID_TEXTCTRLAMOUNT 1010 -#define wxID_CHOICETRANSFERTYPE 1011 -#define wxID_LISTCTRL 1012 -#define wxID_BUTTONRENAME 1013 -#define wxID_BUTTONNEW 1014 -#define wxID_BUTTONEDIT 1015 -#define wxID_BUTTONDELETE 1016 -#define wxID_DEL0 1017 -#define wxID_DEL1 1018 -#define wxID_DEL2 1019 -#define wxID_DEL3 1020 -#define wxID_DEL4 1021 -#define wxID_DEL5 1022 -#define wxID_DEL6 1023 -#define wxID_DEL7 1024 -#define wxID_DEL8 1025 -#define wxID_DEL9 1026 -#define wxID_DEL10 1027 -#define wxID_DEL11 1028 -#define wxID_DEL12 1029 -#define wxID_DEL13 1030 -#define wxID_DEL14 1031 -#define wxID_DEL15 1032 -#define wxID_DEL16 1033 -#define wxID_DEL17 1034 -#define wxID_DEL18 1035 -#define wxID_DEL19 1036 -#define wxID_BUTTONPREVIEW 1037 -#define wxID_BUTTONSAMPLE 1038 -#define wxID_CANCEL2 1039 -#define wxID_BUTTONBACK 1040 -#define wxID_BUTTONNEXT 1041 -#define wxID_SUBMIT 1042 -#define wxID_OPENNEWTABLE 1043 -#define wxID_DEALHAND 1044 -#define wxID_FOLD 1045 -#define wxID_CALL 1046 -#define wxID_RAISE 1047 -#define wxID_LEAVETABLE 1048 -#define wxID_DITCHPLAYER 1049 -#define wxID_TEXTCTRL 1050 -#define wxID_TRANSACTIONFEE 1051 -#define wxID_MINTOTRAY 1052 +#define wxID_VIEWSHOWGENERATED 1001 +#define wxID_OPTIONSGENERATEBITCOINS 1002 +#define wxID_BUTTONSEND 1003 +#define wxID_BUTTONRECEIVE 1004 +#define wxID_TEXTCTRLADDRESS 1005 +#define wxID_BUTTONCOPY 1006 +#define wxID_BUTTONCHANGE 1007 +#define wxID_TRANSACTIONFEE 1008 +#define wxID_TEXTCTRLPAYTO 1009 +#define wxID_BUTTONPASTE 1010 +#define wxID_BUTTONADDRESSBOOK 1011 +#define wxID_TEXTCTRLAMOUNT 1012 +#define wxID_CHOICETRANSFERTYPE 1013 +#define wxID_LISTCTRL 1014 +#define wxID_BUTTONRENAME 1015 +#define wxID_BUTTONNEW 1016 +#define wxID_BUTTONEDIT 1017 +#define wxID_BUTTONDELETE 1018 +#define wxID_DEL0 1019 +#define wxID_DEL1 1020 +#define wxID_DEL2 1021 +#define wxID_DEL3 1022 +#define wxID_DEL4 1023 +#define wxID_DEL5 1024 +#define wxID_DEL6 1025 +#define wxID_DEL7 1026 +#define wxID_DEL8 1027 +#define wxID_DEL9 1028 +#define wxID_DEL10 1029 +#define wxID_DEL11 1030 +#define wxID_DEL12 1031 +#define wxID_DEL13 1032 +#define wxID_DEL14 1033 +#define wxID_DEL15 1034 +#define wxID_DEL16 1035 +#define wxID_DEL17 1036 +#define wxID_DEL18 1037 +#define wxID_DEL19 1038 +#define wxID_BUTTONPREVIEW 1039 +#define wxID_BUTTONSAMPLE 1040 +#define wxID_CANCEL2 1041 +#define wxID_BUTTONBACK 1042 +#define wxID_BUTTONNEXT 1043 +#define wxID_SUBMIT 1044 +#define wxID_OPENNEWTABLE 1045 +#define wxID_DEALHAND 1046 +#define wxID_FOLD 1047 +#define wxID_CALL 1048 +#define wxID_RAISE 1049 +#define wxID_LEAVETABLE 1050 +#define wxID_DITCHPLAYER 1051 +#define wxID_TEXTCTRL 1052 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase @@ -104,6 +105,7 @@ class CMainFrameBase : public wxFrame protected: wxMenuBar* m_menubar; wxMenu* m_menuFile; + wxMenu* m_menuView; wxMenu* m_menuHelp; wxToolBar* m_toolBar; wxStatusBar* m_statusBar; @@ -132,8 +134,10 @@ class CMainFrameBase : public wxFrame virtual void OnMouseEvents( wxMouseEvent& event ){ event.Skip(); } virtual void OnPaint( wxPaintEvent& event ){ event.Skip(); } virtual void OnMenuFileExit( wxCommandEvent& event ){ event.Skip(); } + virtual void OnMenuViewShowGenerated( wxCommandEvent& event ){ event.Skip(); } + virtual void OnUpdateUIViewShowGenerated( wxUpdateUIEvent& event ){ event.Skip(); } virtual void OnMenuOptionsGenerate( wxCommandEvent& event ){ event.Skip(); } - virtual void OnUpdateMenuGenerate( wxUpdateUIEvent& event ){ event.Skip(); } + virtual void OnUpdateUIOptionsGenerate( wxUpdateUIEvent& event ){ event.Skip(); } virtual void OnMenuOptionsChangeYourAddress( wxCommandEvent& event ){ event.Skip(); } virtual void OnMenuOptionsOptions( wxCommandEvent& event ){ event.Skip(); } virtual void OnMenuHelpAbout( wxCommandEvent& event ){ event.Skip(); } @@ -193,21 +197,42 @@ class COptionsDialogBase : public wxDialog private: protected: - wxBoxSizer* panelSizer; - wxTreeCtrl* m_treeCtrl; - wxStaticLine* m_staticline1; + wxListBox* m_listBox; + wxScrolledWindow* m_scrolledWindow; + wxPanel* m_panelMain; + + wxStaticText* m_staticText32; + wxStaticText* m_staticText31; + wxTextCtrl* m_textCtrlTransactionFee; + wxCheckBox* m_checkBoxLimitProcessors; + wxSpinCtrl* m_spinCtrlLimitProcessors; + wxStaticText* m_staticText35; + wxCheckBox* m_checkBoxStartOnSystemStartup; + wxCheckBox* m_checkBoxMinimizeToTray; + + wxCheckBox* m_checkBoxMinimizeOnClose; + wxPanel* m_panelTest2; + + wxStaticText* m_staticText321; + wxStaticText* m_staticText69; + wxStaticText* m_staticText70; + wxStaticText* m_staticText71; wxButton* m_buttonOK; wxButton* m_buttonCancel; wxButton* m_buttonApply; // Virtual event handlers, overide them in your derived class - virtual void MenuSelChanged( wxTreeEvent& event ){ event.Skip(); } + virtual void OnListBox( wxCommandEvent& event ){ event.Skip(); } + virtual void OnKillFocusTransactionFee( wxFocusEvent& event ){ event.Skip(); } + virtual void OnCheckBoxLimitProcessors( wxCommandEvent& event ){ event.Skip(); } + virtual void OnCheckBoxMinimizeToTray( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonApply( wxCommandEvent& event ){ event.Skip(); } public: - COptionsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Options"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 555,377 ), long style = wxDEFAULT_DIALOG_STYLE ); + COptionsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Options"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 540,360 ), long style = wxDEFAULT_DIALOG_STYLE ); ~COptionsDialogBase(); }; @@ -720,48 +745,4 @@ class CGetTextFromUserDialogBase : public wxDialog }; -/////////////////////////////////////////////////////////////////////////////// -/// Class COptionsPanelBitcoinBase -/////////////////////////////////////////////////////////////////////////////// -class COptionsPanelBitcoinBase : public wxPanel -{ - private: - - protected: - - wxStaticText* m_staticText32; - wxStaticText* m_staticText31; - - // Virtual event handlers, overide them in your derived class - virtual void OnKillFocusTransactionFee( wxFocusEvent& event ){ event.Skip(); } - - - public: - wxTextCtrl* m_textCtrlTransactionFee; - COptionsPanelBitcoinBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( -1,-1 ), long style = wxTAB_TRAVERSAL ); - ~COptionsPanelBitcoinBase(); - -}; - -/////////////////////////////////////////////////////////////////////////////// -/// Class COptionsPanelUIBase -/////////////////////////////////////////////////////////////////////////////// -class COptionsPanelUIBase : public wxPanel -{ - private: - - protected: - - - public: - wxCheckBox* m_checkMinToTray; - wxCheckBox* m_checkCloseToTray; - wxCheckBox* m_checkStartOnSysBoot; - wxCheckBox* m_checkAskBeforeClosing; - wxCheckBox* m_checkAlwaysShowTray; - COptionsPanelUIBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( -1,-1 ), long style = wxTAB_TRAVERSAL ); - ~COptionsPanelUIBase(); - -}; - #endif //__uibase__ diff --git a/uiproject.fbp b/uiproject.fbp index 315cc725..af5876b4 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -18,7 +18,7 @@ 1 0 0 - + wxSYS_COLOUR_BTNFACE @@ -134,6 +134,26 @@ + + &View + m_menuView + protected + + + 0 + 1 + + wxID_VIEWSHOWGENERATED + wxITEM_CHECK + &Show Generated Coins + m_menuViewShowGenerated + none + + + OnMenuViewShowGenerated + OnUpdateUIViewShowGenerated + + &Options m_menuOptions @@ -151,7 +171,7 @@ OnMenuOptionsGenerate - OnUpdateMenuGenerate + OnUpdateUIOptionsGenerate @@ -161,7 +181,7 @@ wxID_ANY wxITEM_NORMAL &Change Your Address... - m_menuChangeYourAddress + m_menuOptionsChangeYourAddress none @@ -1679,7 +1699,7 @@ COptionsDialogBase - 555,377 + 540,360 wxDEFAULT_DIALOG_STYLE Options @@ -1723,20 +1743,21 @@ wxVERTICAL none - 5 - wxEXPAND + 9 + wxEXPAND|wxALL 1 - panelSizer + bSizer66 wxHORIZONTAL - protected + none 5 - wxALL|wxEXPAND + wxEXPAND|wxRIGHT 0 - + + 1 @@ -1745,11 +1766,11 @@ wxID_ANY - m_treeCtrl + m_listBox protected - 100,-1 - wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT + 110,-1 + wxLB_NEEDED_SB|wxLB_SINGLE @@ -1765,6 +1786,8 @@ + OnListBox + @@ -1777,79 +1800,911 @@ - - - - - - - - - - - - - - - - - - MenuSelChanged - - - - - - - 5 - wxEXPAND | wxALL - 0 - - - - 1 - - - 0 - wxID_ANY - - - m_staticline1 - protected - - - wxLI_HORIZONTAL - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + 5 + wxEXPAND|wxLEFT + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_scrolledWindow + protected + + 5 + 5 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer63 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + + + 1 + + + 0 + wxID_ANY + + + m_panelMain + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer69 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 14 + protected + 0 + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + + 1 + + + 1 + wxID_ANY + Optional transaction fee you give to the nodes that process your transactions. + + + m_staticText32 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + bSizer56 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + + 1 + wxID_ANY + Transaction fee: + + + m_staticText31 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 1 + wxID_TRANSACTIONFEE + + 0 + + m_textCtrlTransactionFee + protected + + 70,-1 + + + + + + + + + + + + + OnKillFocusTransactionFee + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + + 0 + + + bSizer71 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + &Limit coin generation to + + + m_checkBoxLimitProcessors + protected + + + + + + + + + + OnCheckBoxLimitProcessors + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxTOP|wxBOTTOM + 0 + + + + 1 + + + 0 + wxID_ANY + 1 + 999 + + 1 + + m_spinCtrlLimitProcessors + protected + + 48,-1 + wxSP_ARROW_KEYS + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + processors + + + m_staticText35 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + &Start Bitcoin on system startup + + + m_checkBoxStartOnSystemStartup + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + &Minimize to the system tray instead of the taskbar + + + m_checkBoxMinimizeToTray + protected + + + + + + + + + + OnCheckBoxMinimizeToTray + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + + bSizer101 + wxHORIZONTAL + none + + 5 + + 0 + + 0 + protected + 16 + + + + 5 + wxALL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + M&inimize to system tray on close + + + m_checkBoxMinimizeOnClose + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + + + 1 + + + 0 + wxID_ANY + + + m_panelTest2 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer64 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 14 + protected + 0 + + + + 5 + wxALIGN_CENTER_VERTICAL|wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + Test panel 2 for future expansion + + + m_staticText321 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + MyLabel + + + m_staticText69 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + MyLabel + + + m_staticText70 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL + 0 + + + + 1 + + + 0 + wxID_ANY + MyLabel + + + m_staticText71 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -1973,25 +2828,25 @@ 0 - 0 + 1 - 1 - wxID_ANY - Apply + 0 + wxID_APPLY + &Apply - + 85,25 m_buttonApply protected - + -1,-1 - + OnButtonApply @@ -11849,551 +12704,5 @@ - - - - 1 - - - 0 - wxID_ANY - - - COptionsPanelBitcoinBase - - -1,-1 - - - - - wxTAB_TRAVERSAL - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer62 - wxVERTICAL - none - - 5 - wxEXPAND - 0 - - 20 - protected - 0 - - - - 5 - wxALIGN_CENTER_VERTICAL|wxALL - 0 - - - - 1 - - - 0 - wxID_ANY - Optional transaction fee you give to the nodes that process your transactions. - - - m_staticText32 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxEXPAND - 0 - - - bSizer56 - wxHORIZONTAL - none - - 5 - wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Transaction fee: - - - m_staticText31 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_TRANSACTIONFEE - - 0 - - m_textCtrlTransactionFee - public - - 70,-1 - - - - - - - - - - - - - OnKillFocusTransactionFee - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 1 - - - 0 - wxID_ANY - - - COptionsPanelUIBase - - -1,-1 - - - - - wxTAB_TRAVERSAL - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer57 - wxVERTICAL - none - - 5 - wxEXPAND - 1 - - 20 - protected - 0 - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_MINTOTRAY - Minimize to tray - - - m_checkMinToTray - public - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - Close to tray - - - m_checkCloseToTray - public - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - Start with Windows - - - m_checkStartOnSysBoot - public - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - Ask before closing - - - m_checkAskBeforeClosing - public - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - Always show tray icon - - - m_checkAlwaysShowTray - public - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/util.cpp b/util.cpp index 9c0ab142..8271474e 100644 --- a/util.cpp +++ b/util.cpp @@ -86,7 +86,7 @@ void RandAddSeed(bool fPerfmon) struct tm* ptmTime = gmtime(&nTime); char pszTime[200]; strftime(pszTime, sizeof(pszTime), "%x %H:%M:%S", ptmTime); - printf("%s RandAddSeed() got %d bytes of performance data\n", pszTime, nSize); + printf("%s RandAddSeed() %d bytes\n", pszTime, nSize); } } } @@ -174,7 +174,7 @@ bool error(const char* format, ...) void PrintException(std::exception* pex, const char* pszThread) { - char pszModule[260]; + char pszModule[MAX_PATH]; pszModule[0] = '\0'; GetModuleFileName(NULL, pszModule, sizeof(pszModule)); _strlwr(pszModule); From e39dfe8ea64a360978db0239fbd6ea4ef24de55d Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sat, 24 Oct 2009 16:50:39 +0000 Subject: [PATCH 010/133] Removed autorun regkey creation git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@16 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- setup.nsi | 2 -- 1 file changed, 2 deletions(-) diff --git a/setup.nsi b/setup.nsi index 56a46ea3..023914fa 100644 --- a/setup.nsi +++ b/setup.nsi @@ -64,7 +64,6 @@ Section -Main SEC0000 File libeay32.dll File mingwm10.dll WriteRegStr HKCU "${REGKEY}\Components" Main 1 - WriteRegStr HKCU SOFTWARE\Microsoft\Windows\CurrentVersion\Run Bitcoin "$INSTDIR\bitcoin.exe /min" SectionEnd Section -post SEC0001 @@ -105,7 +104,6 @@ Section /o -un.Main UNSEC0000 Delete /REBOOTOK $INSTDIR\libeay32.dll Delete /REBOOTOK $INSTDIR\bitcoin.exe DeleteRegValue HKCU "${REGKEY}\Components" Main - DeleteRegValue HKCU SOFTWARE\Microsoft\Windows\CurrentVersion\Run Bitcoin SectionEnd Section -un.post UNSEC0001 From fa2a0338d3f8b1c3a1d75bff39ff42e436cee0dc Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 25 Oct 2009 04:35:01 +0000 Subject: [PATCH 011/133] fix display of new generated coins, fix assertion in bitcoinminer git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@17 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 45 +++++++++++++++++++++---------------------- serialize.h | 2 +- ui.cpp | 55 +++++++++++++++++++++++++++++++++++++++++++++++++---- ui.h | 1 + 4 files changed, 75 insertions(+), 28 deletions(-) diff --git a/main.cpp b/main.cpp index de5a5939..dbdee702 100644 --- a/main.cpp +++ b/main.cpp @@ -1129,9 +1129,6 @@ bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) } } - // Notify UI to update prev block coinbase if it was ours - vWalletUpdated.push_back(hashBestChain); - // New best link hashBestChain = hash; pindexBest = pindexNew; @@ -1143,10 +1140,18 @@ bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) txdb.TxnCommit(); txdb.Close(); - // Relay wallet transactions that haven't gotten in yet if (pindexNew == pindexBest) + { + // Relay wallet transactions that haven't gotten in yet RelayWalletTransactions(); + // Notify UI to display prev block's coinbase if it was ours + static uint256 hashPrevBestCoinBase; + CRITICAL_BLOCK(cs_mapWallet) + vWalletUpdated.push_back(hashPrevBestCoinBase); + hashPrevBestCoinBase = vtx[0].GetHash(); + } + MainFrameRepaint(); return true; } @@ -2074,13 +2079,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) else { // Ignore unknown commands for extensibility - printf("ProcessMessage(%s) : Ignored unknown message\n", strCommand.c_str()); } - - if (!vRecv.empty()) - printf("ProcessMessage(%s) : %d extra bytes\n", strCommand.c_str(), vRecv.size()); - return true; } @@ -2349,7 +2349,7 @@ bool BitcoinMiner() } pblock->nBits = nBits; pblock->vtx[0].vout[0].nValue = pblock->GetBlockValue(nFees); - printf("\n\nRunning BitcoinMiner with %d transactions in block\n", pblock->vtx.size()); + printf("Running BitcoinMiner with %d transactions in block\n", pblock->vtx.size()); // @@ -2408,20 +2408,17 @@ bool BitcoinMiner() SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL); CRITICAL_BLOCK(cs_main) { - if (pindexPrev != pindexBest) + if (pindexPrev == pindexBest) { - SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); - break; + // Save key + if (!AddKey(key)) + return false; + key.MakeNewKey(); + + // Process this block the same as if we had received it from another node + if (!ProcessBlock(NULL, pblock.release())) + printf("ERROR in BitcoinMiner, ProcessBlock, block not accepted\n"); } - - // Save key - if (!AddKey(key)) - return false; - key.MakeNewKey(); - - // Process this block the same as if we had received it from another node - if (!ProcessBlock(NULL, pblock.release())) - printf("ERROR in BitcoinMiner, ProcessBlock, block not accepted\n"); } SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); @@ -2439,8 +2436,10 @@ bool BitcoinMiner() break; if (nTransactionsUpdated != nTransactionsUpdatedLast && GetTime() - nStart > 60) break; - if (!fGenerateBitcoins) + if (vNodes.empty()) break; + if (!fGenerateBitcoins) + return true; if (fLimitProcessors && vnThreadsRunning[3] > nLimitProcessors) return true; tmp.block.nTime = pblock->nTime = max(pindexPrev->GetMedianTimePast()+1, GetAdjustedTime()); diff --git a/serialize.h b/serialize.h index 9b20e2a0..b7ab86d2 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 106; +static const int VERSION = 105; diff --git a/ui.cpp b/ui.cpp index ce43a786..42830479 100644 --- a/ui.cpp +++ b/ui.cpp @@ -306,6 +306,8 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) // Init column headers int nDateWidth = DateTimeStr(1229413914).size() * 6 + 8; + if (!strstr(DateTimeStr(1229413914).c_str(), "2008")) + nDateWidth += 12; m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, 0); m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, 0); m_listCtrl->InsertColumn(2, "Status", wxLIST_FORMAT_LEFT, 90); @@ -441,12 +443,33 @@ void CMainFrame::InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSo m_listCtrl->SetItemData(nIndex, nData); } +bool CMainFrame::DeleteLine(uint256 hashKey) +{ + long nData = *(long*)&hashKey; + + // Find item + int nIndex = -1; + while ((nIndex = m_listCtrl->FindItem(nIndex, nData)) != -1) + if (GetItemText(m_listCtrl, nIndex, 1) == hashKey.ToString()) + break; + + if (nIndex != -1) + m_listCtrl->DeleteItem(nIndex); + + return nIndex != -1; +} + string FormatTxStatus(const CWalletTx& wtx) { // Status int nDepth = wtx.GetDepthInMainChain(); if (!wtx.IsFinal()) - return strprintf("Open for %d blocks", nBestHeight - wtx.nLockTime); + { + if (wtx.nLockTime < 500000000) + return strprintf("Open for %d blocks", nBestHeight - wtx.nLockTime); + else + return strprintf("Open until %s", DateTimeStr(wtx.nLockTime).c_str()); + } else if (nDepth < 6) return strprintf("%d/unconfirmed", nDepth); else @@ -503,7 +526,11 @@ void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) // are special because if their block is not accepted, they are not valid. // if (wtx.GetDepthInMainChain() < 2) + { + // In case it was previously displayed + DeleteLine(hash); return; + } } // Find the block the tx is in @@ -800,6 +827,17 @@ void CMainFrame::OnPaint(wxPaintEvent& event) event.Skip(); } +void DelayedRepaint(void* parg) +{ + static bool fOneThread; + if (fOneThread) + return; + fOneThread = true; + Sleep(1000); + MainFrameRepaint(); + fOneThread = false; +} + void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) { // Update listctrl contents @@ -824,7 +862,7 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) // Update status bar string strGen = ""; if (fGenerateBitcoins) - strGen = " Generating"; + strGen = " Generating"; if (fGenerateBitcoins && vNodes.empty()) strGen = "(not connected)"; m_statusBar->SetStatusText(strGen, 1); @@ -833,8 +871,16 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) m_statusBar->SetStatusText(strStatus, 2); // Balance total + bool fRefreshed = false; TRY_CRITICAL_BLOCK(cs_mapWallet) + { m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); + fRefreshed = true; + } + + // mapWallet was locked, try again later + if (!vWalletUpdated.empty() || !fRefreshed) + _beginthread(DelayedRepaint, 0, NULL); m_listCtrl->OnPaint(event); } @@ -1414,7 +1460,7 @@ void COptionsDialog::OnButtonApply(wxCommandEvent& event) CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) { - m_staticTextVersion->SetLabel(strprintf("version 0.%d.%d Alpha", VERSION/100, VERSION%100)); + m_staticTextVersion->SetLabel(strprintf("version 0.%d.%d Beta", VERSION/100, VERSION%100)); // Workaround until upgrade to wxWidgets supporting UTF-8 wxString str = m_staticTextMain->GetLabel(); @@ -3358,6 +3404,8 @@ bool CMyApp::OnInit2() return false; } + //RandAddSeedPerfmon(); + if (!StartNode(strErrors)) wxMessageBox(strErrors, "Bitcoin"); @@ -3517,7 +3565,6 @@ void SetStartOnSystemStartup(bool fAutoStart) // Get the current executable path char pszExePath[MAX_PATH]; GetModuleFileName(NULL, pszExePath, sizeof(pszExePath)); - _strlwr(pszExePath); // Set the path to the shortcut target psl->SetPath(pszExePath); diff --git a/ui.h b/ui.h index f89456bf..f451d935 100644 --- a/ui.h +++ b/ui.h @@ -88,6 +88,7 @@ public: void OnCrossThreadCall(wxCommandEvent& event); void InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5); + bool DeleteLine(uint256 hashKey); void InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); void RefreshListCtrl(); void RefreshStatus(); From dd519206a684c772a4a06ceecc87c665ad09d8be Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 29 Oct 2009 02:52:48 +0000 Subject: [PATCH 012/133] addr relaying fixes, proxy option and privacy patches, detect connect to self, non-final tx locktime changes, fix hide unconfirmed generated git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@18 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build.txt | 12 +- db.cpp | 19 +++- headers.h | 2 + irc.cpp | 10 +- key.h | 2 +- main.cpp | 71 +++++++++--- main.h | 16 ++- makefile | 4 +- net.cpp | 173 ++++++++++++++++------------ net.h | 131 ++++++++++++++++++++- serialize.h | 2 +- strlcpy.h | 84 ++++++++++++++ ui.cpp | 223 ++++++++++++++++++++++++++++-------- ui.h | 8 +- uibase.cpp | 51 ++++++++- uibase.h | 98 ++++++++-------- uiproject.fbp | 308 +++++++++++++++++++++++++++++++++++++++++++++++++- util.cpp | 170 +++++++++++++++++++--------- util.h | 164 ++++++++++++++++----------- 19 files changed, 1209 insertions(+), 339 deletions(-) create mode 100644 strlcpy.h diff --git a/build.txt b/build.txt index b6b526f1..f5100557 100644 --- a/build.txt +++ b/build.txt @@ -1,4 +1,4 @@ -BitCoin v0.1.6 ALPHA +BitCoin v0.1.6 BETA Copyright (c) 2009 Satoshi Nakamoto Distributed under the MIT/X11 software license, see the accompanying @@ -19,10 +19,10 @@ Dependencies Libraries you need to obtain separately to build: default path download -wxWidgets \wxWidgets http://www.wxwidgets.org/downloads/ -OpenSSL \OpenSSL http://www.openssl.org/source/ -Berkeley DB \DB http://www.oracle.com/technology/software/products/berkeley-db/index.html -Boost \Boost http://www.boost.org/users/download/ +wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ +OpenSSL \openssl http://www.openssl.org/source/ +Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html +Boost \boost http://www.boost.org/users/download/ Their licenses: wxWidgets LGPL 2.1 with very liberal exceptions @@ -75,7 +75,7 @@ If you want to use it with MSVC, generate the .lib file Berkeley DB ----------- Using MinGW and MSYS: -cd \DB\build_unix +cd \db\build_unix sh ../dist/configure --enable-mingw --enable-cxx make diff --git a/db.cpp b/db.cpp index a9f5b795..315e93b7 100644 --- a/db.cpp +++ b/db.cpp @@ -121,10 +121,12 @@ void CDB::Close() pdb->close(0); delete pdb; pdb = NULL; - dbenv.txn_checkpoint(0, 0, 0); CRITICAL_BLOCK(cs_db) + { + dbenv.txn_checkpoint(0, 0, 0); --mapFileUseCount[strFile]; + } RandAddSeed(); } @@ -376,11 +378,11 @@ bool CTxDB::LoadBlockIndex() { if (pindexGenesisBlock == NULL) return true; - return error("CTxDB::LoadBlockIndex() : hashBestChain not found\n"); + return error("CTxDB::LoadBlockIndex() : hashBestChain not found"); } if (!mapBlockIndex.count(hashBestChain)) - return error("CTxDB::LoadBlockIndex() : blockindex for hashBestChain not found\n"); + return error("CTxDB::LoadBlockIndex() : blockindex for hashBestChain not found"); pindexBest = mapBlockIndex[hashBestChain]; nBestHeight = pindexBest->nHeight; printf("LoadBlockIndex(): hashBestChain=%s height=%d\n", hashBestChain.ToString().substr(0,14).c_str(), nBestHeight); @@ -500,16 +502,15 @@ bool CReviewDB::WriteReviews(uint256 hash, const vector& vReviews) CWalletDB::~CWalletDB() { // Flush whenever all handles to wallet.dat are closed - Close(); CRITICAL_BLOCK(cs_db) { + Close(); // close includes a txn_checkpoint map::iterator mi = mapFileUseCount.find(strFile); if (mi != mapFileUseCount.end()) { int nRefCount = (*mi).second; if (nRefCount == 0) { - dbenv.txn_checkpoint(0, 0, 0); dbenv.lsn_reset(strFile.c_str(), 0); mapFileUseCount.erase(mi++); } @@ -600,6 +601,9 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) if (strKey == "nLimitProcessors") ssValue >> nLimitProcessors; if (strKey == "fMinimizeToTray") ssValue >> fMinimizeToTray; if (strKey == "fMinimizeOnClose") ssValue >> fMinimizeOnClose; + if (strKey == "fUseProxy") ssValue >> fUseProxy; + if (strKey == "addrProxy") ssValue >> addrProxy; + } } } @@ -610,6 +614,9 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) printf("addrIncoming = %s\n", addrIncoming.ToString().c_str()); printf("fMinimizeToTray = %d\n", fMinimizeToTray); printf("fMinimizeOnClose = %d\n", fMinimizeOnClose); + printf("fUseProxy = %d\n", fUseProxy); + printf("addrProxy = %s\n", addrProxy.ToString().c_str()); + // The transaction fee setting won't be needed for many years to come. // Setting it to zero here in case they set it to something in an earlier version. @@ -639,7 +646,7 @@ bool LoadWallet(bool& fFirstRunRet) else { // Create new keyUser and set as default key - RandAddSeed(true); + RandAddSeedPerfmon(); keyUser.MakeNewKey(); if (!AddKey(keyUser)) return false; diff --git a/headers.h b/headers.h index f7e88e0c..29b16fb7 100644 --- a/headers.h +++ b/headers.h @@ -5,6 +5,7 @@ #ifdef _MSC_VER #pragma warning(disable:4786) #pragma warning(disable:4804) +#pragma warning(disable:4805) #pragma warning(disable:4717) #endif #ifdef _WIN32_WINNT @@ -62,6 +63,7 @@ using namespace boost; +#include "strlcpy.h" #include "serialize.h" #include "uint256.h" #include "util.h" diff --git a/irc.cpp b/irc.cpp index cfc9464a..707b4fe1 100644 --- a/irc.cpp +++ b/irc.cpp @@ -163,6 +163,9 @@ void ThreadIRCSeed(void* parg) int nErrorWait = 10; int nRetryWait = 10; + if (fUseProxy && addrProxy.port == htons(9050)) + return; + while (!fShutdown) { CAddress addrConnect("216.155.130.130:6667"); @@ -191,9 +194,10 @@ void ThreadIRCSeed(void* parg) return; } - string strMyName = EncodeAddress(addrLocalHost); - - if (!addrLocalHost.IsRoutable()) + string strMyName; + if (addrLocalHost.IsRoutable() && !fUseProxy) + strMyName = EncodeAddress(addrLocalHost); + else strMyName = strprintf("x%u", GetRand(1000000000)); diff --git a/key.h b/key.h index 39ca86d3..71fce64a 100644 --- a/key.h +++ b/key.h @@ -35,7 +35,7 @@ public: }; -// secure_allocator is defined is serialize.h +// secure_allocator is defined in serialize.h typedef vector > CPrivKey; diff --git a/main.cpp b/main.cpp index dbdee702..710b7892 100644 --- a/main.cpp +++ b/main.cpp @@ -415,6 +415,10 @@ bool CTransaction::AcceptTransaction(CTxDB& txdb, bool fCheckInputs, bool* pfMis if (!CheckTransaction()) return error("AcceptTransaction() : CheckTransaction failed"); + // To help v0.1.5 clients who would see it as negative number. please delete this later. + if (nLockTime > INT_MAX) + return error("AcceptTransaction() : not accepting nLockTime beyond 2038"); + // Do we already have it? uint256 hash = GetHash(); CRITICAL_BLOCK(cs_mapTransactions) @@ -1214,6 +1218,12 @@ bool CBlock::AcceptBlock() if (nTime <= pindexPrev->GetMedianTimePast()) return error("AcceptBlock() : block's timestamp is too early"); + // Check that all transactions are finalized (starting around 30 Nov 2009) + if (nBestHeight > 31000) // 25620 + 5320 + foreach(const CTransaction& tx, vtx) + if (!tx.IsFinal(nTime)) + return error("AcceptBlock() : contains a non-final transaction"); + // Check proof of work if (nBits != GetNextWorkRequired(pindexPrev)) return error("AcceptBlock() : incorrect proof of work"); @@ -1649,7 +1659,7 @@ bool ProcessMessages(CNode* pfrom) CDataStream& vRecv = pfrom->vRecv; if (vRecv.empty()) return true; - printf("ProcessMessages(%d bytes)\n", vRecv.size()); + //printf("ProcessMessages(%d bytes)\n", vRecv.size()); // // Message format @@ -1692,7 +1702,7 @@ bool ProcessMessages(CNode* pfrom) { // Rewind and wait for rest of message ///// need a mechanism to give up waiting for overlong message size error - printf("MESSAGE-BREAK\n"); + //printf("message-break\n"); vRecv.insert(vRecv.begin(), BEGIN(hdr), END(hdr)); Sleep(100); break; @@ -1711,7 +1721,20 @@ bool ProcessMessages(CNode* pfrom) fRet = ProcessMessage(pfrom, strCommand, vMsg); CheckForShutdown(2); } - CATCH_PRINT_EXCEPTION("ProcessMessage()") + catch (std::ios_base::failure& e) { + if (strstr(e.what(), "CDataStream::read() : end of data")) + { + // Allow exceptions from underlength message on vRecv + LogException(&e, "ProcessMessage()"); + } + else + PrintException(&e, "ProcessMessage()"); + } catch (std::exception& e) { + PrintException(&e, "ProcessMessage()"); + } catch (...) { + PrintException(NULL, "ProcessMessage()"); + } + if (!fRet) printf("ProcessMessage(%s, %d bytes) FAILED\n", strCommand.c_str(), nMessageSize); } @@ -1726,7 +1749,8 @@ bool ProcessMessages(CNode* pfrom) bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) { static map > mapReuseKey; - printf("received: %-12s (%d bytes)\n", strCommand.c_str(), vRecv.size()); + RandAddSeedPerfmon(); + printf("received: %s (%d bytes)\n", strCommand.c_str(), vRecv.size()); if (nDropMessagesTest > 0 && GetRand(nDropMessagesTest) == 0) { printf("dropmessages DROPPING RECV MESSAGE\n"); @@ -1735,18 +1759,32 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) + if (strCommand == "version") { - // Can only do this once + // Each connection can only send one version message if (pfrom->nVersion != 0) return false; int64 nTime; CAddress addrMe; + CAddress addrFrom; + uint64 nNonce = 1; vRecv >> pfrom->nVersion >> pfrom->nServices >> nTime >> addrMe; + if (pfrom->nVersion >= 106 && !vRecv.empty()) + vRecv >> addrFrom >> nNonce; if (pfrom->nVersion == 0) return false; + // Disconnect if we connected to ourself + if (nNonce == nLocalHostNonce) + { + pfrom->fDisconnect = true; + pfrom->vRecv.clear(); + pfrom->vSend.clear(); + return true; + } + pfrom->vSend.SetVersion(min(pfrom->nVersion, VERSION)); pfrom->vRecv.SetVersion(min(pfrom->nVersion, VERSION)); @@ -1767,6 +1805,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), uint256(0)); } + pfrom->fSuccessfullyConnected = true; + printf("version message: version %d\n", pfrom->nVersion); } @@ -1800,16 +1840,16 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) if (fShutdown) return true; AddAddress(addrdb, addr); - if (addr.IsRoutable() && addr.ip != addrLocalHost.ip) + pfrom->AddAddressKnown(addr); + if (!pfrom->fGetAddr && addr.IsRoutable()) { // Put on lists to send to other nodes - pfrom->setAddrKnown.insert(addr); CRITICAL_BLOCK(cs_vNodes) foreach(CNode* pnode, vNodes) - if (!pnode->setAddrKnown.count(addr)) - pnode->vAddrToSend.push_back(addr); + pnode->PushAddress(addr); } } + pfrom->fGetAddr = false; } @@ -2009,7 +2049,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) return true; const CAddress& addr = item.second; if (addr.nTime > nSince) - pfrom->vAddrToSend.push_back(addr); + pfrom->PushAddress(addr); } } } @@ -2108,8 +2148,11 @@ bool SendMessages(CNode* pto) vector vAddrToSend; vAddrToSend.reserve(pto->vAddrToSend.size()); foreach(const CAddress& addr, pto->vAddrToSend) - if (!pto->setAddrKnown.count(addr)) + { + // returns true if wasn't already contained in the set + if (pto->setAddrKnown.insert(addr).second) vAddrToSend.push_back(addr); + } pto->vAddrToSend.clear(); if (!vAddrToSend.empty()) pto->PushMessage("addr", vAddrToSend); @@ -2193,7 +2236,7 @@ void GenerateBitcoins(bool fGenerate) if (fLimitProcessors && nProcessors > nLimitProcessors) nProcessors = nLimitProcessors; int nAddThreads = nProcessors - vnThreadsRunning[3]; - printf("starting %d bitcoinminer threads\n", nAddThreads); + printf("Starting %d BitcoinMiner threads\n", nAddThreads); for (int i = 0; i < nAddThreads; i++) if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); @@ -2207,7 +2250,7 @@ void ThreadBitcoinMiner(void* parg) try { bool fRet = BitcoinMiner(); - printf("BitcoinMiner returned %s\n\n\n", fRet ? "true" : "false"); + printf("BitcoinMiner returned %s\n", fRet ? "true" : "false"); vnThreadsRunning[3]--; } catch (std::exception& e) { @@ -2737,7 +2780,7 @@ bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) else strError = "Error: Transaction creation failed "; wxMessageBox(strError, "Sending..."); - return error("SendMoney() : %s\n", strError.c_str()); + return error("SendMoney() : %s", strError.c_str()); } if (!CommitTransactionSpent(wtxNew, key)) { diff --git a/main.h b/main.h index 6b11285e..958f7a5f 100644 --- a/main.h +++ b/main.h @@ -366,7 +366,7 @@ public: int nVersion; vector vin; vector vout; - int nLockTime; + unsigned int nLockTime; CTransaction() @@ -401,9 +401,15 @@ public: return SerializeHash(*this); } - bool IsFinal() const + bool IsFinal(int64 nBlockTime=0) const { - if (nLockTime == 0 || nLockTime < nBestHeight) + // Time based nLockTime implemented in 0.1.6, + // do not use time based until most 0.1.5 nodes have upgraded. + if (nBlockTime == 0) + nBlockTime = GetAdjustedTime(); + if (nLockTime == 0) + return true; + if (nLockTime < (nLockTime < 500000000 ? nBestHeight : nBlockTime)) return true; foreach(const CTxIn& txin, vin) if (!txin.IsFinal()) @@ -686,8 +692,9 @@ public: char fSpent; //// probably need to sign the order info so know it came from payer - // memory only + // memory only UI hints mutable unsigned int nTimeDisplayed; + mutable int nLinesDisplayed; CWalletTx() @@ -712,6 +719,7 @@ public: fFromMe = false; fSpent = false; nTimeDisplayed = 0; + nLinesDisplayed = 0; } IMPLEMENT_SERIALIZE diff --git a/makefile b/makefile index 64bb7730..2d932f02 100644 --- a/makefile +++ b/makefile @@ -17,8 +17,8 @@ endif -INCLUDEPATHS=-I"/boost" -I"/DB/build_unix" -I"/OpenSSL/include" -I"/wxWidgets/lib/vc_lib/mswd" -I"/wxWidgets/include" -LIBPATHS=-L"/DB/build_unix" -L"/OpenSSL/out" -L"/wxWidgets/lib/gcc_lib" +INCLUDEPATHS=-I"/boost" -I"/db/build_unix" -I"/openssl/include" -I"/wxwidgets/lib/vc_lib/mswd" -I"/wxwidgets/include" +LIBPATHS=-L"/db/build_unix" -L"/openssl/out" -L"/wxwidgets/lib/gcc_lib" LIBS= \ -l db_cxx \ -l eay32 \ diff --git a/net.cpp b/net.cpp index db138e13..22b84f9d 100644 --- a/net.cpp +++ b/net.cpp @@ -8,6 +8,7 @@ void ThreadMessageHandler2(void* parg); void ThreadSocketHandler2(void* parg); void ThreadOpenConnections2(void* parg); +bool OpenNetworkConnection(const CAddress& addrConnect); @@ -22,8 +23,10 @@ uint64 nLocalServices = (fClient ? 0 : NODE_NETWORK); CAddress addrLocalHost(0, DEFAULT_PORT, nLocalServices); CNode nodeLocalHost(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); CNode* pnodeLocalHost = &nodeLocalHost; +uint64 nLocalHostNonce = 0; bool fShutdown = false; array vnThreadsRunning; +SOCKET hListenSocket = INVALID_SOCKET; vector vNodes; CCriticalSection cs_vNodes; @@ -34,9 +37,11 @@ deque > vRelayExpiration; CCriticalSection cs_mapRelay; map mapAlreadyAskedFor; +// Settings +int fUseProxy = false; +CAddress addrProxy("127.0.0.1:9050"); -CAddress addrProxy; bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) { @@ -47,7 +52,7 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) return false; bool fRoutable = !(addrConnect.GetByte(3) == 10 || (addrConnect.GetByte(3) == 192 && addrConnect.GetByte(2) == 168)); - bool fProxy = (addrProxy.ip && fRoutable); + bool fProxy = (fUseProxy && fRoutable); struct sockaddr_in sockaddr = (fProxy ? addrProxy.GetSockAddr() : addrConnect.GetSockAddr()); if (connect(hSocket, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) == SOCKET_ERROR) @@ -69,18 +74,18 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) if (ret != nSize) { closesocket(hSocket); - return error("Error sending to proxy\n"); + return error("Error sending to proxy"); } char pchRet[8]; if (recv(hSocket, pchRet, 8, 0) != 8) { closesocket(hSocket); - return error("Error reading proxy response\n"); + return error("Error reading proxy response"); } if (pchRet[1] != 0x5a) { closesocket(hSocket); - return error("Proxy returned error %d\n", pchRet[1]); + return error("Proxy returned error %d", pchRet[1]); } printf("Proxy connection established %s\n", addrConnect.ToStringLog().c_str()); } @@ -95,7 +100,7 @@ bool GetMyExternalIP2(const CAddress& addrConnect, const char* pszGet, const cha { SOCKET hSocket; if (!ConnectSocket(addrConnect, hSocket)) - return error("GetMyExternalIP() : connection to %s failed\n", addrConnect.ToString().c_str()); + return error("GetMyExternalIP() : connection to %s failed", addrConnect.ToString().c_str()); send(hSocket, pszGet, strlen(pszGet), 0); @@ -131,7 +136,7 @@ bool GetMyExternalIP2(const CAddress& addrConnect, const char* pszGet, const cha } } closesocket(hSocket); - return error("GetMyExternalIP() : connection closed\n"); + return error("GetMyExternalIP() : connection closed"); } @@ -141,6 +146,9 @@ bool GetMyExternalIP(unsigned int& ipRet) char* pszGet; char* pszKeyword; + if (fUseProxy) + return false; + for (int nLookup = 0; nLookup <= 1; nLookup++) for (int nHost = 1; nHost <= 2; nHost++) { @@ -416,14 +424,14 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) } } -void CNode::Disconnect() +void CNode::DoDisconnect() { printf("disconnecting node %s\n", addr.ToStringLog().c_str()); closesocket(hSocket); // If outbound and never got version message, mark address as failed - if (!fInbound && nVersion == 0) + if (!fInbound && !fSuccessfullyConnected) CRITICAL_BLOCK(cs_mapAddresses) mapAddresses[addr.GetKey()].nLastFailed = GetTime(); @@ -458,18 +466,18 @@ void ThreadSocketHandler(void* parg) loop { - vnThreadsRunning[0] = true; + vnThreadsRunning[0]++; CheckForShutdown(0); try { ThreadSocketHandler2(parg); - vnThreadsRunning[0] = false; + vnThreadsRunning[0]--; } catch (std::exception& e) { - vnThreadsRunning[0] = false; + vnThreadsRunning[0]--; PrintException(&e, "ThreadSocketHandler()"); } catch (...) { - vnThreadsRunning[0] = false; + vnThreadsRunning[0]--; PrintException(NULL, "ThreadSocketHandler()"); } Sleep(5000); @@ -479,7 +487,6 @@ void ThreadSocketHandler(void* parg) void ThreadSocketHandler2(void* parg) { printf("ThreadSocketHandler started\n"); - SOCKET hListenSocket = *(SOCKET*)parg; list vNodesDisconnected; int nPrevNodeCount = 0; @@ -498,7 +505,7 @@ void ThreadSocketHandler2(void* parg) { // remove from vNodes vNodes.erase(remove(vNodes.begin(), vNodes.end(), pnode), vNodes.end()); - pnode->Disconnect(); + pnode->DoDisconnect(); // hold in disconnected pool until all refs are released pnode->nReleaseTime = max(pnode->nReleaseTime, GetTime() + 5 * 60); @@ -562,9 +569,9 @@ void ThreadSocketHandler2(void* parg) } } - vnThreadsRunning[0] = false; + vnThreadsRunning[0]--; int nSelect = select(hSocketMax + 1, &fdsetRecv, &fdsetSend, NULL, &timeout); - vnThreadsRunning[0] = true; + vnThreadsRunning[0]++; CheckForShutdown(0); if (nSelect == SOCKET_ERROR) { @@ -577,7 +584,6 @@ void ThreadSocketHandler2(void* parg) } Sleep(timeout.tv_usec/1000); } - RandAddSeed(); //// debug print //foreach(CNode* pnode, vNodes) @@ -711,18 +717,18 @@ void ThreadOpenConnections(void* parg) loop { - vnThreadsRunning[1] = true; + vnThreadsRunning[1]++; CheckForShutdown(1); try { ThreadOpenConnections2(parg); - vnThreadsRunning[1] = false; + vnThreadsRunning[1]--; } catch (std::exception& e) { - vnThreadsRunning[1] = false; + vnThreadsRunning[1]--; PrintException(&e, "ThreadOpenConnections()"); } catch (...) { - vnThreadsRunning[1] = false; + vnThreadsRunning[1]--; PrintException(NULL, "ThreadOpenConnections()"); } Sleep(5000); @@ -733,6 +739,13 @@ void ThreadOpenConnections2(void* parg) { printf("ThreadOpenConnections started\n"); + // Connect to one specified address + while (mapArgs.count("/connect")) + { + OpenNetworkConnection(CAddress(mapArgs["/connect"].c_str())); + Sleep(10000); + } + // Initiate network connections int nTry = 0; bool fIRCOnly = false; @@ -740,14 +753,14 @@ void ThreadOpenConnections2(void* parg) loop { // Wait - vnThreadsRunning[1] = false; + vnThreadsRunning[1]--; Sleep(500); while (vNodes.size() >= nMaxConnections || vNodes.size() >= mapAddresses.size()) { CheckForShutdown(1); Sleep(2000); } - vnThreadsRunning[1] = true; + vnThreadsRunning[1]++; CheckForShutdown(1); @@ -835,43 +848,48 @@ void ThreadOpenConnections2(void* parg) // Once we've chosen an IP, we'll try every given port before moving on foreach(const CAddress& addrConnect, (*mi).second) - { - // - // Initiate outbound network connection - // - CheckForShutdown(1); - if (addrConnect.ip == addrLocalHost.ip || !addrConnect.IsIPv4() || FindNode(addrConnect.ip)) - continue; + if (OpenNetworkConnection(addrConnect)) + break; + } +} - vnThreadsRunning[1] = false; - CNode* pnode = ConnectNode(addrConnect); - vnThreadsRunning[1] = true; - CheckForShutdown(1); - if (!pnode) - continue; - pnode->fNetworkNode = true; +bool OpenNetworkConnection(const CAddress& addrConnect) +{ + // + // Initiate outbound network connection + // + CheckForShutdown(1); + if (addrConnect.ip == addrLocalHost.ip || !addrConnect.IsIPv4() || FindNode(addrConnect.ip)) + return false; - if (addrLocalHost.IsRoutable()) - { - // Advertise our address - vector vAddrToSend; - vAddrToSend.push_back(addrLocalHost); - pnode->PushMessage("addr", vAddrToSend); - } + vnThreadsRunning[1]--; + CNode* pnode = ConnectNode(addrConnect); + vnThreadsRunning[1]++; + CheckForShutdown(1); + if (!pnode) + return false; + pnode->fNetworkNode = true; - // Get as many addresses as we can - pnode->PushMessage("getaddr"); + if (addrLocalHost.IsRoutable() && !fUseProxy) + { + // Advertise our address + vector vAddrToSend; + vAddrToSend.push_back(addrLocalHost); + pnode->PushMessage("addr", vAddrToSend); + } - ////// should the one on the receiving end do this too? - // Subscribe our local subscription list - const unsigned int nHops = 0; - for (unsigned int nChannel = 0; nChannel < pnodeLocalHost->vfSubscribe.size(); nChannel++) - if (pnodeLocalHost->vfSubscribe[nChannel]) - pnode->PushMessage("subscribe", nChannel, nHops); + // Get as many addresses as we can + pnode->PushMessage("getaddr"); + pnode->fGetAddr = true; // don't relay the results of the getaddr - break; - } - } + ////// should the one on the receiving end do this too? + // Subscribe our local subscription list + const unsigned int nHops = 0; + for (unsigned int nChannel = 0; nChannel < pnodeLocalHost->vfSubscribe.size(); nChannel++) + if (pnodeLocalHost->vfSubscribe[nChannel]) + pnode->PushMessage("subscribe", nChannel, nHops); + + return true; } @@ -887,18 +905,18 @@ void ThreadMessageHandler(void* parg) loop { - vnThreadsRunning[2] = true; + vnThreadsRunning[2]++; CheckForShutdown(2); try { ThreadMessageHandler2(parg); - vnThreadsRunning[2] = false; + vnThreadsRunning[2]--; } catch (std::exception& e) { - vnThreadsRunning[2] = false; + vnThreadsRunning[2]--; PrintException(&e, "ThreadMessageHandler()"); } catch (...) { - vnThreadsRunning[2] = false; + vnThreadsRunning[2]--; PrintException(NULL, "ThreadMessageHandler()"); } Sleep(5000); @@ -931,9 +949,9 @@ void ThreadMessageHandler2(void* parg) } // Wait and allow messages to bunch up - vnThreadsRunning[2] = false; + vnThreadsRunning[2]--; Sleep(100); - vnThreadsRunning[2] = true; + vnThreadsRunning[2]++; CheckForShutdown(2); } } @@ -982,7 +1000,7 @@ bool StartNode(string& strError) printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); // Create socket for listening for incoming connections - SOCKET hListenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); + hListenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (hListenSocket == INVALID_SOCKET) { strError = strprintf("Error: Couldn't open socket for incoming connections (socket returned error %d)", WSAGetLastError()); @@ -1024,13 +1042,21 @@ bool StartNode(string& strError) } // Get our external IP address for incoming connections - if (addrIncoming.ip) - addrLocalHost.ip = addrIncoming.ip; - - if (GetMyExternalIP(addrLocalHost.ip)) + if (fUseProxy) { - addrIncoming = addrLocalHost; - CWalletDB().WriteSetting("addrIncoming", addrIncoming); + // Proxies can't take incoming connections + addrLocalHost.ip = CAddress("0.0.0.0").ip; + } + else + { + if (addrIncoming.ip) + addrLocalHost.ip = addrIncoming.ip; + + if (GetMyExternalIP(addrLocalHost.ip)) + { + addrIncoming = addrLocalHost; + CWalletDB().WriteSetting("addrIncoming", addrIncoming); + } } // Get addresses from IRC and advertise ours @@ -1040,7 +1066,7 @@ bool StartNode(string& strError) // // Start threads // - if (_beginthread(ThreadSocketHandler, 0, new SOCKET(hListenSocket)) == -1) + if (_beginthread(ThreadSocketHandler, 0, NULL) == -1) { strError = "Error: _beginthread(ThreadSocketHandler) failed"; printf("%s\n", strError.c_str()); @@ -1094,10 +1120,15 @@ void CheckForShutdown(int n) if (fShutdown) { if (n != -1) - vnThreadsRunning[n] = false; + if (--vnThreadsRunning[n] < 0) + vnThreadsRunning[n] = 0; if (n == 0) + { foreach(CNode* pnode, vNodes) closesocket(pnode->hSocket); + closesocket(hListenSocket); + } + printf("Thread %d exiting\n", n); _endthread(); } } diff --git a/net.h b/net.h index 33410042..a0b2929d 100644 --- a/net.h +++ b/net.h @@ -174,7 +174,7 @@ public: { nServices = nServicesIn; memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); - ip = 0; + ip = INADDR_NONE; port = DEFAULT_PORT; nTime = GetAdjustedTime(); nLastFailed = 0; @@ -183,7 +183,7 @@ public: if (strlen(pszIn) > ARRAYLEN(psz)-1) return; strcpy(psz, pszIn); - unsigned int a, b, c, d, e; + unsigned int a=0, b=0, c=0, d=0, e=0; if (sscanf(psz, "%u.%u.%u.%u:%u", &a, &b, &c, &d, &e) < 4) return; char* pszPort = strchr(psz, ':'); @@ -191,6 +191,10 @@ public: { *pszPort++ = '\0'; port = htons(atoi(pszPort)); + if (atoi(pszPort) > USHRT_MAX) + port = htons(USHRT_MAX); + if (atoi(pszPort) < 0) + port = htons(0); } ip = inet_addr(psz); } @@ -215,6 +219,11 @@ public: a.port == b.port); } + friend inline bool operator!=(const CAddress& a, const CAddress& b) + { + return (!(a == b)); + } + friend inline bool operator<(const CAddress& a, const CAddress& b) { int ret = memcmp(a.pchReserved, b.pchReserved, sizeof(a.pchReserved)); @@ -277,6 +286,11 @@ public: return strprintf("%u.%u.%u.%u", GetByte(3), GetByte(2), GetByte(1), GetByte(0)); } + string ToStringPort() const + { + return strprintf("%u", ntohs(port)); + } + string ToStringLog() const { return ""; @@ -416,6 +430,7 @@ extern bool fClient; extern uint64 nLocalServices; extern CAddress addrLocalHost; extern CNode* pnodeLocalHost; +extern uint64 nLocalHostNonce; extern bool fShutdown; extern array vnThreadsRunning; extern vector vNodes; @@ -426,6 +441,9 @@ extern map mapRelay; extern deque > vRelayExpiration; extern CCriticalSection cs_mapRelay; extern map mapAlreadyAskedFor; + +// Settings +extern int fUseProxy; extern CAddress addrProxy; @@ -448,6 +466,7 @@ public: bool fClient; bool fInbound; bool fNetworkNode; + bool fSuccessfullyConnected; bool fDisconnect; protected: int nRefCount; @@ -459,6 +478,7 @@ public: // flood vector vAddrToSend; set setAddrKnown; + bool fGetAddr; // inventory based relay set setInventoryKnown; @@ -483,15 +503,20 @@ public: fClient = false; // set by version message fInbound = fInboundIn; fNetworkNode = false; + fSuccessfullyConnected = false; fDisconnect = false; nRefCount = 0; nReleaseTime = 0; + fGetAddr = false; vfSubscribe.assign(256, false); // Push a version message /// when NTP implemented, change to just nTime = GetAdjustedTime() int64 nTime = (fInbound ? GetAdjustedTime() : GetTime()); - PushMessage("version", VERSION, nLocalServices, nTime, addr); + CAddress addrYou = (fUseProxy ? CAddress("0.0.0.0") : addr); + CAddress addrMe = (fUseProxy ? CAddress("0.0.0.0") : addrLocalHost); + RAND_bytes((unsigned char*)&nLocalHostNonce, sizeof(nLocalHostNonce)); + PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce); } ~CNode() @@ -531,6 +556,21 @@ public: + void AddAddressKnown(const CAddress& addr) + { + setAddrKnown.insert(addr); + } + + void PushAddress(const CAddress& addr) + { + // Known checking here is only to save space from duplicates. + // SendMessages will filter it again for knowns that were added + // after addresses were pushed. + if (!setAddrKnown.count(addr)) + vAddrToSend.push_back(addr); + } + + void AddInventoryKnown(const CInv& inv) { CRITICAL_BLOCK(cs_inventory) @@ -562,7 +602,6 @@ public: } - void BeginMessage(const char* pszCommand) { EnterCriticalSection(&cs_vSend); @@ -570,7 +609,7 @@ public: AbortMessage(); nPushPos = vSend.size(); vSend << CMessageHeader(pszCommand, 0); - printf("sending: %-12s ", pszCommand); + printf("sending: %s ", pszCommand); } void AbortMessage() @@ -706,6 +745,86 @@ public: } } + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2, const T3& a3, const T4& a4, const T5& a5) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2 << a3 << a4 << a5; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2, const T3& a3, const T4& a4, const T5& a5, const T6& a6) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2 << a3 << a4 << a5 << a6; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2, const T3& a3, const T4& a4, const T5& a5, const T6& a6, const T7& a7) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2 << a3 << a4 << a5 << a6 << a7; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2, const T3& a3, const T4& a4, const T5& a5, const T6& a6, const T7& a7, const T8& a8) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2 << a3 << a4 << a5 << a6 << a7 << a8; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + + template + void PushMessage(const char* pszCommand, const T1& a1, const T2& a2, const T3& a3, const T4& a4, const T5& a5, const T6& a6, const T7& a7, const T8& a8, const T9& a9) + { + try + { + BeginMessage(pszCommand); + vSend << a1 << a2 << a3 << a4 << a5 << a6 << a7 << a8 << a9; + EndMessage(); + } + catch (...) + { + AbortMessage(); + throw; + } + } + void PushRequest(const char* pszCommand, void (*fn)(void*, CDataStream&), void* param1) @@ -750,7 +869,7 @@ public: bool IsSubscribed(unsigned int nChannel); void Subscribe(unsigned int nChannel, unsigned int nHops=0); void CancelSubscribe(unsigned int nChannel); - void Disconnect(); + void DoDisconnect(); }; diff --git a/serialize.h b/serialize.h index b7ab86d2..9b20e2a0 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 105; +static const int VERSION = 106; diff --git a/strlcpy.h b/strlcpy.h new file mode 100644 index 00000000..a79ee175 --- /dev/null +++ b/strlcpy.h @@ -0,0 +1,84 @@ +/* + * Copyright (c) 1998 Todd C. Miller + * + * Permission to use, copy, modify, and distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +/* + * Copy src to string dst of size siz. At most siz-1 characters + * will be copied. Always NUL terminates (unless siz == 0). + * Returns strlen(src); if retval >= siz, truncation occurred. + */ +inline size_t strlcpy(char *dst, const char *src, size_t siz) +{ + char *d = dst; + const char *s = src; + size_t n = siz; + + /* Copy as many bytes as will fit */ + if (n != 0) + { + while (--n != 0) + { + if ((*d++ = *s++) == '\0') + break; + } + } + + /* Not enough room in dst, add NUL and traverse rest of src */ + if (n == 0) + { + if (siz != 0) + *d = '\0'; /* NUL-terminate dst */ + while (*s++) + ; + } + + return(s - src - 1); /* count does not include NUL */ +} + +/* + * Appends src to string dst of size siz (unlike strncat, siz is the + * full size of dst, not space left). At most siz-1 characters + * will be copied. Always NUL terminates (unless siz <= strlen(dst)). + * Returns strlen(src) + MIN(siz, strlen(initial dst)). + * If retval >= siz, truncation occurred. + */ +inline size_t strlcat(char *dst, const char *src, size_t siz) +{ + char *d = dst; + const char *s = src; + size_t n = siz; + size_t dlen; + + /* Find the end of dst and adjust bytes left but don't go past end */ + while (n-- != 0 && *d != '\0') + d++; + dlen = d - dst; + n = siz - dlen; + + if (n == 0) + return(dlen + strlen(s)); + while (*s != '\0') + { + if (n != 1) + { + *d++ = *s; + n--; + } + s++; + } + *d = '\0'; + + return(dlen + (s - src)); /* count does not include NUL */ +} diff --git a/ui.cpp b/ui.cpp index 42830479..917c64b4 100644 --- a/ui.cpp +++ b/ui.cpp @@ -497,7 +497,7 @@ string SingleLine(const string& strIn) return strOut; } -void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) +bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) { int64 nTime = wtx.nTimeDisplayed = wtx.GetTxTime(); int64 nCredit = wtx.GetCredit(); @@ -506,14 +506,11 @@ void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) uint256 hash = wtx.GetHash(); string strStatus = FormatTxStatus(wtx); map mapValue = wtx.mapValue; + wtx.nLinesDisplayed = 1; // Filter if (wtx.IsCoinBase()) { - // View->Show Generated - if (!fShowGenerated) - return; - // Don't show generated coin until confirmed by at least one block after it // so we don't get the user's hopes up until it looks like it's probably accepted. // @@ -527,10 +524,13 @@ void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) // if (wtx.GetDepthInMainChain() < 2) { - // In case it was previously displayed - DeleteLine(hash); - return; + wtx.nLinesDisplayed = 0; + return false; } + + // View->Show Generated + if (!fShowGenerated) + return false; } // Find the block the tx is in @@ -644,6 +644,7 @@ void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) // Debit // int64 nTxFee = nDebit - wtx.GetValueOut(); + wtx.nLinesDisplayed = 0; for (int nOut = 0; nOut < wtx.vout.size(); nOut++) { const CTxOut& txout = wtx.vout[nOut]; @@ -685,6 +686,7 @@ void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) SingleLine(strDescription), FormatMoney(-nValue, true), ""); + wtx.nLinesDisplayed++; } } else @@ -706,12 +708,14 @@ void CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) ""); } } + + return true; } void CMainFrame::RefreshStatus() { static int nLastTop; - int nTop = m_listCtrl->GetTopItem(); + int nTop = max((int)m_listCtrl->GetTopItem(), 0); if (nTop == nLastTop && pindexBestLast == pindexBest) return; @@ -729,7 +733,7 @@ void CMainFrame::RefreshStatus() nLastTop = nTop; pindexBestLast = pindexBest; - for (int nIndex = nStart; nIndex < nEnd; nIndex++) + for (int nIndex = nStart; nIndex < min(nEnd, m_listCtrl->GetItemCount()); nIndex++) { uint256 hash((string)GetItemText(m_listCtrl, nIndex, 1)); map::iterator mi = mapWallet.find(hash); @@ -738,9 +742,12 @@ void CMainFrame::RefreshStatus() printf("CMainFrame::RefreshStatus() : tx not found in mapWallet\n"); continue; } - const CWalletTx& wtx = (*mi).second; + CWalletTx& wtx = (*mi).second; if (wtx.IsCoinBase() || wtx.GetTxTime() != wtx.nTimeDisplayed) - InsertTransaction(wtx, false, nIndex); + { + if (!InsertTransaction(wtx, false, nIndex)) + m_listCtrl->DeleteItem(nIndex--); + } else m_listCtrl->SetItem(nIndex, 2, FormatTxStatus(wtx)); } @@ -801,6 +808,9 @@ void CMainFrame::OnIdle(wxIdleEvent& event) } printf("RefreshListCtrl done\n"); + + // Update transaction total display + MainFrameRepaint(); } else { @@ -834,31 +844,54 @@ void DelayedRepaint(void* parg) return; fOneThread = true; Sleep(1000); + printf("DelayedRepaint()\n"); MainFrameRepaint(); fOneThread = false; } void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) { + if (ptaskbaricon) + ptaskbaricon->UpdateTooltip(); + // Update listctrl contents if (!vWalletUpdated.empty()) { TRY_CRITICAL_BLOCK(cs_mapWallet) { + bool fInserted = false; foreach(uint256 hash, vWalletUpdated) { map::iterator mi = mapWallet.find(hash); if (mi != mapWallet.end()) - InsertTransaction((*mi).second, false); + fInserted |= InsertTransaction((*mi).second, false); } - m_listCtrl->ScrollList(0, INT_MAX); vWalletUpdated.clear(); + if (fInserted) + m_listCtrl->ScrollList(0, INT_MAX); } } // Update status column of visible items only RefreshStatus(); + // Balance total + bool fRefreshed = false; + static int nTransactionCount; + TRY_CRITICAL_BLOCK(cs_mapWallet) + { + fRefreshed = true; + m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); + + // Count hidden and multi-line transactions + nTransactionCount = 0; + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + CWalletTx& wtx = (*it).second; + nTransactionCount += wtx.nLinesDisplayed; + } + } + // Update status bar string strGen = ""; if (fGenerateBitcoins) @@ -867,17 +900,9 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) strGen = "(not connected)"; m_statusBar->SetStatusText(strGen, 1); - string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, m_listCtrl->GetItemCount()); + string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, nTransactionCount); m_statusBar->SetStatusText(strStatus, 2); - // Balance total - bool fRefreshed = false; - TRY_CRITICAL_BLOCK(cs_mapWallet) - { - m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); - fRefreshed = true; - } - // mapWallet was locked, try again later if (!vWalletUpdated.empty() || !fRefreshed) _beginthread(DelayedRepaint, 0, NULL); @@ -1350,6 +1375,14 @@ COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) m_checkBoxMinimizeOnClose->Enable(fMinimizeToTray); m_checkBoxMinimizeOnClose->SetValue(fMinimizeToTray && fMinimizeOnClose); fTmpMinimizeOnClose = fMinimizeOnClose; + m_checkBoxUseProxy->SetValue(fUseProxy); + m_textCtrlProxyIP->Enable(fUseProxy); + m_textCtrlProxyPort->Enable(fUseProxy); + m_staticTextProxyIP->Enable(fUseProxy); + m_staticTextProxyPort->Enable(fUseProxy); + m_textCtrlProxyIP->SetValue(addrProxy.ToStringIP()); + m_textCtrlProxyPort->SetValue(addrProxy.ToStringPort()); + m_buttonOK->SetFocus(); } @@ -1395,6 +1428,34 @@ void COptionsDialog::OnCheckBoxMinimizeToTray(wxCommandEvent& event) } +void COptionsDialog::OnCheckBoxUseProxy(wxCommandEvent& event) +{ + m_textCtrlProxyIP->Enable(event.IsChecked()); + m_textCtrlProxyPort->Enable(event.IsChecked()); + m_staticTextProxyIP->Enable(event.IsChecked()); + m_staticTextProxyPort->Enable(event.IsChecked()); +} + +CAddress COptionsDialog::GetProxyAddr() +{ + // Be careful about byte order, addr.ip and addr.port are big endian + CAddress addr(m_textCtrlProxyIP->GetValue() + ":" + m_textCtrlProxyPort->GetValue()); + if (addr.ip == INADDR_NONE) + addr.ip = addrProxy.ip; + int nPort = atoi(m_textCtrlProxyPort->GetValue()); + addr.port = htons(nPort); + if (nPort <= 0 || nPort > USHRT_MAX) + addr.port = addrProxy.port; + return addr; +} + +void COptionsDialog::OnKillFocusProxy(wxFocusEvent& event) +{ + m_textCtrlProxyIP->SetValue(GetProxyAddr().ToStringIP()); + m_textCtrlProxyPort->SetValue(GetProxyAddr().ToStringPort()); +} + + void COptionsDialog::OnButtonOK(wxCommandEvent& event) { OnButtonApply(event); @@ -1446,6 +1507,18 @@ void COptionsDialog::OnButtonApply(wxCommandEvent& event) fMinimizeOnClose = (fMinimizeToTray ? m_checkBoxMinimizeOnClose->GetValue() : fTmpMinimizeOnClose); walletdb.WriteSetting("fMinimizeOnClose", fMinimizeOnClose); } + + if (fUseProxy != m_checkBoxUseProxy->GetValue()) + { + fUseProxy = m_checkBoxUseProxy->GetValue(); + walletdb.WriteSetting("fUseProxy", fUseProxy); + } + + if (addrProxy != GetProxyAddr()) + { + addrProxy = GetProxyAddr(); + walletdb.WriteSetting("addrProxy", addrProxy); + } } @@ -1657,7 +1730,7 @@ CSendingDialog::CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 n nPrice = nPriceIn; wtx = wtxIn; start = wxDateTime::UNow(); - strStatus = ""; + memset(pszStatus, 0, sizeof(pszStatus)); fCanCancel = true; fAbort = false; fSuccess = false; @@ -1721,10 +1794,10 @@ void CSendingDialog::OnButtonCancel(wxCommandEvent& event) void CSendingDialog::OnPaint(wxPaintEvent& event) { - if (strStatus.size() > 130) - m_textCtrlStatus->SetValue(string("\n") + strStatus); + if (strlen(pszStatus) > 130) + m_textCtrlStatus->SetValue(string("\n") + pszStatus); else - m_textCtrlStatus->SetValue(string("\n\n") + strStatus); + m_textCtrlStatus->SetValue(string("\n\n") + pszStatus); m_staticTextSending->SetFocus(); if (!fCanCancel) m_buttonCancel->Enable(false); @@ -1736,7 +1809,7 @@ void CSendingDialog::OnPaint(wxPaintEvent& event) } if (fAbort && fCanCancel && IsShown()) { - strStatus = "CANCELLED"; + strcpy(pszStatus, "CANCELLED"); m_buttonOK->Enable(true); m_buttonOK->SetFocus(); m_buttonCancel->Enable(false); @@ -1777,7 +1850,8 @@ bool CSendingDialog::Status() } if (fAbort && fCanCancel) { - strStatus = "CANCELLED"; + memset(pszStatus, 0, 10); + strcpy(pszStatus, "CANCELLED"); Repaint(); fWorkDone = true; return false; @@ -1789,7 +1863,12 @@ bool CSendingDialog::Status(const string& str) { if (!Status()) return false; - strStatus = str; + + // This can be read by the UI thread at any time, + // so copy in a way that can be read cleanly at all times. + memset(pszStatus, 0, min(str.size()+1, sizeof(pszStatus))); + strlcpy(pszStatus, str.c_str(), sizeof(pszStatus)); + Repaint(); return true; } @@ -1950,8 +2029,8 @@ void CSendingDialog::OnReply3(CDataStream& vRecv) if (nRet > 0) { Error("The payment was sent, but the recipient was unable to verify it.\n" - "The transaction is recorded and will credit to the recipient if it is valid,\n" - "but without comment information."); + "The transaction is recorded and will credit to the recipient,\n" + "but the comment information will be blank."); return; } } @@ -3092,6 +3171,7 @@ END_EVENT_TABLE() void CMyTaskBarIcon::Show(bool fShow) { + static char pszPrevTip[200]; if (fShow) { string strTooltip = "Bitcoin"; @@ -3099,10 +3179,17 @@ void CMyTaskBarIcon::Show(bool fShow) strTooltip = "Bitcoin - Generating"; if (fGenerateBitcoins && vNodes.empty()) strTooltip = "Bitcoin - (not connected)"; - SetIcon(wxICON(bitcoin), strTooltip); + + // Optimization, only update when changed, using char array to be reentrant + if (strncmp(pszPrevTip, strTooltip.c_str(), sizeof(pszPrevTip)-1) != 0) + { + strlcpy(pszPrevTip, strTooltip.c_str(), sizeof(pszPrevTip)); + SetIcon(wxICON(bitcoin), strTooltip); + } } else { + strlcpy(pszPrevTip, "", sizeof(pszPrevTip)); RemoveIcon(); } } @@ -3301,12 +3388,12 @@ bool CMyApp::OnInit2() if (mapArgs.count("/datadir")) strSetDataDir = mapArgs["/datadir"]; - if (mapArgs.count("/proxy")) - addrProxy = CAddress(mapArgs["/proxy"].c_str()); - if (mapArgs.count("/debug")) fDebug = true; + if (mapArgs.count("/printtodebugger")) + fPrintToDebugger = true; + if (mapArgs.count("/dropmessages")) { nDropMessagesTest = atoi(mapArgs["/dropmessages"]); @@ -3380,6 +3467,20 @@ bool CMyApp::OnInit2() return false; } + if (mapArgs.count("/proxy")) + { + fUseProxy = true; + addrProxy = CAddress(mapArgs["/proxy"].c_str()); + if (addrProxy.ip == INADDR_NONE) + { + wxMessageBox("Invalid /proxy address", "Bitcoin"); + OnExit(); + } + CWalletDB walletdb; + walletdb.WriteSetting("fUseProxy", fUseProxy); + walletdb.WriteSetting("addrProxy", addrProxy); + } + if (mapArgs.count("/gen")) { if (mapArgs["/gen"].empty()) @@ -3404,7 +3505,7 @@ bool CMyApp::OnInit2() return false; } - //RandAddSeedPerfmon(); + RandAddSeedPerfmon(); if (!StartNode(strErrors)) wxMessageBox(strErrors, "Bitcoin"); @@ -3514,7 +3615,7 @@ void CMyApp::OnFatalException() void MainFrameRepaint() { - // This is called by network code that shouldn't access pframeMain and ptaskbaricon + // This is called by network code that shouldn't access pframeMain // directly because it could still be running after the UI is closed. if (pframeMain) { @@ -3523,20 +3624,47 @@ void MainFrameRepaint() pframeMain->Refresh(); pframeMain->AddPendingEvent(event); } - if (ptaskbaricon) - ptaskbaricon->UpdateTooltip(); } +typedef WINSHELLAPI BOOL WINAPI (*PSHGETSPECIALFOLDERPATHA)(HWND hwndOwner, LPSTR lpszPath, int nFolder, BOOL fCreate); + +string MyGetSpecialFolderPath(int nFolder, bool fCreate) +{ + char pszPath[MAX_PATH+100] = ""; + + // SHGetSpecialFolderPath is not usually available on NT 4.0 + HMODULE hShell32 = LoadLibrary("shell32.dll"); + if (hShell32) + { + PSHGETSPECIALFOLDERPATHA pSHGetSpecialFolderPath = + (PSHGETSPECIALFOLDERPATHA)GetProcAddress(hShell32, "SHGetSpecialFolderPathA"); + if (pSHGetSpecialFolderPath) + (*pSHGetSpecialFolderPath)(NULL, pszPath, nFolder, fCreate); + FreeModule(hShell32); + } + + // Backup option + if (pszPath[0] == '\0') + { + if (nFolder == CSIDL_STARTUP) + { + strcpy(pszPath, getenv("USERPROFILE")); + strcat(pszPath, "\\Start Menu\\Programs\\Startup"); + } + else if (nFolder == CSIDL_APPDATA) + { + strcpy(pszPath, getenv("APPDATA")); + } + } + + return pszPath; +} + string StartupShortcutPath() { - // Get the startup folder shortcut path - char pszLinkPath[MAX_PATH+100]; - pszLinkPath[0] = '\0'; - SHGetSpecialFolderPath(0, pszLinkPath, CSIDL_STARTUP, 0); - strcat(pszLinkPath, "\\Bitcoin.lnk"); - return pszLinkPath; + return MyGetSpecialFolderPath(CSIDL_STARTUP, true) + "\\Bitcoin.lnk"; } bool GetStartOnSystemStartup() @@ -3630,7 +3758,8 @@ void ThreadRandSendTest(void* parg) if (GetBalance() < nValue) { wxMessageBox("Out of money "); - return; + while (GetBalance() < 1000) + Sleep(1000); } nValue += (nRep % 100) * CENT; diff --git a/ui.h b/ui.h index f451d935..9fc7e0eb 100644 --- a/ui.h +++ b/ui.h @@ -89,7 +89,7 @@ public: void OnCrossThreadCall(wxCommandEvent& event); void InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5); bool DeleteLine(uint256 hashKey); - void InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); + bool InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); void RefreshListCtrl(); void RefreshStatus(); }; @@ -121,6 +121,9 @@ protected: void OnKillFocusTransactionFee(wxFocusEvent& event); void OnCheckBoxLimitProcessors(wxCommandEvent& event); void OnCheckBoxMinimizeToTray(wxCommandEvent& event); + void OnCheckBoxUseProxy(wxCommandEvent& event); + void OnKillFocusProxy(wxFocusEvent& event); + void OnButtonOK(wxCommandEvent& event); void OnButtonCancel(wxCommandEvent& event); void OnButtonApply(wxCommandEvent& event); @@ -133,6 +136,7 @@ public: bool fTmpStartOnSystemStartup; bool fTmpMinimizeOnClose; void SelectPage(int nPage); + CAddress GetProxyAddr(); }; @@ -193,7 +197,7 @@ public: int64 nPrice; CWalletTx wtx; wxDateTime start; - string strStatus; + char pszStatus[10000]; bool fCanCancel; bool fAbort; bool fSuccess; diff --git a/uibase.cpp b/uibase.cpp index 0f9e22c5..b03e578b 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -380,7 +380,7 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer69 = new wxBoxSizer( wxVERTICAL ); - bSizer69->Add( 0, 14, 0, wxEXPAND, 5 ); + bSizer69->Add( 0, 16, 0, wxEXPAND, 5 ); m_staticText32 = new wxStaticText( m_panelMain, wxID_ANY, wxT("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText32->Wrap( -1 ); @@ -412,7 +412,7 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer71->Add( m_checkBoxLimitProcessors, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); m_spinCtrlLimitProcessors = new wxSpinCtrl( m_panelMain, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( 48,-1 ), wxSP_ARROW_KEYS, 1, 999, 1 ); - bSizer71->Add( m_spinCtrlLimitProcessors, 0, wxTOP|wxBOTTOM, 5 ); + bSizer71->Add( m_spinCtrlLimitProcessors, 0, wxALIGN_CENTER_VERTICAL, 5 ); m_staticText35 = new wxStaticText( m_panelMain, wxID_ANY, wxT("processors"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText35->Wrap( -1 ); @@ -434,12 +434,45 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer101->Add( 16, 0, 0, 0, 5 ); - m_checkBoxMinimizeOnClose = new wxCheckBox( m_panelMain, wxID_ANY, wxT("M&inimize to system tray on close"), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkBoxMinimizeOnClose = new wxCheckBox( m_panelMain, wxID_ANY, wxT("Mi&nimize to system tray on close"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer101->Add( m_checkBoxMinimizeOnClose, 0, wxALL, 5 ); + bSizer101->Add( m_checkBoxMinimizeOnClose, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); bSizer69->Add( bSizer101, 1, wxEXPAND, 5 ); + wxBoxSizer* bSizer102; + bSizer102 = new wxBoxSizer( wxHORIZONTAL ); + + m_checkBoxUseProxy = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Connect through socks4 proxy: "), wxDefaultPosition, wxDefaultSize, 0 ); + + bSizer102->Add( m_checkBoxUseProxy, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + bSizer69->Add( bSizer102, 1, wxEXPAND, 5 ); + + wxBoxSizer* bSizer103; + bSizer103 = new wxBoxSizer( wxHORIZONTAL ); + + + bSizer103->Add( 18, 0, 0, 0, 5 ); + + m_staticTextProxyIP = new wxStaticText( m_panelMain, wxID_ANY, wxT("Proxy &IP:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextProxyIP->Wrap( -1 ); + bSizer103->Add( m_staticTextProxyIP, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlProxyIP = new wxTextCtrl( m_panelMain, wxID_PROXYIP, wxEmptyString, wxDefaultPosition, wxSize( 140,-1 ), 0 ); + m_textCtrlProxyIP->SetMaxLength( 15 ); + bSizer103->Add( m_textCtrlProxyIP, 0, wxALIGN_CENTER_VERTICAL, 5 ); + + m_staticTextProxyPort = new wxStaticText( m_panelMain, wxID_ANY, wxT(" &Port:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextProxyPort->Wrap( -1 ); + bSizer103->Add( m_staticTextProxyPort, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); + + m_textCtrlProxyPort = new wxTextCtrl( m_panelMain, wxID_PROXYPORT, wxEmptyString, wxDefaultPosition, wxSize( 55,-1 ), 0 ); + m_textCtrlProxyPort->SetMaxLength( 5 ); + bSizer103->Add( m_textCtrlProxyPort, 0, wxALIGN_CENTER_VERTICAL, 5 ); + + bSizer69->Add( bSizer103, 1, wxEXPAND, 5 ); + m_panelMain->SetSizer( bSizer69 ); m_panelMain->Layout(); bSizer69->Fit( m_panelMain ); @@ -450,13 +483,13 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer64 = new wxBoxSizer( wxVERTICAL ); - bSizer64->Add( 0, 14, 0, wxEXPAND, 5 ); + bSizer64->Add( 0, 16, 0, wxEXPAND, 5 ); m_staticText321 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("Test panel 2 for future expansion"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText321->Wrap( -1 ); bSizer64->Add( m_staticText321, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); - m_staticText69 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText69 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("Let's not start multiple pages until the first page is filled up"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText69->Wrap( -1 ); bSizer64->Add( m_staticText69, 0, wxALL, 5 ); @@ -506,6 +539,9 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w m_textCtrlTransactionFee->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusTransactionFee ), NULL, this ); m_checkBoxLimitProcessors->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxLimitProcessors ), NULL, this ); m_checkBoxMinimizeToTray->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxMinimizeToTray ), NULL, this ); + m_checkBoxUseProxy->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxUseProxy ), NULL, this ); + m_textCtrlProxyIP->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusProxy ), NULL, this ); + m_textCtrlProxyPort->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusProxy ), NULL, this ); m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonOK ), NULL, this ); m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonCancel ), NULL, this ); m_buttonApply->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonApply ), NULL, this ); @@ -518,6 +554,9 @@ COptionsDialogBase::~COptionsDialogBase() m_textCtrlTransactionFee->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusTransactionFee ), NULL, this ); m_checkBoxLimitProcessors->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxLimitProcessors ), NULL, this ); m_checkBoxMinimizeToTray->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxMinimizeToTray ), NULL, this ); + m_checkBoxUseProxy->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnCheckBoxUseProxy ), NULL, this ); + m_textCtrlProxyIP->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusProxy ), NULL, this ); + m_textCtrlProxyPort->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( COptionsDialogBase::OnKillFocusProxy ), NULL, this ); m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonOK ), NULL, this ); m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonCancel ), NULL, this ); m_buttonApply->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( COptionsDialogBase::OnButtonApply ), NULL, this ); diff --git a/uibase.h b/uibase.h index e262a3dc..d52158f1 100644 --- a/uibase.h +++ b/uibase.h @@ -50,50 +50,52 @@ #define wxID_BUTTONCOPY 1006 #define wxID_BUTTONCHANGE 1007 #define wxID_TRANSACTIONFEE 1008 -#define wxID_TEXTCTRLPAYTO 1009 -#define wxID_BUTTONPASTE 1010 -#define wxID_BUTTONADDRESSBOOK 1011 -#define wxID_TEXTCTRLAMOUNT 1012 -#define wxID_CHOICETRANSFERTYPE 1013 -#define wxID_LISTCTRL 1014 -#define wxID_BUTTONRENAME 1015 -#define wxID_BUTTONNEW 1016 -#define wxID_BUTTONEDIT 1017 -#define wxID_BUTTONDELETE 1018 -#define wxID_DEL0 1019 -#define wxID_DEL1 1020 -#define wxID_DEL2 1021 -#define wxID_DEL3 1022 -#define wxID_DEL4 1023 -#define wxID_DEL5 1024 -#define wxID_DEL6 1025 -#define wxID_DEL7 1026 -#define wxID_DEL8 1027 -#define wxID_DEL9 1028 -#define wxID_DEL10 1029 -#define wxID_DEL11 1030 -#define wxID_DEL12 1031 -#define wxID_DEL13 1032 -#define wxID_DEL14 1033 -#define wxID_DEL15 1034 -#define wxID_DEL16 1035 -#define wxID_DEL17 1036 -#define wxID_DEL18 1037 -#define wxID_DEL19 1038 -#define wxID_BUTTONPREVIEW 1039 -#define wxID_BUTTONSAMPLE 1040 -#define wxID_CANCEL2 1041 -#define wxID_BUTTONBACK 1042 -#define wxID_BUTTONNEXT 1043 -#define wxID_SUBMIT 1044 -#define wxID_OPENNEWTABLE 1045 -#define wxID_DEALHAND 1046 -#define wxID_FOLD 1047 -#define wxID_CALL 1048 -#define wxID_RAISE 1049 -#define wxID_LEAVETABLE 1050 -#define wxID_DITCHPLAYER 1051 -#define wxID_TEXTCTRL 1052 +#define wxID_PROXYIP 1009 +#define wxID_PROXYPORT 1010 +#define wxID_TEXTCTRLPAYTO 1011 +#define wxID_BUTTONPASTE 1012 +#define wxID_BUTTONADDRESSBOOK 1013 +#define wxID_TEXTCTRLAMOUNT 1014 +#define wxID_CHOICETRANSFERTYPE 1015 +#define wxID_LISTCTRL 1016 +#define wxID_BUTTONRENAME 1017 +#define wxID_BUTTONNEW 1018 +#define wxID_BUTTONEDIT 1019 +#define wxID_BUTTONDELETE 1020 +#define wxID_DEL0 1021 +#define wxID_DEL1 1022 +#define wxID_DEL2 1023 +#define wxID_DEL3 1024 +#define wxID_DEL4 1025 +#define wxID_DEL5 1026 +#define wxID_DEL6 1027 +#define wxID_DEL7 1028 +#define wxID_DEL8 1029 +#define wxID_DEL9 1030 +#define wxID_DEL10 1031 +#define wxID_DEL11 1032 +#define wxID_DEL12 1033 +#define wxID_DEL13 1034 +#define wxID_DEL14 1035 +#define wxID_DEL15 1036 +#define wxID_DEL16 1037 +#define wxID_DEL17 1038 +#define wxID_DEL18 1039 +#define wxID_DEL19 1040 +#define wxID_BUTTONPREVIEW 1041 +#define wxID_BUTTONSAMPLE 1042 +#define wxID_CANCEL2 1043 +#define wxID_BUTTONBACK 1044 +#define wxID_BUTTONNEXT 1045 +#define wxID_SUBMIT 1046 +#define wxID_OPENNEWTABLE 1047 +#define wxID_DEALHAND 1048 +#define wxID_FOLD 1049 +#define wxID_CALL 1050 +#define wxID_RAISE 1051 +#define wxID_LEAVETABLE 1052 +#define wxID_DITCHPLAYER 1053 +#define wxID_TEXTCTRL 1054 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase @@ -211,6 +213,12 @@ class COptionsDialogBase : public wxDialog wxCheckBox* m_checkBoxMinimizeToTray; wxCheckBox* m_checkBoxMinimizeOnClose; + wxCheckBox* m_checkBoxUseProxy; + + wxStaticText* m_staticTextProxyIP; + wxTextCtrl* m_textCtrlProxyIP; + wxStaticText* m_staticTextProxyPort; + wxTextCtrl* m_textCtrlProxyPort; wxPanel* m_panelTest2; wxStaticText* m_staticText321; @@ -226,6 +234,8 @@ class COptionsDialogBase : public wxDialog virtual void OnKillFocusTransactionFee( wxFocusEvent& event ){ event.Skip(); } virtual void OnCheckBoxLimitProcessors( wxCommandEvent& event ){ event.Skip(); } virtual void OnCheckBoxMinimizeToTray( wxCommandEvent& event ){ event.Skip(); } + virtual void OnCheckBoxUseProxy( wxCommandEvent& event ){ event.Skip(); } + virtual void OnKillFocusProxy( wxFocusEvent& event ){ event.Skip(); } virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonApply( wxCommandEvent& event ){ event.Skip(); } diff --git a/uiproject.fbp b/uiproject.fbp index af5876b4..313e5aa6 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -1912,7 +1912,7 @@ wxEXPAND 0 - 14 + 16 protected 0 @@ -2148,7 +2148,7 @@ 5 - wxTOP|wxBOTTOM + wxALIGN_CENTER_VERTICAL 0 @@ -2379,7 +2379,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL 0 @@ -2390,7 +2390,7 @@ 0 wxID_ANY - M&inimize to system tray on close + Mi&nimize to system tray on close m_checkBoxMinimizeOnClose @@ -2431,6 +2431,302 @@ + + 5 + wxEXPAND + 1 + + + bSizer102 + wxHORIZONTAL + none + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + &Connect through socks4 proxy: + + + m_checkBoxUseProxy + protected + + + + + + + + + + OnCheckBoxUseProxy + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 1 + + + bSizer103 + wxHORIZONTAL + none + + 5 + + 0 + + 0 + protected + 18 + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + Proxy &IP: + + + m_staticTextProxyIP + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_PROXYIP + + 15 + + m_textCtrlProxyIP + protected + + 140,-1 + + + + + + + + + + + + + OnKillFocusProxy + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_ANY + &Port: + + + m_staticTextProxyPort + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL + 0 + + + + 1 + + + 0 + wxID_PROXYPORT + + 5 + + m_textCtrlProxyPort + protected + + 55,-1 + + + + + + + + + + + + + OnKillFocusProxy + + + + + + + + + + + + + + + + + + + + + + + + + @@ -2490,7 +2786,7 @@ wxEXPAND 0 - 14 + 16 protected 0 @@ -2558,7 +2854,7 @@ 0 wxID_ANY - MyLabel + Let's not start multiple pages until the first page is filled up m_staticText69 diff --git a/util.cpp b/util.cpp index 8271474e..6ba84a51 100644 --- a/util.cpp +++ b/util.cpp @@ -5,8 +5,9 @@ #include "headers.h" - bool fDebug = false; +bool fPrintToDebugger = false; +bool fPrintToConsole = false; @@ -37,8 +38,8 @@ public: // Seed random number generator with screen scrape and other hardware sources RAND_screen(); - // Seed random number generator with perfmon data - RandAddSeed(true); + // Seed random number generator with performance counter + RandAddSeed(); } ~CInit() { @@ -54,40 +55,43 @@ instance_of_cinit; -void RandAddSeed(bool fPerfmon) +void RandAddSeed() { // Seed with CPU performance counter LARGE_INTEGER PerformanceCount; QueryPerformanceCounter(&PerformanceCount); RAND_add(&PerformanceCount, sizeof(PerformanceCount), 1.5); memset(&PerformanceCount, 0, sizeof(PerformanceCount)); +} +void RandAddSeedPerfmon() +{ + // This can take up to 2 seconds, so only do it every 10 minutes static int64 nLastPerfmon; - if (fPerfmon || GetTime() > nLastPerfmon + 5 * 60) + if (GetTime() < nLastPerfmon + 10 * 60) + return; + nLastPerfmon = GetTime(); + + // Seed with the entire set of perfmon data + unsigned char pdata[250000]; + memset(pdata, 0, sizeof(pdata)); + unsigned long nSize = sizeof(pdata); + long ret = RegQueryValueEx(HKEY_PERFORMANCE_DATA, "Global", NULL, NULL, pdata, &nSize); + RegCloseKey(HKEY_PERFORMANCE_DATA); + if (ret == ERROR_SUCCESS) { - nLastPerfmon = GetTime(); - - // Seed with the entire set of perfmon data - unsigned char pdata[250000]; - memset(pdata, 0, sizeof(pdata)); - unsigned long nSize = sizeof(pdata); - long ret = RegQueryValueEx(HKEY_PERFORMANCE_DATA, "Global", NULL, NULL, pdata, &nSize); - RegCloseKey(HKEY_PERFORMANCE_DATA); - if (ret == ERROR_SUCCESS) - { - uint256 hash; - SHA256(pdata, nSize, (unsigned char*)&hash); - RAND_add(&hash, sizeof(hash), min(nSize/500.0, (double)sizeof(hash))); - hash = 0; - memset(pdata, 0, nSize); - - time_t nTime; - time(&nTime); - struct tm* ptmTime = gmtime(&nTime); - char pszTime[200]; - strftime(pszTime, sizeof(pszTime), "%x %H:%M:%S", ptmTime); - printf("%s RandAddSeed() %d bytes\n", pszTime, nSize); - } + uint256 hash; + SHA256(pdata, nSize, (unsigned char*)&hash); + RAND_add(&hash, sizeof(hash), min(nSize/500.0, (double)sizeof(hash))); + hash = 0; + memset(pdata, 0, nSize); + + time_t nTime; + time(&nTime); + struct tm* ptmTime = gmtime(&nTime); + char pszTime[200]; + strftime(pszTime, sizeof(pszTime), "%x %H:%M:%S", ptmTime); + printf("%s RandAddSeed() %d bytes\n", pszTime, nSize); } } @@ -99,7 +103,6 @@ void RandAddSeed(bool fPerfmon) - // Safer snprintf // - prints up to limit-1 characters // - output string is always null terminated even if limit reached @@ -172,27 +175,6 @@ bool error(const char* format, ...) } -void PrintException(std::exception* pex, const char* pszThread) -{ - char pszModule[MAX_PATH]; - pszModule[0] = '\0'; - GetModuleFileName(NULL, pszModule, sizeof(pszModule)); - _strlwr(pszModule); - char pszMessage[1000]; - if (pex) - snprintf(pszMessage, sizeof(pszMessage), - "EXCEPTION: %s \n%s \n%s in %s \n", typeid(*pex).name(), pex->what(), pszModule, pszThread); - else - snprintf(pszMessage, sizeof(pszMessage), - "UNKNOWN EXCEPTION \n%s in %s \n", pszModule, pszThread); - printf("\n\n************************\n%s", pszMessage); - if (wxTheApp) - wxMessageBox(pszMessage, "Error", wxOK | wxICON_ERROR); - throw; - //DebugBreak(); -} - - void ParseString(const string& str, char c, vector& v) { unsigned int i1 = 0; @@ -268,6 +250,92 @@ bool ParseMoney(const char* pszIn, int64& nRet) } +vector ParseHex(const char* psz) +{ + vector vch; + while (isspace(*psz)) + psz++; + vch.reserve((strlen(psz)+1)/3); + + static char phexdigit[256] = + { -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + 0,1,2,3,4,5,6,7,8,9,-1,-1,-1,-1,-1,-1, + -1,0xa,0xb,0xc,0xd,0xe,0xf,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,0xa,0xb,0xc,0xd,0xe,0xf,-1,-1,-1,-1,-1,-1,-1,-1,-1 + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1, }; + + while (*psz) + { + char c = phexdigit[(unsigned char)*psz++]; + if (c == -1) + break; + unsigned char n = (c << 4); + if (*psz) + { + char c = phexdigit[(unsigned char)*psz++]; + if (c == -1) + break; + n |= c; + vch.push_back(n); + } + while (isspace(*psz)) + psz++; + } + + return vch; +} + +vector ParseHex(const std::string& str) +{ + return ParseHex(str.c_str()); +} + + + + + + +void FormatException(char* pszMessage, std::exception* pex, const char* pszThread) +{ + char pszModule[MAX_PATH]; + pszModule[0] = '\0'; + GetModuleFileName(NULL, pszModule, sizeof(pszModule)); + if (pex) + snprintf(pszMessage, 1000, + "EXCEPTION: %s \n%s \n%s in %s \n", typeid(*pex).name(), pex->what(), pszModule, pszThread); + else + snprintf(pszMessage, 1000, + "UNKNOWN EXCEPTION \n%s in %s \n", pszModule, pszThread); +} + +void LogException(std::exception* pex, const char* pszThread) +{ + char pszMessage[1000]; + FormatException(pszMessage, pex, pszThread); + printf("\n%s", pszMessage); +} + +void PrintException(std::exception* pex, const char* pszThread) +{ + char pszMessage[1000]; + FormatException(pszMessage, pex, pszThread); + printf("\n\n************************\n%s\n", pszMessage); + if (wxTheApp) + wxMessageBox(pszMessage, "Error", wxOK | wxICON_ERROR); + throw; + //DebugBreak(); +} @@ -363,7 +431,7 @@ void AddTimeData(unsigned int ip, int64 nTime) if (vTimeOffsets.empty()) vTimeOffsets.push_back(0); vTimeOffsets.push_back(nOffsetSample); - printf("Added time data, samples %d, ip %08x, offset %+I64d (%+I64d minutes)\n", vTimeOffsets.size(), ip, vTimeOffsets.back(), vTimeOffsets.back()/60); + printf("Added time data, samples %d, offset %+I64d (%+I64d minutes)\n", vTimeOffsets.size(), vTimeOffsets.back(), vTimeOffsets.back()/60); if (vTimeOffsets.size() >= 5 && vTimeOffsets.size() % 2 == 1) { sort(vTimeOffsets.begin(), vTimeOffsets.end()); diff --git a/util.h b/util.h index 97617ab4..61f774a8 100644 --- a/util.h +++ b/util.h @@ -67,15 +67,22 @@ inline T& REF(const T& val) extern bool fDebug; +extern bool fPrintToDebugger; +extern bool fPrintToConsole; +extern map mapArgs; -void RandAddSeed(bool fPerfmon=false); +void RandAddSeed(); +void RandAddSeedPerfmon(); int my_snprintf(char* buffer, size_t limit, const char* format, ...); string strprintf(const char* format, ...); bool error(const char* format, ...); void PrintException(std::exception* pex, const char* pszThread); +void LogException(std::exception* pex, const char* pszThread); void ParseString(const string& str, char c, vector& v); string FormatMoney(int64 n, bool fPlus=false); bool ParseMoney(const char* pszIn, int64& nRet); +vector ParseHex(const char* psz); +vector ParseHex(const std::string& str); bool FileExists(const char* psz); int GetFilesize(FILE* file); uint64 GetRand(uint64 nMax); @@ -94,6 +101,7 @@ void AddTimeData(unsigned int ip, int64 nTime); + // Wrapper to automatically initialize critical section // Could use wxCriticalSection for portability, but it doesn't support TryEnterCriticalSection class CCriticalSection @@ -156,6 +164,85 @@ public: +inline int OutputDebugStringF(const char* pszFormat, ...) +{ + int ret = 0; +#ifdef __WXDEBUG__ + if (!fPrintToConsole) + { + // print to debug.log + FILE* fileout = fopen("debug.log", "a"); + if (fileout) + { + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + ret = vfprintf(fileout, pszFormat, arg_ptr); + va_end(arg_ptr); + fclose(fileout); + } + } + + if (fPrintToDebugger) + { + // accumulate a line at a time + static CCriticalSection cs_OutputDebugStringF; + CRITICAL_BLOCK(cs_OutputDebugStringF) + { + static char pszBuffer[50000]; + static char* pend; + if (pend == NULL) + pend = pszBuffer; + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + int limit = END(pszBuffer) - pend - 2; + int ret = _vsnprintf(pend, limit, pszFormat, arg_ptr); + va_end(arg_ptr); + if (ret < 0 || ret >= limit) + { + pend = END(pszBuffer) - 2; + *pend++ = '\n'; + } + else + pend += ret; + *pend = '\0'; + char* p1 = pszBuffer; + char* p2; + while (p2 = strchr(p1, '\n')) + { + p2++; + char c = *p2; + *p2 = '\0'; + OutputDebugString(p1); + *p2 = c; + p1 = p2; + } + if (p1 != pszBuffer) + memmove(pszBuffer, p1, pend - p1 + 1); + pend -= (p1 - pszBuffer); + } + } +#endif + + if (fPrintToConsole) + { + // print to console + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + ret = vprintf(pszFormat, arg_ptr); + va_end(arg_ptr); + } + return ret; +} + + + + + + + + + + inline string i64tostr(int64 n) { return strprintf("%"PRId64, n); @@ -205,6 +292,11 @@ string HexStr(const T itbegin, const T itend, bool fSpaces=true) return str; } +inline string HexStr(vector vch, bool fSpaces=true) +{ + return HexStr(vch.begin(), vch.end(), fSpaces); +} + template string HexNumStr(const T itbegin, const T itend, bool f0x=true) { @@ -222,75 +314,9 @@ void PrintHex(const T pbegin, const T pend, const char* pszFormat="%s", bool fSp printf(pszFormat, HexStr(pbegin, pend, fSpaces).c_str()); } - - - - - - - -inline int OutputDebugStringF(const char* pszFormat, ...) +inline void PrintHex(vector vch, const char* pszFormat="%s", bool fSpaces=true) { -#ifdef __WXDEBUG__ - // log file - FILE* fileout = fopen("debug.log", "a"); - if (fileout) - { - va_list arg_ptr; - va_start(arg_ptr, pszFormat); - vfprintf(fileout, pszFormat, arg_ptr); - va_end(arg_ptr); - fclose(fileout); - } - - // accumulate a line at a time - static CCriticalSection cs_OutputDebugStringF; - CRITICAL_BLOCK(cs_OutputDebugStringF) - { - static char pszBuffer[50000]; - static char* pend; - if (pend == NULL) - pend = pszBuffer; - va_list arg_ptr; - va_start(arg_ptr, pszFormat); - int limit = END(pszBuffer) - pend - 2; - int ret = _vsnprintf(pend, limit, pszFormat, arg_ptr); - va_end(arg_ptr); - if (ret < 0 || ret >= limit) - { - pend = END(pszBuffer) - 2; - *pend++ = '\n'; - } - else - pend += ret; - *pend = '\0'; - char* p1 = pszBuffer; - char* p2; - while (p2 = strchr(p1, '\n')) - { - p2++; - char c = *p2; - *p2 = '\0'; - OutputDebugString(p1); - *p2 = c; - p1 = p2; - } - if (p1 != pszBuffer) - memmove(pszBuffer, p1, pend - p1 + 1); - pend -= (p1 - pszBuffer); - return ret; - } -#endif - - if (!wxTheApp) - { - // print to console - va_list arg_ptr; - va_start(arg_ptr, pszFormat); - vprintf(pszFormat, arg_ptr); - va_end(arg_ptr); - } - return 0; + printf(pszFormat, HexStr(vch, fSpaces).c_str()); } From dc73b326f97f2ed7ec7b7e8485ebc9eb46e05ddb Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 29 Oct 2009 05:55:56 +0000 Subject: [PATCH 013/133] CCriticalSection using wxWidgets instead of Windows OS calls git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@19 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- net.h | 6 +++--- util.cpp | 20 ++++++++++++-------- util.h | 31 ++++++++++++++++++++----------- 3 files changed, 35 insertions(+), 22 deletions(-) diff --git a/net.h b/net.h index a0b2929d..5995ac33 100644 --- a/net.h +++ b/net.h @@ -604,7 +604,7 @@ public: void BeginMessage(const char* pszCommand) { - EnterCriticalSection(&cs_vSend); + cs_vSend.Enter(); if (nPushPos != -1) AbortMessage(); nPushPos = vSend.size(); @@ -618,7 +618,7 @@ public: return; vSend.resize(nPushPos); nPushPos = -1; - LeaveCriticalSection(&cs_vSend); + cs_vSend.Leave(); printf("(aborted)\n"); } @@ -643,7 +643,7 @@ public: printf("\n"); nPushPos = -1; - LeaveCriticalSection(&cs_vSend); + cs_vSend.Leave(); } void EndMessageAbortIfEmpty() diff --git a/util.cpp b/util.cpp index 6ba84a51..bdf899d7 100644 --- a/util.cpp +++ b/util.cpp @@ -13,14 +13,14 @@ bool fPrintToConsole = false; // Init openssl library multithreading support -static HANDLE* lock_cs; +static wxMutex** ppmutexOpenSSL; -void win32_locking_callback(int mode, int type, const char* file, int line) +void win32_locking_callback(int mode, int i, const char* file, int line) { if (mode & CRYPTO_LOCK) - WaitForSingleObject(lock_cs[type], INFINITE); + ppmutexOpenSSL[i]->Lock(); else - ReleaseMutex(lock_cs[type]); + ppmutexOpenSSL[i]->Unlock(); } // Init @@ -30,9 +30,9 @@ public: CInit() { // Init openssl library multithreading support - lock_cs = (HANDLE*)OPENSSL_malloc(CRYPTO_num_locks() * sizeof(HANDLE)); + ppmutexOpenSSL = (wxMutex**)OPENSSL_malloc(CRYPTO_num_locks() * sizeof(wxMutex*)); for (int i = 0; i < CRYPTO_num_locks(); i++) - lock_cs[i] = CreateMutex(NULL,FALSE,NULL); + ppmutexOpenSSL[i] = new wxMutex(); CRYPTO_set_locking_callback(win32_locking_callback); // Seed random number generator with screen scrape and other hardware sources @@ -46,8 +46,8 @@ public: // Shutdown openssl library multithreading support CRYPTO_set_locking_callback(NULL); for (int i =0 ; i < CRYPTO_num_locks(); i++) - CloseHandle(lock_cs[i]); - OPENSSL_free(lock_cs); + delete ppmutexOpenSSL[i]; + OPENSSL_free(ppmutexOpenSSL); } } instance_of_cinit; @@ -55,6 +55,10 @@ instance_of_cinit; + + + + void RandAddSeed() { // Seed with CPU performance counter diff --git a/util.h b/util.h index 61f774a8..436281c9 100644 --- a/util.h +++ b/util.h @@ -106,28 +106,38 @@ void AddTimeData(unsigned int ip, int64 nTime); // Could use wxCriticalSection for portability, but it doesn't support TryEnterCriticalSection class CCriticalSection { +#ifdef __WXMSW__ protected: CRITICAL_SECTION cs; public: - char* pszFile; - int nLine; explicit CCriticalSection() { InitializeCriticalSection(&cs); } ~CCriticalSection() { DeleteCriticalSection(&cs); } void Enter() { EnterCriticalSection(&cs); } void Leave() { LeaveCriticalSection(&cs); } bool TryEnter() { return TryEnterCriticalSection(&cs); } - CRITICAL_SECTION* operator&() { return &cs; } +#else +protected: + wxMutex mutex; +public: + explicit CCriticalSection() { } + ~CCriticalSection() { } + void Enter() { mutex.Lock(); } + void Leave() { mutex.Unlock(); } + bool TryEnter() { return mutex.TryLock() == wxMUTEX_NO_ERROR; } +#endif +public: + char* pszFile; + int nLine; }; // Automatically leave critical section when leaving block, needed for exception safety class CCriticalBlock { protected: - CRITICAL_SECTION* pcs; + CCriticalSection* pcs; public: - CCriticalBlock(CRITICAL_SECTION& csIn) { pcs = &csIn; EnterCriticalSection(pcs); } - CCriticalBlock(CCriticalSection& csIn) { pcs = &csIn; EnterCriticalSection(pcs); } - ~CCriticalBlock() { LeaveCriticalSection(pcs); } + CCriticalBlock(CCriticalSection& csIn) { pcs = &csIn; pcs->Enter(); } + ~CCriticalBlock() { pcs->Leave(); } }; // WARNING: This will catch continue and break! @@ -141,11 +151,10 @@ public: class CTryCriticalBlock { protected: - CRITICAL_SECTION* pcs; + CCriticalSection* pcs; public: - CTryCriticalBlock(CRITICAL_SECTION& csIn) { pcs = (TryEnterCriticalSection(&csIn) ? &csIn : NULL); } - CTryCriticalBlock(CCriticalSection& csIn) { pcs = (TryEnterCriticalSection(&csIn) ? &csIn : NULL); } - ~CTryCriticalBlock() { if (pcs) LeaveCriticalSection(pcs); } + CTryCriticalBlock(CCriticalSection& csIn) { pcs = (csIn.TryEnter() ? &csIn : NULL); } + ~CTryCriticalBlock() { if (pcs) pcs->Leave(); } bool Entered() { return pcs != NULL; } }; From e8474beb6f2c5e2654f8ebc671b3dbf5fae78563 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 29 Oct 2009 20:10:46 +0000 Subject: [PATCH 014/133] better wallet.dat flush, consolidated QueryPerformanceCounter, PRI64d printf portability git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@20 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build.txt | 3 ++- db.cpp | 76 +++++++++++++++++++++++++++++++++++++---------------- db.h | 12 ++++++++- main.cpp | 6 ++--- main.h | 2 +- makefile.vc | 4 +-- net.h | 2 +- ui.cpp | 26 +++++++++--------- util.cpp | 22 +++++++--------- util.h | 43 ++++++++++++++++++------------ 10 files changed, 121 insertions(+), 75 deletions(-) diff --git a/build.txt b/build.txt index f5100557..67b03fe2 100644 --- a/build.txt +++ b/build.txt @@ -10,7 +10,7 @@ cryptographic software written by Eric Young (eay@cryptsoft.com). Compilers Supported ------------------- -MinGW GCC (currently v3.4.5) +MinGW GCC Microsoft Visual C++ 6.0 SP6 @@ -20,6 +20,7 @@ Libraries you need to obtain separately to build: default path download wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ + or prebuilt: http://wxpack.sourceforge.net OpenSSL \openssl http://www.openssl.org/source/ Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html Boost \boost http://www.boost.org/users/download/ diff --git a/db.cpp b/db.cpp index 315e93b7..699a94f2 100644 --- a/db.cpp +++ b/db.cpp @@ -4,8 +4,11 @@ #include "headers.h" +void ThreadFlushWalletDB(void* parg); +unsigned int nWalletDBUpdated; + @@ -56,6 +59,8 @@ CDB::CDB(const char* pszFile, const char* pszMode, bool fTxn) : pdb(NULL) { if (!fDbEnvInit) { + if (fShutdown) + return; string strAppDir = GetAppDir(); string strLogDir = strAppDir + "\\database"; _mkdir(strLogDir.c_str()); @@ -121,12 +126,10 @@ void CDB::Close() pdb->close(0); delete pdb; pdb = NULL; + dbenv.txn_checkpoint(0, 0, 0); CRITICAL_BLOCK(cs_db) - { - dbenv.txn_checkpoint(0, 0, 0); --mapFileUseCount[strFile]; - } RandAddSeed(); } @@ -499,25 +502,6 @@ bool CReviewDB::WriteReviews(uint256 hash, const vector& vReviews) // CWalletDB // -CWalletDB::~CWalletDB() -{ - // Flush whenever all handles to wallet.dat are closed - CRITICAL_BLOCK(cs_db) - { - Close(); // close includes a txn_checkpoint - map::iterator mi = mapFileUseCount.find(strFile); - if (mi != mapFileUseCount.end()) - { - int nRefCount = (*mi).second; - if (nRefCount == 0) - { - dbenv.lsn_reset(strFile.c_str(), 0); - mapFileUseCount.erase(mi++); - } - } - } -} - bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) { vchDefaultKeyRet.clear(); @@ -610,7 +594,7 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) printf("fShowGenerated = %d\n", fShowGenerated); printf("fGenerateBitcoins = %d\n", fGenerateBitcoins); - printf("nTransactionFee = %I64d\n", nTransactionFee); + printf("nTransactionFee = %"PRI64d"\n", nTransactionFee); printf("addrIncoming = %s\n", addrIncoming.ToString().c_str()); printf("fMinimizeToTray = %d\n", fMinimizeToTray); printf("fMinimizeOnClose = %d\n", fMinimizeOnClose); @@ -655,5 +639,51 @@ bool LoadWallet(bool& fFirstRunRet) CWalletDB().WriteDefaultKey(keyUser.GetPubKey()); } + _beginthread(ThreadFlushWalletDB, 0, NULL); return true; } + +void ThreadFlushWalletDB(void* parg) +{ + static bool fOneThread; + if (fOneThread) + return; + fOneThread = true; + + unsigned int nLastSeen = nWalletDBUpdated; + unsigned int nLastFlushed = nWalletDBUpdated; + int64 nLastWalletUpdate = GetTime(); + while (!fShutdown) + { + Sleep(500); + + if (nLastSeen != nWalletDBUpdated) + { + nLastSeen = nWalletDBUpdated; + nLastWalletUpdate = GetTime(); + } + + if (nLastFlushed != nWalletDBUpdated && nLastWalletUpdate < GetTime() - 1) + { + TRY_CRITICAL_BLOCK(cs_db) + { + string strFile = "wallet.dat"; + map::iterator mi = mapFileUseCount.find(strFile); + if (mi != mapFileUseCount.end()) + { + int nRefCount = (*mi).second; + if (nRefCount == 0 && !fShutdown) + { + // Flush wallet.dat so it's self contained + nLastFlushed == nWalletDBUpdated; + int64 nStart = PerformanceCounter(); + dbenv.txn_checkpoint(0, 0, 0); + dbenv.lsn_reset(strFile.c_str(), 0); + printf("Flushed wallet.dat %15"PRI64d"\n", PerformanceCounter() - nStart); + mapFileUseCount.erase(mi++); + } + } + } + } + } +} diff --git a/db.h b/db.h index 49614687..d11b397e 100644 --- a/db.h +++ b/db.h @@ -17,7 +17,10 @@ extern map mapAddressBook; extern bool fClient; +extern unsigned int nWalletDBUpdated; extern DbEnv dbenv; + + extern void DBFlush(bool fShutdown); @@ -334,11 +337,11 @@ bool LoadAddresses(); + class CWalletDB : public CDB { public: CWalletDB(const char* pszMode="r+", bool fTxn=false) : CDB("wallet.dat", pszMode, fTxn) { } - ~CWalletDB(); private: CWalletDB(const CWalletDB&); void operator=(const CWalletDB&); @@ -351,12 +354,14 @@ public: bool WriteName(const string& strAddress, const string& strName) { + nWalletDBUpdated++; mapAddressBook[strAddress] = strName; return Write(make_pair(string("name"), strAddress), strName); } bool EraseName(const string& strAddress) { + nWalletDBUpdated++; mapAddressBook.erase(strAddress); return Erase(make_pair(string("name"), strAddress)); } @@ -368,11 +373,13 @@ public: bool WriteTx(uint256 hash, const CWalletTx& wtx) { + nWalletDBUpdated++; return Write(make_pair(string("tx"), hash), wtx); } bool EraseTx(uint256 hash) { + nWalletDBUpdated++; return Erase(make_pair(string("tx"), hash)); } @@ -384,6 +391,7 @@ public: bool WriteKey(const vector& vchPubKey, const CPrivKey& vchPrivKey) { + nWalletDBUpdated++; return Write(make_pair(string("key"), vchPubKey), vchPrivKey, false); } @@ -395,6 +403,7 @@ public: bool WriteDefaultKey(const vector& vchPubKey) { + nWalletDBUpdated++; return Write(string("defaultkey"), vchPubKey); } @@ -407,6 +416,7 @@ public: template bool WriteSetting(const string& strKey, const T& value) { + nWalletDBUpdated++; return Write(make_pair(string("setting"), strKey), value); } diff --git a/main.cpp b/main.cpp index 710b7892..4194333d 100644 --- a/main.cpp +++ b/main.cpp @@ -2518,8 +2518,7 @@ bool BitcoinMiner() int64 GetBalance() { - int64 nStart, nEnd; - QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + int64 nStart = PerformanceCounter(); int64 nTotal = 0; CRITICAL_BLOCK(cs_mapWallet) @@ -2533,8 +2532,7 @@ int64 GetBalance() } } - QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); - ///printf(" GetBalance() time = %16I64d\n", nEnd - nStart); + ///printf(" GetBalance() time = %15"PRI64d"\n", PerformanceCounter() - nStart); return nTotal; } diff --git a/main.h b/main.h index 958f7a5f..fcfd33d1 100644 --- a/main.h +++ b/main.h @@ -344,7 +344,7 @@ public: { if (scriptPubKey.size() < 6) return "CTxOut(error)"; - return strprintf("CTxOut(nValue=%I64d.%08I64d, scriptPubKey=%s)", nValue / COIN, nValue % COIN, scriptPubKey.ToString().substr(0,24).c_str()); + return strprintf("CTxOut(nValue=%"PRI64d".%08"PRI64d", scriptPubKey=%s)", nValue / COIN, nValue % COIN, scriptPubKey.ToString().substr(0,24).c_str()); } void print() const diff --git a/makefile.vc b/makefile.vc index c3bd0c4b..fb7f3086 100644 --- a/makefile.vc +++ b/makefile.vc @@ -13,8 +13,8 @@ DEBUGFLAGS=/Zi /Od /D__WXDEBUG__ -INCLUDEPATHS=/I"/boost" /I"/DB/build_windows" /I"/OpenSSL/include" /I"/wxWidgets/lib/vc_lib/mswd" /I"/wxWidgets/include" -LIBPATHS=/LIBPATH:"/DB/build_windows/$(BUILD)" /LIBPATH:"/OpenSSL/out" /LIBPATH:"/wxWidgets/lib/vc_lib" +INCLUDEPATHS=/I"/boost" /I"/db/build_windows" /I"/openssl/include" /I"/wxwidgets/lib/vc_lib/mswd" /I"/wxwidgets/include" +LIBPATHS=/LIBPATH:"/db/build_windows/$(BUILD)" /LIBPATH:"/openssl/out" /LIBPATH:"/wxwidgets/lib/vc_lib" LIBS= \ libdb47s$(D).lib \ libeay32.lib \ diff --git a/net.h b/net.h index 5995ac33..4011a3ef 100644 --- a/net.h +++ b/net.h @@ -589,7 +589,7 @@ public: // We're using mapAskFor as a priority queue, // the key is the earliest time the request can be sent int64& nRequestTime = mapAlreadyAskedFor[inv]; - printf("askfor %s %I64d\n", inv.ToString().c_str(), nRequestTime); + printf("askfor %s %"PRI64d"\n", inv.ToString().c_str(), nRequestTime); // Make sure not to reuse time indexes to keep things in the same order int64 nNow = (GetTime() - 1) * 1000000; diff --git a/ui.cpp b/ui.cpp index 917c64b4..ce287e39 100644 --- a/ui.cpp +++ b/ui.cpp @@ -3414,28 +3414,25 @@ bool CMyApp::OnInit2() // bool fFirstRun; string strErrors; - int64 nStart, nEnd; + int64 nStart; printf("Loading addresses...\n"); - QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + nStart = PerformanceCounter(); if (!LoadAddresses()) strErrors += "Error loading addr.dat \n"; - QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); - printf(" addresses %20I64d\n", nEnd - nStart); + printf(" addresses %15"PRI64d"\n", PerformanceCounter() - nStart); printf("Loading block index...\n"); - QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + nStart = PerformanceCounter(); if (!LoadBlockIndex()) strErrors += "Error loading blkindex.dat \n"; - QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); - printf(" block index %20I64d\n", nEnd - nStart); + printf(" block index %15"PRI64d"\n", PerformanceCounter() - nStart); printf("Loading wallet...\n"); - QueryPerformanceCounter((LARGE_INTEGER*)&nStart); + nStart = PerformanceCounter(); if (!LoadWallet(fFirstRun)) strErrors += "Error loading wallet.dat \n"; - QueryPerformanceCounter((LARGE_INTEGER*)&nEnd); - printf(" wallet %20I64d\n", nEnd - nStart); + printf(" wallet %15"PRI64d"\n", PerformanceCounter() - nStart); printf("Done loading\n"); @@ -3742,7 +3739,7 @@ void ThreadRandSendTest(void* parg) return; } - loop + while (!fShutdown) { Sleep(GetRand(30) * 1000 + 100); @@ -3767,6 +3764,8 @@ void ThreadRandSendTest(void* parg) CScript scriptPubKey; scriptPubKey << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; + if (fShutdown) + return; if (!SendMoney(scriptPubKey, nValue, wtx)) return; } @@ -3776,8 +3775,6 @@ void ThreadRandSendTest(void* parg) // randsendtest to any connected node void RandSend() { - CWalletTx wtx; - while (vNodes.empty()) Sleep(1000); CAddress addr; @@ -3785,6 +3782,7 @@ void RandSend() addr = vNodes[GetRand(vNodes.size())]->addr; // Message + CWalletTx wtx; wtx.mapValue["to"] = addr.ToString(); wtx.mapValue["from"] = addrLocalHost.ToString(); static int nRep; @@ -3799,6 +3797,8 @@ void RandSend() } // Send to IP address + if (fShutdown) + return; CSendingDialog* pdialog = new CSendingDialog(pframeMain, addr, nValue, wtx); if (!pdialog->Show()) wxMessageBox("ShowModal Failed "); diff --git a/util.cpp b/util.cpp index bdf899d7..ef950920 100644 --- a/util.cpp +++ b/util.cpp @@ -14,8 +14,7 @@ bool fPrintToConsole = false; // Init openssl library multithreading support static wxMutex** ppmutexOpenSSL; - -void win32_locking_callback(int mode, int i, const char* file, int line) +void locking_callback(int mode, int i, const char* file, int line) { if (mode & CRYPTO_LOCK) ppmutexOpenSSL[i]->Lock(); @@ -33,7 +32,7 @@ public: ppmutexOpenSSL = (wxMutex**)OPENSSL_malloc(CRYPTO_num_locks() * sizeof(wxMutex*)); for (int i = 0; i < CRYPTO_num_locks(); i++) ppmutexOpenSSL[i] = new wxMutex(); - CRYPTO_set_locking_callback(win32_locking_callback); + CRYPTO_set_locking_callback(locking_callback); // Seed random number generator with screen scrape and other hardware sources RAND_screen(); @@ -45,7 +44,7 @@ public: { // Shutdown openssl library multithreading support CRYPTO_set_locking_callback(NULL); - for (int i =0 ; i < CRYPTO_num_locks(); i++) + for (int i = 0; i < CRYPTO_num_locks(); i++) delete ppmutexOpenSSL[i]; OPENSSL_free(ppmutexOpenSSL); } @@ -62,10 +61,9 @@ instance_of_cinit; void RandAddSeed() { // Seed with CPU performance counter - LARGE_INTEGER PerformanceCount; - QueryPerformanceCounter(&PerformanceCount); - RAND_add(&PerformanceCount, sizeof(PerformanceCount), 1.5); - memset(&PerformanceCount, 0, sizeof(PerformanceCount)); + int64 nCounter = PerformanceCounter(); + RAND_add(&nCounter, sizeof(nCounter), 1.5); + memset(&nCounter, 0, sizeof(nCounter)); } void RandAddSeedPerfmon() @@ -196,7 +194,7 @@ void ParseString(const string& str, char c, vector& v) string FormatMoney(int64 n, bool fPlus) { n /= CENT; - string str = strprintf("%I64d.%02I64d", (n > 0 ? n : -n)/100, (n > 0 ? n : -n)%100); + string str = strprintf("%"PRI64d".%02"PRI64d, (n > 0 ? n : -n)/100, (n > 0 ? n : -n)%100); for (int i = 6; i < str.size(); i += 4) if (isdigit(str[str.size() - i - 1])) str.insert(str.size() - i, 1, ','); @@ -435,7 +433,7 @@ void AddTimeData(unsigned int ip, int64 nTime) if (vTimeOffsets.empty()) vTimeOffsets.push_back(0); vTimeOffsets.push_back(nOffsetSample); - printf("Added time data, samples %d, offset %+I64d (%+I64d minutes)\n", vTimeOffsets.size(), vTimeOffsets.back(), vTimeOffsets.back()/60); + printf("Added time data, samples %d, offset %+"PRI64d" (%+"PRI64d" minutes)\n", vTimeOffsets.size(), vTimeOffsets.back(), vTimeOffsets.back()/60); if (vTimeOffsets.size() >= 5 && vTimeOffsets.size() % 2 == 1) { sort(vTimeOffsets.begin(), vTimeOffsets.end()); @@ -449,7 +447,7 @@ void AddTimeData(unsigned int ip, int64 nTime) /// to make sure it doesn't get changed again } foreach(int64 n, vTimeOffsets) - printf("%+I64d ", n); - printf("| nTimeOffset = %+I64d (%+I64d minutes)\n", nTimeOffset, nTimeOffset/60); + printf("%+"PRI64d" ", n); + printf("| nTimeOffset = %+"PRI64d" (%+"PRI64d" minutes)\n", nTimeOffset, nTimeOffset/60); } } diff --git a/util.h b/util.h index 436281c9..1c7215d2 100644 --- a/util.h +++ b/util.h @@ -13,7 +13,6 @@ typedef unsigned long long uint64; #if defined(_MSC_VER) && _MSC_VER < 1300 #define for if (false) ; else for #endif - #ifndef _MSC_VER #define __forceinline inline #endif @@ -25,25 +24,22 @@ typedef unsigned long long uint64; #define UBEGIN(a) ((unsigned char*)&(a)) #define UEND(a) ((unsigned char*)&((&(a))[1])) #define ARRAYLEN(array) (sizeof(array)/sizeof((array)[0])) - -#ifdef _WINDOWS #define printf OutputDebugStringF -#endif #ifdef snprintf #undef snprintf #endif #define snprintf my_snprintf -#ifndef PRId64 +#ifndef PRI64d #if defined(_MSC_VER) || defined(__BORLANDC__) || defined(__MSVCRT__) -#define PRId64 "I64d" -#define PRIu64 "I64u" -#define PRIx64 "I64x" +#define PRI64d "I64d" +#define PRI64u "I64u" +#define PRI64x "I64x" #else -#define PRId64 "lld" -#define PRIu64 "llu" -#define PRIx64 "llx" +#define PRI64d "lld" +#define PRI64u "llu" +#define PRI64x "llx" #endif #endif @@ -64,8 +60,6 @@ inline T& REF(const T& val) - - extern bool fDebug; extern bool fPrintToDebugger; extern bool fPrintToConsole; @@ -101,9 +95,7 @@ void AddTimeData(unsigned int ip, int64 nTime); - -// Wrapper to automatically initialize critical section -// Could use wxCriticalSection for portability, but it doesn't support TryEnterCriticalSection +// Wrapper to automatically initialize critical sections class CCriticalSection { #ifdef __WXMSW__ @@ -191,6 +183,7 @@ inline int OutputDebugStringF(const char* pszFormat, ...) } } +#ifdef __WXMSW__ if (fPrintToDebugger) { // accumulate a line at a time @@ -230,6 +223,7 @@ inline int OutputDebugStringF(const char* pszFormat, ...) pend -= (p1 - pszBuffer); } } +#endif #endif if (fPrintToConsole) @@ -254,7 +248,7 @@ inline int OutputDebugStringF(const char* pszFormat, ...) inline string i64tostr(int64 n) { - return strprintf("%"PRId64, n); + return strprintf("%"PRI64d, n); } inline string itostr(int n) @@ -328,6 +322,20 @@ inline void PrintHex(vector vch, const char* pszFormat="%s", bool printf(pszFormat, HexStr(vch, fSpaces).c_str()); } +inline int64 PerformanceCounter() +{ + int64 nCounter = 0; + QueryPerformanceCounter((LARGE_INTEGER*)&nCounter); + return nCounter; +} + +#ifndef __WXMSW__ +inline void Sleep(unsigned int nMilliseconds) +{ + wxMilliSleep(nMilliseconds); +} +#endif + @@ -370,6 +378,7 @@ inline void heapchk() + template inline uint256 Hash(const T1 pbegin, const T1 pend) { From a0c17c03eb323ddec28e36cfc30174a58264cbcf Mon Sep 17 00:00:00 2001 From: sirius-m Date: Fri, 30 Oct 2009 00:45:35 +0000 Subject: [PATCH 015/133] setup script removes startup shortcut and runs with highest exec level git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@21 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- setup.nsi | 3 +++ 1 file changed, 3 insertions(+) diff --git a/setup.nsi b/setup.nsi index 023914fa..e30ff13c 100644 --- a/setup.nsi +++ b/setup.nsi @@ -3,6 +3,8 @@ Name Bitcoin +RequestExecutionLevel highest + # General Symbol Definitions !define REGKEY "SOFTWARE\$(^Name)" !define VERSION 0.1.6 @@ -110,6 +112,7 @@ Section -un.post UNSEC0001 DeleteRegKey HKCU "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$(^Name)" Delete /REBOOTOK "$SMPROGRAMS\$StartMenuGroup\Uninstall Bitcoin.lnk" Delete /REBOOTOK "$SMPROGRAMS\$StartMenuGroup\Bitcoin.lnk" + Delete /REBOOTOK "$SMSTARTUP\Bitcoin.lnk" Delete /REBOOTOK $INSTDIR\uninstall.exe Delete /REBOOTOK $INSTDIR\db.log DeleteRegValue HKCU "${REGKEY}" StartMenuGroup From 32d490313b21d5cea64f99ca9db4388591bb3ab9 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 30 Oct 2009 00:57:05 +0000 Subject: [PATCH 016/133] make CheckDiskSpace portable git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@22 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 14 ++++---------- 1 file changed, 4 insertions(+), 10 deletions(-) diff --git a/main.cpp b/main.cpp index 4194333d..b98477b0 100644 --- a/main.cpp +++ b/main.cpp @@ -1398,21 +1398,15 @@ string GetAppDir() bool CheckDiskSpace(int64 nAdditionalBytes) { - uint64 nFreeBytesAvailable = 0; // bytes available to caller - uint64 nTotalNumberOfBytes = 0; // bytes on disk - uint64 nTotalNumberOfFreeBytes = 0; // free bytes on disk - - if (!GetDiskFreeSpaceEx(GetAppDir().c_str(), - (PULARGE_INTEGER)&nFreeBytesAvailable, - (PULARGE_INTEGER)&nTotalNumberOfBytes, - (PULARGE_INTEGER)&nTotalNumberOfFreeBytes)) + wxLongLong nFreeBytesAvailable = 0; + if (!wxGetDiskSpace(GetDataDir(), NULL, &nFreeBytesAvailable)) { - printf("ERROR: GetDiskFreeSpaceEx() failed\n"); + printf("ERROR: wxGetDiskSpace() failed\n"); return true; } // Check for 15MB because database could create another 10MB log file at any time - if ((int64)nFreeBytesAvailable < 15000000 + nAdditionalBytes) + if (nFreeBytesAvailable < (int64)15000000 + nAdditionalBytes) { fShutdown = true; wxMessageBox("Warning: Your disk space is low ", "Bitcoin", wxICON_EXCLAMATION); From fe9f3d626d79f71d819caf43878299fa5e6af83e Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sat, 31 Oct 2009 09:11:43 +0000 Subject: [PATCH 017/133] Linux alternatives for the Windows headers and PerformanceCounter. Some typedefs and #defines for the Linux build. Fixed GetDataDir. git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@23 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 27 ++++++++++++++++++--------- irc.h | 5 +++++ main.cpp | 2 +- net.h | 6 ++++++ util.h | 12 ++++++++++-- 5 files changed, 40 insertions(+), 12 deletions(-) diff --git a/headers.h b/headers.h index 29b16fb7..16238f89 100644 --- a/headers.h +++ b/headers.h @@ -25,20 +25,13 @@ #include #include #include -#include -#include -#include -#include -#include -#include +#include #include #include -#include #include #include #include #include -#include #include #include #define BOUNDSCHECK 1 @@ -56,7 +49,23 @@ #include #include #include -#include +#include + +#ifdef __WXMSW__ +#include +#include +#include +#include +#include +#include +#include +#else +#include +#include +#include +#include +#endif + #pragma hdrstop using namespace std; using namespace boost; diff --git a/irc.h b/irc.h index 91c3ffe6..1dc348a8 100644 --- a/irc.h +++ b/irc.h @@ -1,6 +1,11 @@ // Copyright (c) 2009 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#ifndef __WXMSW__ +#define closesocket(s) close(s) +typedef u_int SOCKET; +#endif extern bool RecvLine(SOCKET hSocket, string& strLine); extern void ThreadIRCSeed(void* parg); diff --git a/main.cpp b/main.cpp index b98477b0..15626777 100644 --- a/main.cpp +++ b/main.cpp @@ -1399,7 +1399,7 @@ string GetAppDir() bool CheckDiskSpace(int64 nAdditionalBytes) { wxLongLong nFreeBytesAvailable = 0; - if (!wxGetDiskSpace(GetDataDir(), NULL, &nFreeBytesAvailable)) + if (!wxGetDiskSpace(wxStandardPaths::Get().GetDataDir(), NULL, &nFreeBytesAvailable)) { printf("ERROR: wxGetDiskSpace() failed\n"); return true; diff --git a/net.h b/net.h index 4011a3ef..2eece2c0 100644 --- a/net.h +++ b/net.h @@ -1,6 +1,12 @@ // Copyright (c) 2009 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#ifndef __WXMSW__ +#define closesocket(s) close(s) +#define INVALID_SOCKET (SOCKET)(~0) +typedef u_int SOCKET; +#endif class CMessageHeader; class CAddress; diff --git a/util.h b/util.h index 1c7215d2..5d187760 100644 --- a/util.h +++ b/util.h @@ -321,11 +321,19 @@ inline void PrintHex(vector vch, const char* pszFormat="%s", bool { printf(pszFormat, HexStr(vch, fSpaces).c_str()); } + inline int64 PerformanceCounter() { - int64 nCounter = 0; - QueryPerformanceCounter((LARGE_INTEGER*)&nCounter); + int64 nCounter = 0; +#ifdef __WXMSW__ + QueryPerformanceCounter((LARGE_INTEGER*)&nCounter); +#else + // this could be changed to reading /dev/urandom + timeval t; + gettimeofday(&t, NULL); + nCounter += t.tv_sec * 1000000 + t.tv_usec; +#endif return nCounter; } From 5750932cdf72ea9b5e64b8a05b43c42f5acb417d Mon Sep 17 00:00:00 2001 From: sirius-m Date: Sat, 31 Oct 2009 09:52:24 +0000 Subject: [PATCH 018/133] added wx/stdpaths.h git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@24 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 1 + 1 file changed, 1 insertion(+) diff --git a/headers.h b/headers.h index 16238f89..f00b7dde 100644 --- a/headers.h +++ b/headers.h @@ -21,6 +21,7 @@ #include #include #include +#include #include #include #include From 4ac57f013e20da2d0f87fff69625cbd4419089f3 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 1 Nov 2009 01:16:51 +0000 Subject: [PATCH 019/133] move debug.log and db.log to data dir, portable GetDataDir, optimize GetBalance, fix repaint bogdown, -addnode and -? switches git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@25 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build.txt | 2 +- db.cpp | 19 ++- irc.cpp | 7 +- irc.h | 5 - main.cpp | 50 +----- main.h | 20 ++- net.cpp | 34 +++- net.h | 78 ++++++---- ui.cpp | 457 ++++++++++++++++++++++++------------------------------ ui.h | 5 +- util.cpp | 82 ++++++++-- util.h | 39 +++-- 12 files changed, 416 insertions(+), 382 deletions(-) diff --git a/build.txt b/build.txt index 67b03fe2..4368af85 100644 --- a/build.txt +++ b/build.txt @@ -10,7 +10,7 @@ cryptographic software written by Eric Young (eay@cryptsoft.com). Compilers Supported ------------------- -MinGW GCC +MinGW GCC (v3.4.5) Microsoft Visual C++ 6.0 SP6 diff --git a/db.cpp b/db.cpp index 699a94f2..f9e25834 100644 --- a/db.cpp +++ b/db.cpp @@ -61,18 +61,19 @@ CDB::CDB(const char* pszFile, const char* pszMode, bool fTxn) : pdb(NULL) { if (fShutdown) return; - string strAppDir = GetAppDir(); - string strLogDir = strAppDir + "\\database"; + string strDataDir = GetDataDir(); + string strLogDir = strDataDir + "\\database"; _mkdir(strLogDir.c_str()); - printf("dbenv.open strAppDir=%s\n", strAppDir.c_str()); + string strErrorFile = strDataDir + "\\db.log"; + printf("dbenv.open strLogDir=%s strErrorFile=%s\n", strLogDir.c_str(), strErrorFile.c_str()); dbenv.set_lg_dir(strLogDir.c_str()); dbenv.set_lg_max(10000000); dbenv.set_lk_max_locks(10000); dbenv.set_lk_max_objects(10000); - dbenv.set_errfile(fopen("db.log", "a")); /// debug + dbenv.set_errfile(fopen(strErrorFile.c_str(), "a")); /// debug ///dbenv.log_set_config(DB_LOG_AUTO_REMOVE, 1); /// causes corruption - ret = dbenv.open(strAppDir.c_str(), + ret = dbenv.open(strDataDir.c_str(), DB_CREATE | DB_INIT_LOCK | DB_INIT_LOG | @@ -139,6 +140,8 @@ void DBFlush(bool fShutdown) // Flush log data to the actual data file // on all files that are not in use printf("DBFlush(%s)\n", fShutdown ? "true" : "false"); + if (!fDbEnvInit) + return; CRITICAL_BLOCK(cs_db) { dbenv.txn_checkpoint(0, 0, 0); @@ -421,7 +424,7 @@ bool CAddrDB::LoadAddresses() while (fgets(psz, sizeof(psz), filein)) { CAddress addr(psz, NODE_NETWORK); - if (addr.ip != 0) + if (addr.IsValid()) { AddAddress(*this, addr); mapIRCAddresses.insert(make_pair(addr.GetKey(), addr)); @@ -676,10 +679,10 @@ void ThreadFlushWalletDB(void* parg) { // Flush wallet.dat so it's self contained nLastFlushed == nWalletDBUpdated; - int64 nStart = PerformanceCounter(); + int64 nStart = GetTimeMillis(); dbenv.txn_checkpoint(0, 0, 0); dbenv.lsn_reset(strFile.c_str(), 0); - printf("Flushed wallet.dat %15"PRI64d"\n", PerformanceCounter() - nStart); + printf("Flushed wallet.dat %"PRI64d"ms\n", GetTimeMillis() - nStart); mapFileUseCount.erase(mi++); } } diff --git a/irc.cpp b/irc.cpp index 707b4fe1..f839dc52 100644 --- a/irc.cpp +++ b/irc.cpp @@ -40,7 +40,7 @@ bool DecodeAddress(string str, CAddress& addr) return false; memcpy(&tmp, &vch[0], sizeof(tmp)); - addr = CAddress(tmp.ip, tmp.port); + addr = CAddress(tmp.ip, tmp.port, NODE_NETWORK); return true; } @@ -163,6 +163,7 @@ void ThreadIRCSeed(void* parg) int nErrorWait = 10; int nRetryWait = 10; + // IRC server blocks TOR users if (fUseProxy && addrProxy.port == htons(9050)) return; @@ -237,14 +238,14 @@ void ThreadIRCSeed(void* parg) { // index 7 is limited to 16 characters // could get full length name at index 10, but would be different from join messages - strcpy(pszName, vWords[7].c_str()); + strlcpy(pszName, vWords[7].c_str(), sizeof(pszName)); printf("IRC got who\n"); } if (vWords[1] == "JOIN" && vWords[0].size() > 1) { // :username!username@50000007.F000000B.90000002.IP JOIN :#channelname - strcpy(pszName, vWords[0].c_str() + 1); + strlcpy(pszName, vWords[0].c_str() + 1, sizeof(pszName)); if (strchr(pszName, '!')) *strchr(pszName, '!') = '\0'; printf("IRC got join\n"); diff --git a/irc.h b/irc.h index 1dc348a8..91c3ffe6 100644 --- a/irc.h +++ b/irc.h @@ -1,11 +1,6 @@ // Copyright (c) 2009 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. - -#ifndef __WXMSW__ -#define closesocket(s) close(s) -typedef u_int SOCKET; -#endif extern bool RecvLine(SOCKET hSocket, string& strLine); extern void ThreadIRCSeed(void* parg); diff --git a/main.cpp b/main.cpp index 15626777..ade3b51c 100644 --- a/main.cpp +++ b/main.cpp @@ -42,7 +42,6 @@ map > mapPubKeys; CCriticalSection cs_mapKeys; CKey keyUser; -string strSetDataDir; int nDropMessagesTest = 0; // Settings @@ -1361,52 +1360,17 @@ bool ScanMessageStart(Stream& s) } } -string GetAppDir() -{ - string strDir; - if (!strSetDataDir.empty()) - { - strDir = strSetDataDir; - } - else if (getenv("APPDATA")) - { - strDir = strprintf("%s\\Bitcoin", getenv("APPDATA")); - } - else if (getenv("USERPROFILE")) - { - string strAppData = strprintf("%s\\Application Data", getenv("USERPROFILE")); - static bool fMkdirDone; - if (!fMkdirDone) - { - fMkdirDone = true; - _mkdir(strAppData.c_str()); - } - strDir = strprintf("%s\\Bitcoin", strAppData.c_str()); - } - else - { - return "."; - } - static bool fMkdirDone; - if (!fMkdirDone) - { - fMkdirDone = true; - _mkdir(strDir.c_str()); - } - return strDir; -} - bool CheckDiskSpace(int64 nAdditionalBytes) { wxLongLong nFreeBytesAvailable = 0; - if (!wxGetDiskSpace(wxStandardPaths::Get().GetDataDir(), NULL, &nFreeBytesAvailable)) + if (!wxGetDiskSpace(GetDataDir(), NULL, &nFreeBytesAvailable)) { printf("ERROR: wxGetDiskSpace() failed\n"); return true; } // Check for 15MB because database could create another 10MB log file at any time - if (nFreeBytesAvailable < (int64)15000000 + nAdditionalBytes) + if (nFreeBytesAvailable.GetValue() < (int64)15000000 + nAdditionalBytes) { fShutdown = true; wxMessageBox("Warning: Your disk space is low ", "Bitcoin", wxICON_EXCLAMATION); @@ -1420,7 +1384,7 @@ FILE* OpenBlockFile(unsigned int nFile, unsigned int nBlockPos, const char* pszM { if (nFile == -1) return NULL; - FILE* file = fopen(strprintf("%s\\blk%04d.dat", GetAppDir().c_str(), nFile).c_str(), pszMode); + FILE* file = fopen(strprintf("%s\\blk%04d.dat", GetDataDir().c_str(), nFile).c_str(), pszMode); if (!file) return NULL; if (nBlockPos != 0 && !strchr(pszMode, 'a') && !strchr(pszMode, 'w')) @@ -1719,7 +1683,7 @@ bool ProcessMessages(CNode* pfrom) if (strstr(e.what(), "CDataStream::read() : end of data")) { // Allow exceptions from underlength message on vRecv - LogException(&e, "ProcessMessage()"); + printf("ProcessMessage(%s, %d bytes) : Exception '%s' caught, normally caused by a message being shorter than its stated length\n", strCommand.c_str(), nMessageSize, e.what()); } else PrintException(&e, "ProcessMessage()"); @@ -2512,7 +2476,7 @@ bool BitcoinMiner() int64 GetBalance() { - int64 nStart = PerformanceCounter(); + int64 nStart = GetTimeMillis(); int64 nTotal = 0; CRITICAL_BLOCK(cs_mapWallet) @@ -2522,11 +2486,11 @@ int64 GetBalance() CWalletTx* pcoin = &(*it).second; if (!pcoin->IsFinal() || pcoin->fSpent) continue; - nTotal += pcoin->GetCredit(); + nTotal += pcoin->GetCredit(true); } } - ///printf(" GetBalance() time = %15"PRI64d"\n", PerformanceCounter() - nStart); + //printf("GetBalance() %"PRI64d"ms\n", GetTimeMillis() - nStart); return nTotal; } diff --git a/main.h b/main.h index fcfd33d1..a0258cf2 100644 --- a/main.h +++ b/main.h @@ -34,7 +34,6 @@ extern int nBestHeight; extern uint256 hashBestChain; extern CBlockIndex* pindexBest; extern unsigned int nTransactionsUpdated; -extern string strSetDataDir; extern int nDropMessagesTest; // Settings @@ -50,7 +49,6 @@ extern int nLimitProcessors; -string GetAppDir(); bool CheckDiskSpace(int64 nAdditionalBytes=0); FILE* OpenBlockFile(unsigned int nFile, unsigned int nBlockPos, const char* pszMode="rb"); FILE* AppendBlockFile(unsigned int& nFileRet); @@ -405,10 +403,10 @@ public: { // Time based nLockTime implemented in 0.1.6, // do not use time based until most 0.1.5 nodes have upgraded. - if (nBlockTime == 0) - nBlockTime = GetAdjustedTime(); if (nLockTime == 0) return true; + if (nBlockTime == 0) + nBlockTime = GetAdjustedTime(); if (nLockTime < (nLockTime < 500000000 ? nBestHeight : nBlockTime)) return true; foreach(const CTxIn& txin, vin) @@ -627,6 +625,8 @@ public: // memory only mutable bool fMerkleVerified; + mutable bool fGetCreditCached; + mutable int64 nGetCreditCached; CMerkleTx() @@ -644,14 +644,22 @@ public: hashBlock = 0; nIndex = -1; fMerkleVerified = false; + fGetCreditCached = false; + nGetCreditCached = 0; } - int64 GetCredit() const + int64 GetCredit(bool fUseCache=false) const { // Must wait until coinbase is safely deep enough in the chain before valuing it if (IsCoinBase() && GetBlocksToMaturity() > 0) return 0; - return CTransaction::GetCredit(); + + // GetBalance can assume transactions in mapWallet won't change + if (fUseCache && fGetCreditCached) + return nGetCreditCached; + nGetCreditCached = CTransaction::GetCredit(); + fGetCreditCached = true; + return nGetCreditCached; } IMPLEMENT_SERIALIZE diff --git a/net.cpp b/net.cpp index 22b84f9d..8ccf48b8 100644 --- a/net.cpp +++ b/net.cpp @@ -21,8 +21,7 @@ bool OpenNetworkConnection(const CAddress& addrConnect); bool fClient = false; uint64 nLocalServices = (fClient ? 0 : NODE_NETWORK); CAddress addrLocalHost(0, DEFAULT_PORT, nLocalServices); -CNode nodeLocalHost(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); -CNode* pnodeLocalHost = &nodeLocalHost; +CNode* pnodeLocalHost = NULL; uint64 nLocalHostNonce = 0; bool fShutdown = false; array vnThreadsRunning; @@ -129,7 +128,7 @@ bool GetMyExternalIP2(const CAddress& addrConnect, const char* pszGet, const cha strLine = wxString(strLine).Trim(); CAddress addr(strLine.c_str()); printf("GetMyExternalIP() received [%s] %s\n", strLine.c_str(), addr.ToString().c_str()); - if (addr.ip == 0 || !addr.IsRoutable()) + if (addr.ip == 0 || addr.ip == INADDR_NONE || !addr.IsRoutable()) return false; ipRet = addr.ip; return true; @@ -740,10 +739,29 @@ void ThreadOpenConnections2(void* parg) printf("ThreadOpenConnections started\n"); // Connect to one specified address - while (mapArgs.count("/connect")) + while (mapArgs.count("-connect")) { - OpenNetworkConnection(CAddress(mapArgs["/connect"].c_str())); - Sleep(10000); + OpenNetworkConnection(CAddress(mapArgs["-connect"])); + for (int i = 0; i < 10; i++) + { + Sleep(1000); + CheckForShutdown(1); + } + } + + // Connect to manually added nodes first + if (mapArgs.count("-addnode")) + { + foreach(string strAddr, mapMultiArgs["-addnode"]) + { + CAddress addr(strAddr, NODE_NETWORK); + if (addr.IsValid()) + { + OpenNetworkConnection(addr); + Sleep(1000); + CheckForShutdown(1); + } + } } // Initiate network connections @@ -967,6 +985,8 @@ void ThreadMessageHandler2(void* parg) bool StartNode(string& strError) { + if (pnodeLocalHost == NULL) + pnodeLocalHost = new CNode(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); strError = ""; // Sockets startup @@ -1031,7 +1051,7 @@ bool StartNode(string& strError) printf("%s\n", strError.c_str()); return false; } - printf("bound to addrLocalHost = %s\n\n", addrLocalHost.ToString().c_str()); + printf("bound to addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); // Listen for incoming connections if (listen(hListenSocket, SOMAXCONN) == SOCKET_ERROR) diff --git a/net.h b/net.h index 2eece2c0..7716426f 100644 --- a/net.h +++ b/net.h @@ -1,12 +1,6 @@ // Copyright (c) 2009 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. - -#ifndef __WXMSW__ -#define closesocket(s) close(s) -#define INVALID_SOCKET (SOCKET)(~0) -typedef u_int SOCKET; -#endif class CMessageHeader; class CAddress; @@ -148,61 +142,73 @@ public: CAddress() { - nServices = 0; - memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); - ip = 0; - port = DEFAULT_PORT; - nTime = GetAdjustedTime(); - nLastFailed = 0; + Init(); } - CAddress(unsigned int ipIn, unsigned short portIn=DEFAULT_PORT, uint64 nServicesIn=0) + CAddress(unsigned int ipIn, unsigned short portIn=DEFAULT_PORT, uint64 nServicesIn=NODE_NETWORK) { - nServices = nServicesIn; - memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); + Init(); ip = ipIn; port = portIn; - nTime = GetAdjustedTime(); - nLastFailed = 0; + nServices = nServicesIn; } - explicit CAddress(const struct sockaddr_in& sockaddr, uint64 nServicesIn=0) + explicit CAddress(const struct sockaddr_in& sockaddr, uint64 nServicesIn=NODE_NETWORK) { - nServices = nServicesIn; - memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); + Init(); ip = sockaddr.sin_addr.s_addr; port = sockaddr.sin_port; - nTime = GetAdjustedTime(); - nLastFailed = 0; + nServices = nServicesIn; } - explicit CAddress(const char* pszIn, uint64 nServicesIn=0) + explicit CAddress(const char* pszIn, uint64 nServicesIn=NODE_NETWORK) { + Init(); + SetAddress(pszIn); nServices = nServicesIn; + } + + explicit CAddress(string strIn, uint64 nServicesIn=NODE_NETWORK) + { + Init(); + SetAddress(strIn.c_str()); + nServices = nServicesIn; + } + + void Init() + { + nServices = NODE_NETWORK; memcpy(pchReserved, pchIPv4, sizeof(pchReserved)); ip = INADDR_NONE; port = DEFAULT_PORT; nTime = GetAdjustedTime(); nLastFailed = 0; + } + bool SetAddress(const char* pszIn) + { + ip = INADDR_NONE; + port = DEFAULT_PORT; char psz[100]; - if (strlen(pszIn) > ARRAYLEN(psz)-1) - return; - strcpy(psz, pszIn); + strlcpy(psz, pszIn, sizeof(psz)); unsigned int a=0, b=0, c=0, d=0, e=0; if (sscanf(psz, "%u.%u.%u.%u:%u", &a, &b, &c, &d, &e) < 4) - return; + return false; char* pszPort = strchr(psz, ':'); if (pszPort) { *pszPort++ = '\0'; port = htons(atoi(pszPort)); - if (atoi(pszPort) > USHRT_MAX) + if (atoi(pszPort) < 0 || atoi(pszPort) > USHRT_MAX) port = htons(USHRT_MAX); - if (atoi(pszPort) < 0) - port = htons(0); } ip = inet_addr(psz); + return IsValid(); + } + + bool SetAddress(string strIn) + { + return SetAddress(strIn.c_str()); } IMPLEMENT_SERIALIZE @@ -274,7 +280,17 @@ public: bool IsRoutable() const { - return !(GetByte(3) == 10 || (GetByte(3) == 192 && GetByte(2) == 168) || GetByte(3) == 127 || GetByte(3) == 0); + return !(GetByte(3) == 10 || + (GetByte(3) == 192 && GetByte(2) == 168) || + GetByte(3) == 127 || + GetByte(3) == 0 || + ip == 0 || + ip == INADDR_NONE); + } + + bool IsValid() const + { + return (ip != 0 && ip != INADDR_NONE && port != htons(USHRT_MAX)); } unsigned char GetByte(int n) const diff --git a/ui.cpp b/ui.cpp index ce287e39..9185b815 100644 --- a/ui.cpp +++ b/ui.cpp @@ -25,7 +25,6 @@ DEFINE_EVENT_TYPE(wxEVT_TABLEDELETED) CMainFrame* pframeMain = NULL; CMyTaskBarIcon* ptaskbaricon = NULL; map mapAddressBook; -map mapArgs; bool fRandSendTest = false; void RandSend(); extern int g_isPainting; @@ -283,7 +282,6 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) fRefreshListCtrl = false; fRefreshListCtrlRunning = false; fOnSetFocusAddress = false; - pindexBestLast = NULL; m_choiceFilter->SetSelection(0); m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); @@ -507,6 +505,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) string strStatus = FormatTxStatus(wtx); map mapValue = wtx.mapValue; wtx.nLinesDisplayed = 1; + nListViewUpdated++; // Filter if (wtx.IsCoinBase()) @@ -712,48 +711,6 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) return true; } -void CMainFrame::RefreshStatus() -{ - static int nLastTop; - int nTop = max((int)m_listCtrl->GetTopItem(), 0); - if (nTop == nLastTop && pindexBestLast == pindexBest) - return; - - TRY_CRITICAL_BLOCK(cs_mapWallet) - { - int nStart = nTop; - int nEnd = min(nStart + 100, m_listCtrl->GetItemCount()); - if (pindexBestLast == pindexBest) - { - if (nStart >= nLastTop && nStart < nLastTop + 100) - nStart = nLastTop + 100; - if (nEnd >= nLastTop && nEnd < nLastTop + 100) - nEnd = nLastTop; - } - nLastTop = nTop; - pindexBestLast = pindexBest; - - for (int nIndex = nStart; nIndex < min(nEnd, m_listCtrl->GetItemCount()); nIndex++) - { - uint256 hash((string)GetItemText(m_listCtrl, nIndex, 1)); - map::iterator mi = mapWallet.find(hash); - if (mi == mapWallet.end()) - { - printf("CMainFrame::RefreshStatus() : tx not found in mapWallet\n"); - continue; - } - CWalletTx& wtx = (*mi).second; - if (wtx.IsCoinBase() || wtx.GetTxTime() != wtx.nTimeDisplayed) - { - if (!InsertTransaction(wtx, false, nIndex)) - m_listCtrl->DeleteItem(nIndex--); - } - else - m_listCtrl->SetItem(nIndex, 2, FormatTxStatus(wtx)); - } - } -} - void CMainFrame::RefreshListCtrl() { fRefreshListCtrl = true; @@ -832,21 +789,104 @@ void CMainFrame::OnIdle(wxIdleEvent& event) } } +void CMainFrame::RefreshStatusColumn() +{ + static int nLastTop; + static CBlockIndex* pindexLastBest; + static unsigned int nLastRefreshed; + + int nTop = max((int)m_listCtrl->GetTopItem(), 0); + if (nTop == nLastTop && pindexLastBest == pindexBest) + return; + + TRY_CRITICAL_BLOCK(cs_mapWallet) + { + int nStart = nTop; + int nEnd = min(nStart + 100, m_listCtrl->GetItemCount()); + + if (pindexLastBest == pindexBest && nLastRefreshed == nListViewUpdated) + { + // If no updates, only need to do the part that moved onto the screen + if (nStart >= nLastTop && nStart < nLastTop + 100) + nStart = nLastTop + 100; + if (nEnd >= nLastTop && nEnd < nLastTop + 100) + nEnd = nLastTop; + } + nLastTop = nTop; + pindexLastBest = pindexBest; + nLastRefreshed = nListViewUpdated; + + for (int nIndex = nStart; nIndex < min(nEnd, m_listCtrl->GetItemCount()); nIndex++) + { + uint256 hash((string)GetItemText(m_listCtrl, nIndex, 1)); + map::iterator mi = mapWallet.find(hash); + if (mi == mapWallet.end()) + { + printf("CMainFrame::RefreshStatusColumn() : tx not found in mapWallet\n"); + continue; + } + CWalletTx& wtx = (*mi).second; + if (wtx.IsCoinBase() || wtx.GetTxTime() != wtx.nTimeDisplayed) + { + if (!InsertTransaction(wtx, false, nIndex)) + m_listCtrl->DeleteItem(nIndex--); + } + else + m_listCtrl->SetItem(nIndex, 2, FormatTxStatus(wtx)); + } + } +} + void CMainFrame::OnPaint(wxPaintEvent& event) { event.Skip(); } -void DelayedRepaint(void* parg) + +unsigned int nNeedRepaint = 0; +unsigned int nLastRepaint = 0; +int64 nLastRepaintTime = 0; +int64 nRepaintInterval = 500; + +void ThreadDelayedRepaint(void* parg) { - static bool fOneThread; - if (fOneThread) - return; - fOneThread = true; - Sleep(1000); - printf("DelayedRepaint()\n"); - MainFrameRepaint(); - fOneThread = false; + while (!fShutdown) + { + if (nLastRepaint != nNeedRepaint && GetTimeMillis() - nLastRepaintTime >= nRepaintInterval) + { + nLastRepaint = nNeedRepaint; + if (pframeMain) + { + printf("DelayedRepaint\n"); + wxPaintEvent event; + pframeMain->Refresh(); + pframeMain->AddPendingEvent(event); + } + } + Sleep(nRepaintInterval); + } +} + +void MainFrameRepaint() +{ + // This is called by network code that shouldn't access pframeMain + // directly because it could still be running after the UI is closed. + if (pframeMain) + { + // Don't repaint too often + static int64 nLastRepaintRequest; + if (GetTimeMillis() - nLastRepaintRequest < 100) + { + nNeedRepaint++; + return; + } + nLastRepaintRequest = GetTimeMillis(); + + printf("MainFrameRepaint\n"); + wxPaintEvent event; + pframeMain->Refresh(); + pframeMain->AddPendingEvent(event); + } } void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) @@ -854,43 +894,54 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) if (ptaskbaricon) ptaskbaricon->UpdateTooltip(); - // Update listctrl contents - if (!vWalletUpdated.empty()) + // + // Slower stuff + // + static int nTransactionCount; + bool fPaintedBalance = false; + if (GetTimeMillis() - nLastRepaintTime >= nRepaintInterval) { - TRY_CRITICAL_BLOCK(cs_mapWallet) + nLastRepaint = nNeedRepaint; + nLastRepaintTime = GetTimeMillis(); + + // Update listctrl contents + if (!vWalletUpdated.empty()) { - bool fInserted = false; - foreach(uint256 hash, vWalletUpdated) + TRY_CRITICAL_BLOCK(cs_mapWallet) { - map::iterator mi = mapWallet.find(hash); - if (mi != mapWallet.end()) - fInserted |= InsertTransaction((*mi).second, false); + bool fInserted = false; + foreach(uint256 hash, vWalletUpdated) + { + map::iterator mi = mapWallet.find(hash); + if (mi != mapWallet.end()) + fInserted |= InsertTransaction((*mi).second, false); + } + vWalletUpdated.clear(); + if (fInserted) + m_listCtrl->ScrollList(0, INT_MAX); } - vWalletUpdated.clear(); - if (fInserted) - m_listCtrl->ScrollList(0, INT_MAX); } - } - - // Update status column of visible items only - RefreshStatus(); - - // Balance total - bool fRefreshed = false; - static int nTransactionCount; - TRY_CRITICAL_BLOCK(cs_mapWallet) - { - fRefreshed = true; - m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); - // Count hidden and multi-line transactions - nTransactionCount = 0; - for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + // Balance total + TRY_CRITICAL_BLOCK(cs_mapWallet) { - CWalletTx& wtx = (*it).second; - nTransactionCount += wtx.nLinesDisplayed; + fPaintedBalance = true; + m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); + + // Count hidden and multi-line transactions + nTransactionCount = 0; + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + CWalletTx& wtx = (*it).second; + nTransactionCount += wtx.nLinesDisplayed; + } } } + if (!vWalletUpdated.empty() || !fPaintedBalance) + nNeedRepaint++; + + // Update status column of visible items only + RefreshStatusColumn(); // Update status bar string strGen = ""; @@ -903,13 +954,10 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, nTransactionCount); m_statusBar->SetStatusText(strStatus, 2); - // mapWallet was locked, try again later - if (!vWalletUpdated.empty() || !fRefreshed) - _beginthread(DelayedRepaint, 0, NULL); - m_listCtrl->OnPaint(event); } + void CrossThreadCall(wxCommandEvent& event) { if (pframeMain) @@ -994,13 +1042,6 @@ void CMainFrame::OnMenuHelpAbout(wxCommandEvent& event) void CMainFrame::OnButtonSend(wxCommandEvent& event) { - /// debug test - if (fRandSendTest) - { - RandSend(); - return; - } - // Toolbar: Send CSendDialog dialog(this); dialog.ShowModal(); @@ -1684,8 +1725,8 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event) else { // Parse IP address - CAddress addr(strAddress.c_str()); - if (addr.ip == 0) + CAddress addr(strAddress); + if (!addr.IsValid()) { wxMessageBox("Invalid address ", "Send Coins"); return; @@ -1818,14 +1859,6 @@ void CSendingDialog::OnPaint(wxPaintEvent& event) wxMessageBox("Transfer cancelled ", "Sending...", wxOK, this); } event.Skip(); - - /// debug test - if (fRandSendTest && fWorkDone && fSuccess) - { - Close(); - Sleep(1000); - RandSend(); - } } @@ -3305,27 +3338,6 @@ bool CMyApp::OnInit() return false; } -map ParseParameters(int argc, char* argv[]) -{ - map mapArgs; - for (int i = 0; i < argc; i++) - { - char psz[10000]; - strcpy(psz, argv[i]); - char* pszValue = ""; - if (strchr(psz, '=')) - { - pszValue = strchr(psz, '='); - *pszValue++ = '\0'; - } - strlwr(psz); - if (psz[0] == '-') - psz[0] = '/'; - mapArgs[psz] = pszValue; - } - return mapArgs; -} - bool CMyApp::OnInit2() { #ifdef _MSC_VER @@ -3337,10 +3349,27 @@ bool CMyApp::OnInit2() // Disable malfunctioning wxWidgets debug assertion g_isPainting = 10000; #endif - - //// debug print - printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); - printf("Bitcoin version %d, Windows version %08x\n", VERSION, GetVersion()); + wxImage::AddHandler(new wxPNGHandler); + SetAppName("Bitcoin"); + + ParseParameters(argc, argv); + if (mapArgs.count("-?") || mapArgs.count("--help")) + { + string strUsage = + "Usage: bitcoin [options]\t\t\t\t\t\t\n" + "Options:\n" + " -gen\t\t Generate coins\n" + " -gen=0\t\t Don't generate coins\n" + " -min\t\t Start minimized\n" + " -datadir=\t Specify data directory\n" + " -proxy=\t Connect through socks4 proxy,\n" + " \t\t e.g. -proxy=127.0.0.1:9050 to use TOR\n" + " -addnode=\t Add a node to connect to\n" + " -connect=\t Connect only to the specified node\n" + " -?\t\t This help message\n"; + wxMessageBox(strUsage, "Bitcoin", wxOK); + exit(0); + } // // Limit to single instance per user @@ -3382,31 +3411,31 @@ bool CMyApp::OnInit2() // // Parameters // - wxImage::AddHandler(new wxPNGHandler); - mapArgs = ParseParameters(argc, argv); - - if (mapArgs.count("/datadir")) - strSetDataDir = mapArgs["/datadir"]; + if (mapArgs.count("-datadir")) + strlcpy(pszSetDataDir, mapArgs["-datadir"].c_str(), sizeof(pszSetDataDir)); - if (mapArgs.count("/debug")) + if (mapArgs.count("-debug")) fDebug = true; - if (mapArgs.count("/printtodebugger")) + if (mapArgs.count("-printtodebugger")) fPrintToDebugger = true; - if (mapArgs.count("/dropmessages")) + printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); + printf("Bitcoin version %d, Windows version %08x\n", VERSION, GetVersion()); + + if (mapArgs.count("-dropmessages")) { - nDropMessagesTest = atoi(mapArgs["/dropmessages"]); + nDropMessagesTest = atoi(mapArgs["-dropmessages"]); if (nDropMessagesTest == 0) nDropMessagesTest = 20; } - if (mapArgs.count("/loadblockindextest")) + if (mapArgs.count("-loadblockindextest")) { CTxDB txdb("r"); txdb.LoadBlockIndex(); PrintBlockTree(); - ExitProcess(0); + exit(0); } // @@ -3417,22 +3446,22 @@ bool CMyApp::OnInit2() int64 nStart; printf("Loading addresses...\n"); - nStart = PerformanceCounter(); + nStart = GetTimeMillis(); if (!LoadAddresses()) strErrors += "Error loading addr.dat \n"; - printf(" addresses %15"PRI64d"\n", PerformanceCounter() - nStart); + printf(" addresses %15"PRI64d"ms\n", GetTimeMillis() - nStart); printf("Loading block index...\n"); - nStart = PerformanceCounter(); + nStart = GetTimeMillis(); if (!LoadBlockIndex()) strErrors += "Error loading blkindex.dat \n"; - printf(" block index %15"PRI64d"\n", PerformanceCounter() - nStart); + printf(" block index %15"PRI64d"ms\n", GetTimeMillis() - nStart); printf("Loading wallet...\n"); - nStart = PerformanceCounter(); + nStart = GetTimeMillis(); if (!LoadWallet(fFirstRun)) strErrors += "Error loading wallet.dat \n"; - printf(" wallet %15"PRI64d"\n", PerformanceCounter() - nStart); + printf(" wallet %15"PRI64d"ms\n", GetTimeMillis() - nStart); printf("Done loading\n"); @@ -3457,45 +3486,59 @@ bool CMyApp::OnInit2() // // Parameters // - if (mapArgs.count("/printblockindex") || mapArgs.count("/printblocktree")) + if (mapArgs.count("-printblockindex") || mapArgs.count("-printblocktree")) { PrintBlockTree(); OnExit(); return false; } - if (mapArgs.count("/proxy")) + if (mapArgs.count("-gen")) + { + if (mapArgs["-gen"].empty()) + fGenerateBitcoins = true; + else + fGenerateBitcoins = atoi(mapArgs["-gen"].c_str()); + } + + if (mapArgs.count("-proxy")) { fUseProxy = true; - addrProxy = CAddress(mapArgs["/proxy"].c_str()); - if (addrProxy.ip == INADDR_NONE) + addrProxy = CAddress(mapArgs["-proxy"]); + if (!addrProxy.IsValid()) { - wxMessageBox("Invalid /proxy address", "Bitcoin"); + wxMessageBox("Invalid -proxy address", "Bitcoin"); OnExit(); + return false; } CWalletDB walletdb; walletdb.WriteSetting("fUseProxy", fUseProxy); walletdb.WriteSetting("addrProxy", addrProxy); } - if (mapArgs.count("/gen")) + if (mapArgs.count("-addnode")) { - if (mapArgs["/gen"].empty()) - fGenerateBitcoins = true; - else - fGenerateBitcoins = atoi(mapArgs["/gen"].c_str()); + CAddrDB addrdb; + foreach(string strAddr, mapMultiArgs["-addnode"]) + { + CAddress addr(strAddr, NODE_NETWORK); + if (addr.IsValid()) + AddAddress(addrdb, addr); + } } // // Create the main frame window // pframeMain = new CMainFrame(NULL); - if (mapArgs.count("/min")) + if (mapArgs.count("-min")) pframeMain->Iconize(true); pframeMain->Show(true); // have to show first to get taskbar button to hide pframeMain->Show(!fMinimizeToTray || !pframeMain->IsIconized()); ptaskbaricon->Show(fMinimizeToTray); + _beginthread(ThreadDelayedRepaint, 0, NULL); + if (!CheckDiskSpace()) { OnExit(); @@ -3516,7 +3559,7 @@ bool CMyApp::OnInit2() // // Tests // - if (argc >= 2 && stricmp(argv[1], "/send") == 0) + if (argc >= 2 && stricmp(argv[1], "-send") == 0) { int64 nValue = 1; if (argc >= 3) @@ -3525,7 +3568,7 @@ bool CMyApp::OnInit2() string strAddress; if (argc >= 4) strAddress = argv[3]; - CAddress addr(strAddress.c_str()); + CAddress addr(strAddress); CWalletTx wtx; wtx.mapValue["to"] = strAddress; @@ -3538,15 +3581,6 @@ bool CMyApp::OnInit2() return false; } - if (mapArgs.count("/randsendtest")) - { - if (!mapArgs["/randsendtest"].empty()) - _beginthread(ThreadRandSendTest, 0, new string(mapArgs["/randsendtest"])); - else - fRandSendTest = true; - fDebug = true; - } - return true; } @@ -3610,19 +3644,6 @@ void CMyApp::OnFatalException() -void MainFrameRepaint() -{ - // This is called by network code that shouldn't access pframeMain - // directly because it could still be running after the UI is closed. - if (pframeMain) - { - printf("MainFrameRepaint()\n"); - wxPaintEvent event; - pframeMain->Refresh(); - pframeMain->AddPendingEvent(event); - } -} - typedef WINSHELLAPI BOOL WINAPI (*PSHGETSPECIALFOLDERPATHA)(HWND hwndOwner, LPSTR lpszPath, int nFolder, BOOL fCreate); @@ -3666,7 +3687,7 @@ string StartupShortcutPath() bool GetStartOnSystemStartup() { - return FileExists(StartupShortcutPath().c_str()); + return wxFileExists(StartupShortcutPath()); } void SetStartOnSystemStartup(bool fAutoStart) @@ -3727,79 +3748,3 @@ void SetStartOnSystemStartup(bool fAutoStart) - -// randsendtest to bitcoin address -void ThreadRandSendTest(void* parg) -{ - string strAddress = *(string*)parg; - uint160 hash160; - if (!AddressToHash160(strAddress, hash160)) - { - wxMessageBox(strprintf("ThreadRandSendTest: Bitcoin address '%s' not valid ", strAddress.c_str())); - return; - } - - while (!fShutdown) - { - Sleep(GetRand(30) * 1000 + 100); - - // Message - CWalletTx wtx; - wtx.mapValue["to"] = strAddress; - wtx.mapValue["from"] = addrLocalHost.ToString(); - static int nRep; - wtx.mapValue["message"] = strprintf("randsendtest %d\n", ++nRep); - - // Value - int64 nValue = (GetRand(9) + 1) * 100 * CENT; - if (GetBalance() < nValue) - { - wxMessageBox("Out of money "); - while (GetBalance() < 1000) - Sleep(1000); - } - nValue += (nRep % 100) * CENT; - - // Send to bitcoin address - CScript scriptPubKey; - scriptPubKey << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; - - if (fShutdown) - return; - if (!SendMoney(scriptPubKey, nValue, wtx)) - return; - } -} - - -// randsendtest to any connected node -void RandSend() -{ - while (vNodes.empty()) - Sleep(1000); - CAddress addr; - CRITICAL_BLOCK(cs_vNodes) - addr = vNodes[GetRand(vNodes.size())]->addr; - - // Message - CWalletTx wtx; - wtx.mapValue["to"] = addr.ToString(); - wtx.mapValue["from"] = addrLocalHost.ToString(); - static int nRep; - wtx.mapValue["message"] = strprintf("randsendtest %d\n", ++nRep); - - // Value - int64 nValue = (GetRand(999) + 1) * CENT; - if (GetBalance() < nValue) - { - wxMessageBox("Out of money "); - return; - } - - // Send to IP address - if (fShutdown) - return; - CSendingDialog* pdialog = new CSendingDialog(pframeMain, addr, nValue, wtx); - if (!pdialog->Show()) - wxMessageBox("ShowModal Failed "); -} diff --git a/ui.h b/ui.h index 9fc7e0eb..47839e81 100644 --- a/ui.h +++ b/ui.h @@ -83,15 +83,14 @@ public: bool fRefreshListCtrl; bool fRefreshListCtrlRunning; bool fOnSetFocusAddress; - CBlockIndex* pindexBestLast; - set setUnmaturedDisplayed; + unsigned int nListViewUpdated; void OnCrossThreadCall(wxCommandEvent& event); void InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5); bool DeleteLine(uint256 hashKey); bool InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); void RefreshListCtrl(); - void RefreshStatus(); + void RefreshStatusColumn(); }; diff --git a/util.cpp b/util.cpp index ef950920..23b59f11 100644 --- a/util.cpp +++ b/util.cpp @@ -5,9 +5,13 @@ #include "headers.h" +map mapArgs; +map > mapMultiArgs; bool fDebug = false; bool fPrintToDebugger = false; bool fPrintToConsole = false; +char pszSetDataDir[MAX_PATH] = ""; + @@ -68,6 +72,8 @@ void RandAddSeed() void RandAddSeedPerfmon() { +#ifdef __WXMSW__ + // Don't need this on Linux, OpenSSL automatically uses /dev/urandom // This can take up to 2 seconds, so only do it every 10 minutes static int64 nLastPerfmon; if (GetTime() < nLastPerfmon + 10 * 60) @@ -95,6 +101,7 @@ void RandAddSeedPerfmon() strftime(pszTime, sizeof(pszTime), "%x %H:%M:%S", ptmTime); printf("%s RandAddSeed() %d bytes\n", pszTime, nSize); } +#endif } @@ -304,6 +311,32 @@ vector ParseHex(const std::string& str) } +void ParseParameters(int argc, char* argv[]) +{ + mapArgs.clear(); + mapMultiArgs.clear(); + for (int i = 0; i < argc; i++) + { + char psz[10000]; + strlcpy(psz, argv[i], sizeof(psz)); + char* pszValue = ""; + if (strchr(psz, '=')) + { + pszValue = strchr(psz, '='); + *pszValue++ = '\0'; + } + strlwr(psz); + #ifdef __WXMSW__ + if (psz[0] == '/') + psz[0] = '-'; + #endif + mapArgs[psz] = pszValue; + mapMultiArgs[psz].push_back(pszValue); + } +} + + + @@ -346,15 +379,6 @@ void PrintException(std::exception* pex, const char* pszThread) -bool FileExists(const char* psz) -{ -#ifdef WIN32 - return GetFileAttributes(psz) != -1; -#else - return access(psz, 0) != -1; -#endif -} - int GetFilesize(FILE* file) { int nSavePos = ftell(file); @@ -365,6 +389,46 @@ int GetFilesize(FILE* file) return nFilesize; } +void GetDataDir(char* pszDir) +{ + // pszDir must be at least MAX_PATH length. + if (pszSetDataDir[0] != 0) + { + strlcpy(pszDir, pszSetDataDir, MAX_PATH); + static bool fMkdirDone; + if (!fMkdirDone) + { + fMkdirDone = true; + _mkdir(pszDir); + } + } + else + { + // This can be called during exceptions by printf, so we cache the + // value so we don't have to do memory allocations after that. + // wxStandardPaths::GetUserDataDir + // Return the directory for the user-dependent application data files: + // Unix: ~/.appname + // Windows: C:\Documents and Settings\username\Application Data\appname + // Mac: ~/Library/Application Support/appname + static char pszCachedDir[MAX_PATH]; + if (pszCachedDir[0] == 0) + { + strlcpy(pszCachedDir, wxStandardPaths::Get().GetUserDataDir().c_str(), sizeof(pszCachedDir)); + _mkdir(pszCachedDir); + } + strlcpy(pszDir, pszCachedDir, MAX_PATH); + } + +} + +string GetDataDir() +{ + char pszDir[MAX_PATH]; + GetDataDir(pszDir); + return pszDir; +} + diff --git a/util.h b/util.h index 5d187760..822a049c 100644 --- a/util.h +++ b/util.h @@ -54,16 +54,23 @@ inline T& REF(const T& val) return (T&)val; } +#ifndef __WXMSW__ +#define closesocket(s) close(s) +#define INVALID_SOCKET (SOCKET)(~0) +typedef u_int SOCKET; +#endif +extern map mapArgs; +extern map > mapMultiArgs; extern bool fDebug; extern bool fPrintToDebugger; extern bool fPrintToConsole; -extern map mapArgs; +extern char pszSetDataDir[MAX_PATH]; void RandAddSeed(); void RandAddSeedPerfmon(); @@ -77,8 +84,10 @@ string FormatMoney(int64 n, bool fPlus=false); bool ParseMoney(const char* pszIn, int64& nRet); vector ParseHex(const char* psz); vector ParseHex(const std::string& str); -bool FileExists(const char* psz); +void ParseParameters(int argc, char* argv[]); int GetFilesize(FILE* file); +void GetDataDir(char* pszDirRet); +string GetDataDir(); uint64 GetRand(uint64 nMax); int64 GetTime(); int64 GetAdjustedTime(); @@ -172,9 +181,14 @@ inline int OutputDebugStringF(const char* pszFormat, ...) if (!fPrintToConsole) { // print to debug.log - FILE* fileout = fopen("debug.log", "a"); + char pszFile[MAX_PATH+100]; + GetDataDir(pszFile); + strlcat(pszFile, "\\debug.log", sizeof(pszFile)); + FILE* fileout = fopen(pszFile, "a"); if (fileout) { + //// Debug print useful for profiling + //fprintf(fileout, " %"PRI64d" ", wxGetLocalTimeMillis().GetValue()); va_list arg_ptr; va_start(arg_ptr, pszFormat); ret = vfprintf(fileout, pszFormat, arg_ptr); @@ -321,22 +335,25 @@ inline void PrintHex(vector vch, const char* pszFormat="%s", bool { printf(pszFormat, HexStr(vch, fSpaces).c_str()); } - inline int64 PerformanceCounter() { - int64 nCounter = 0; + int64 nCounter = 0; #ifdef __WXMSW__ - QueryPerformanceCounter((LARGE_INTEGER*)&nCounter); + QueryPerformanceCounter((LARGE_INTEGER*)&nCounter); #else - // this could be changed to reading /dev/urandom - timeval t; - gettimeofday(&t, NULL); - nCounter += t.tv_sec * 1000000 + t.tv_usec; + timeval t; + gettimeofday(&t, NULL); + nCounter = t.tv_sec * 1000000 + t.tv_usec; #endif return nCounter; } +inline int64 GetTimeMillis() +{ + return wxGetLocalTimeMillis().GetValue(); +} + #ifndef __WXMSW__ inline void Sleep(unsigned int nMilliseconds) { @@ -354,8 +371,10 @@ inline void Sleep(unsigned int nMilliseconds) inline void heapchk() { +#ifdef __WXMSW__ if (_heapchk() != _HEAPOK) DebugBreak(); +#endif } // Randomize the stack to help protect against buffer overrun exploits From b7362c07ae42ac4282361ee95b82424d502f1b82 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 4 Nov 2009 04:58:46 +0000 Subject: [PATCH 020/133] renamed build.txt to build-msw.txt git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@28 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-msw.txt | 91 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 91 insertions(+) create mode 100644 build-msw.txt diff --git a/build-msw.txt b/build-msw.txt new file mode 100644 index 00000000..eb6348ba --- /dev/null +++ b/build-msw.txt @@ -0,0 +1,91 @@ +Bitcoin v0.2.0 BETA + +Copyright (c) 2009 Satoshi Nakamoto +Distributed under the MIT/X11 software license, see the accompanying +file license.txt or http://www.opensource.org/licenses/mit-license.php. +This product includes software developed by the OpenSSL Project for use in +the OpenSSL Toolkit (http://www.openssl.org/). This product includes +cryptographic software written by Eric Young (eay@cryptsoft.com). + + + WINDOWS BUILD NOTES + + +Compilers Supported +------------------- +MinGW GCC +Microsoft Visual C++ 6.0 SP6 + + +Dependencies +------------ +Libraries you need to obtain separately to build: + + default path download +wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ + or prebuilt: http://wxpack.sourceforge.net +OpenSSL \openssl http://www.openssl.org/source/ +Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html +Boost \boost http://www.boost.org/users/download/ + +Their licenses: +wxWidgets LGPL 2.1 with very liberal exceptions +OpenSSL Old BSD license with the problematic advertising requirement +Berkeley DB New BSD license with additional requirement that linked software must be free open source +Boost MIT-like license + +Versions used in this release: +MinGW GCC 3.4.5 +wxWidgets 2.8.9 +OpenSSL 0.9.8k +Berkeley DB 4.7.25.NC +Boost 1.34.1 + + +Notes +----- +The UI layout is edited with wxFormBuilder. Open the project file +uiproject.fbp. It generates uibase.cpp and uibase.h, which define base +classes that do the rote work of constructing all the UI elements. + +The release is built with GCC and then "strip bitcoin.exe" to strip the debug +symbols, which reduces the executable size by about 90%. + + +OpenSSL +------- +Bitcoin does not use any encryption. If you want to do a no-everything +build of OpenSSL to exclude encryption routines, a few patches are required. +(instructions for OpenSSL v0.9.8k) + +Edit engines\e_gmp.c and engines\e_capi.c and add this #ifndef around +the openssl/rsa.h include: + #ifndef OPENSSL_NO_RSA + #include + #endif + +Edit ms\mingw32.bat and replace the Configure line's parameters with this +no-everything list. You have to put this in the batch file because batch +files can't take more than nine command line parameters. + perl Configure mingw threads no-rc2 no-rc4 no-rc5 no-idea no-des no-bf no-cast no-aes no-camellia no-seed no-rsa no-dh + +Also REM out the following line in ms\mingw32.bat after the mingw32-make +line. The build fails after it's already finished building libeay32, which +is all we care about, but the failure aborts the script before it runs +dllwrap to generate libeay32.dll. + REM if errorlevel 1 goto end + +Build + cd \openssl + ms\mingw32.bat + +If you want to use it with MSVC, generate the .lib file + lib /machine:i386 /def:ms\libeay32.def /out:out\libeay32.lib + + +Berkeley DB +----------- +Using MinGW and MSYS: +cd \db\build_unix +sh ../dist/configure --enable-mingw --enable-cxx +make From e4c05d31778a85014b2a52e2f20753b38dfbf950 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 5 Nov 2009 04:41:36 +0000 Subject: [PATCH 021/133] unix build merged in, bitmap resources from xpm instead of rc, better addr relay, better selection of addrs by time last seen for faster connect git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@32 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-unix.txt | 73 ++++++++++ build.txt | 88 ------------ db.cpp | 10 +- headers.h | 38 ++++-- irc.cpp | 16 +-- irc.h | 4 +- main.cpp | 68 +++++---- main.h | 8 ++ makefile.unix | 86 ++++++++++++ net.cpp | 177 ++++++++++++------------ net.h | 3 +- ui.cpp | 103 ++++++++------ uibase.cpp | 10 +- uiproject.fbp | 10 +- util.cpp | 8 +- util.h | 40 ++++-- xpm/addressbook16.xpm | 278 +++++++++++++++++++++++++++++++++++++ xpm/addressbook20.xpm | 282 ++++++++++++++++++++++++++++++++++++++ xpm/bitcoin.xpm | 304 +++++++++++++++++++++++++++++++++++++++++ xpm/check.xpm | 41 ++++++ xpm/send16.xpm | 278 +++++++++++++++++++++++++++++++++++++ xpm/send16noshadow.xpm | 278 +++++++++++++++++++++++++++++++++++++ xpm/send20.xpm | 282 ++++++++++++++++++++++++++++++++++++++ 23 files changed, 2184 insertions(+), 301 deletions(-) create mode 100644 build-unix.txt delete mode 100644 build.txt create mode 100644 makefile.unix create mode 100644 xpm/addressbook16.xpm create mode 100644 xpm/addressbook20.xpm create mode 100644 xpm/bitcoin.xpm create mode 100644 xpm/check.xpm create mode 100644 xpm/send16.xpm create mode 100644 xpm/send16noshadow.xpm create mode 100644 xpm/send20.xpm diff --git a/build-unix.txt b/build-unix.txt new file mode 100644 index 00000000..6b29e1e5 --- /dev/null +++ b/build-unix.txt @@ -0,0 +1,73 @@ +Bitcoin v0.2.0 BETA + +Copyright (c) 2009 Satoshi Nakamoto +Distributed under the MIT/X11 software license, see the accompanying +file license.txt or http://www.opensource.org/licenses/mit-license.php. +This product includes software developed by the OpenSSL Project for use in +the OpenSSL Toolkit (http://www.openssl.org/). This product includes +cryptographic software written by Eric Young (eay@cryptsoft.com). + + +UNIX BUILD NOTES + + +Dependencies +------------ +apt-get install build-essential +apt-get install libgtk2.0-dev +apt-get install libssl-dev + +Libraries you need to obtain separately and build: + default path download +wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ +Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html +Boost \boost http://www.boost.org/users/download/ + +Their licenses: +wxWidgets LGPL 2.1 with very liberal exceptions +Berkeley DB New BSD license with additional requirement that linked software must be free open source +Boost MIT-like license + +Versions used in this release: +GCC 4.3.3 +OpenSSL 0.9.8k +wxWidgets 2.8.9 +Berkeley DB 4.7.25.NC +Boost 1.40.0 + + +Notes +----- +The UI layout is edited with wxFormBuilder. Open the project file +uiproject.fbp. It generates uibase.cpp and uibase.h, which define base +classes that do the rote work of constructing all the UI elements. + +The release is built with GCC and then "strip bitcoin" to strip the debug +symbols, which reduces the executable size by about 90%. + + +wxWidgets +--------- +cd /usr/local/wxWidgets-2.8.9 +mkdir buildgtk +cd buildgtk +../configure --with-gtk --enable-debug --disable-shared --enable-monolithic +make +su +make install +ldconfig + + +Berkeley DB +----------- +cd /usr/local/db-4.7.25.NC/build_unix +../dist/configure --enable-cxx +make + + +Boost +----- +cd /usr/local/boost_1_40_0 +su +./bootstrap.sh +./bjam install diff --git a/build.txt b/build.txt deleted file mode 100644 index 4368af85..00000000 --- a/build.txt +++ /dev/null @@ -1,88 +0,0 @@ -BitCoin v0.1.6 BETA - -Copyright (c) 2009 Satoshi Nakamoto -Distributed under the MIT/X11 software license, see the accompanying -file license.txt or http://www.opensource.org/licenses/mit-license.php. -This product includes software developed by the OpenSSL Project for use in -the OpenSSL Toolkit (http://www.openssl.org/). This product includes -cryptographic software written by Eric Young (eay@cryptsoft.com). - - -Compilers Supported -------------------- -MinGW GCC (v3.4.5) -Microsoft Visual C++ 6.0 SP6 - - -Dependencies ------------- -Libraries you need to obtain separately to build: - - default path download -wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ - or prebuilt: http://wxpack.sourceforge.net -OpenSSL \openssl http://www.openssl.org/source/ -Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html -Boost \boost http://www.boost.org/users/download/ - -Their licenses: -wxWidgets LGPL 2.1 with very liberal exceptions -OpenSSL Old BSD license with the problematic advertising requirement -Berkeley DB New BSD license with additional requirement that linked software must be free open source -Boost MIT-like license - - -Notes ------ -The UI layout is edited with wxFormBuilder. Open the project file -uiproject.fbp. It generates uibase.cpp and uibase.h, which define base -classes that do the rote work of constructing all the UI elements. - -The release is built with GCC and then "strip bitcoin.exe" to strip the debug -symbols, which reduces the executable size by about 90%. - - -OpenSSL -------- -Bitcoin does not use any encryption. If you want to do a no-everything -build of OpenSSL to exclude encryption routines, a few patches are required. -(OpenSSL v0.9.8h) - -Edit engines\e_gmp.c and put this #ifndef around #include - #ifndef OPENSSL_NO_RSA - #include - #endif - -Add this to crypto\err\err_all.c before the ERR_load_crypto_strings line: - void ERR_load_RSA_strings(void) { } - -Edit ms\mingw32.bat and replace the Configure line's parameters with this -no-everything list. You have to put this in the batch file because batch -files can't take more than 9 command line parameters. - perl Configure mingw threads no-rc2 no-rc4 no-rc5 no-idea no-des no-bf no-cast no-aes no-camellia no-seed no-rsa no-dh - -Also REM out the following line in ms\mingw32.bat. The build fails after it's -already finished building libeay32, which is all we care about, but the -failure aborts the script before it runs dllwrap to generate libeay32.dll. - REM if errorlevel 1 goto end - -Build - ms\mingw32.bat - -If you want to use it with MSVC, generate the .lib file - lib /machine:i386 /def:ms\libeay32.def /out:out\libeay32.lib - - -Berkeley DB ------------ -Using MinGW and MSYS: -cd \db\build_unix -sh ../dist/configure --enable-mingw --enable-cxx -make - - -Boost ------ -If you have trouble compiling Boost with Microsoft Visual C++ 6.0, try going -back to Boost version 1.35. It looks like they may be starting to reduce -support for MSVC 6.0. diff --git a/db.cpp b/db.cpp index f9e25834..b702b0cb 100644 --- a/db.cpp +++ b/db.cpp @@ -62,9 +62,9 @@ CDB::CDB(const char* pszFile, const char* pszMode, bool fTxn) : pdb(NULL) if (fShutdown) return; string strDataDir = GetDataDir(); - string strLogDir = strDataDir + "\\database"; + string strLogDir = strDataDir + "/database"; _mkdir(strLogDir.c_str()); - string strErrorFile = strDataDir + "\\db.log"; + string strErrorFile = strDataDir + "/db.log"; printf("dbenv.open strLogDir=%s strErrorFile=%s\n", strLogDir.c_str(), strErrorFile.c_str()); dbenv.set_lg_dir(strLogDir.c_str()); @@ -411,7 +411,6 @@ bool CAddrDB::WriteAddress(const CAddress& addr) bool CAddrDB::LoadAddresses() { - CRITICAL_BLOCK(cs_mapIRCAddresses) CRITICAL_BLOCK(cs_mapAddresses) { // Load user provided addresses @@ -425,10 +424,7 @@ bool CAddrDB::LoadAddresses() { CAddress addr(psz, NODE_NETWORK); if (addr.IsValid()) - { AddAddress(*this, addr); - mapIRCAddresses.insert(make_pair(addr.GetKey(), addr)); - } } } catch (...) { } @@ -678,7 +674,7 @@ void ThreadFlushWalletDB(void* parg) if (nRefCount == 0 && !fShutdown) { // Flush wallet.dat so it's self contained - nLastFlushed == nWalletDBUpdated; + nLastFlushed = nWalletDBUpdated; int64 nStart = GetTimeMillis(); dbenv.txn_checkpoint(0, 0, 0); dbenv.lsn_reset(strFile.c_str(), 0); diff --git a/headers.h b/headers.h index f00b7dde..c7f3cd85 100644 --- a/headers.h +++ b/headers.h @@ -17,16 +17,18 @@ #endif #define _WIN32_IE 0x0400 #define WIN32_LEAN_AND_MEAN 1 +#define __STDC_LIMIT_MACROS // to enable UINT64_MAX from stdint.h #include #include #include #include #include +#include #include #include #include #include -#include +#include #include #include #include @@ -50,22 +52,28 @@ #include #include #include -#include - +#include + #ifdef __WXMSW__ #include #include -#include +#include #include -#include -#include -#include -#else +#include +#include +#include +#else #include +#include #include -#include -#include -#endif +#include +#include +#include +#include +#include +#include +#include +#endif #pragma hdrstop using namespace std; @@ -88,3 +96,11 @@ using namespace boost; #include "market.h" #include "uibase.h" #include "ui.h" + +#include "xpm/addressbook16.xpm" +#include "xpm/addressbook20.xpm" +#include "xpm/bitcoin.xpm" +#include "xpm/check.xpm" +#include "xpm/send16.xpm" +#include "xpm/send16noshadow.xpm" +#include "xpm/send20.xpm" diff --git a/irc.cpp b/irc.cpp index f839dc52..4d4ed0f4 100644 --- a/irc.cpp +++ b/irc.cpp @@ -4,10 +4,7 @@ #include "headers.h" - -map, CAddress> mapIRCAddresses; -CCriticalSection cs_mapIRCAddresses; - +int nGotIRCAddresses = 0; @@ -259,16 +256,7 @@ void ThreadIRCSeed(void* parg) CAddrDB addrdb; if (AddAddress(addrdb, addr)) printf("IRC got new address\n"); - else - { - // make it try connecting again - CRITICAL_BLOCK(cs_mapAddresses) - if (mapAddresses.count(addr.GetKey())) - mapAddresses[addr.GetKey()].nLastFailed = 0; - } - - CRITICAL_BLOCK(cs_mapIRCAddresses) - mapIRCAddresses.insert(make_pair(addr.GetKey(), addr)); + nGotIRCAddresses++; } else { diff --git a/irc.h b/irc.h index 91c3ffe6..c69fd9ee 100644 --- a/irc.h +++ b/irc.h @@ -4,7 +4,5 @@ extern bool RecvLine(SOCKET hSocket, string& strLine); extern void ThreadIRCSeed(void* parg); -extern bool fRestartIRCSeed; -extern map, CAddress> mapIRCAddresses; -extern CCriticalSection cs_mapIRCAddresses; +extern int nGotIRCAddresses; diff --git a/main.cpp b/main.cpp index ade3b51c..2119495e 100644 --- a/main.cpp +++ b/main.cpp @@ -100,13 +100,10 @@ bool AddToWallet(const CWalletTx& wtxIn) if (fInsertedNew) wtx.nTimeReceived = GetAdjustedTime(); - //// debug print - printf("AddToWallet %s %s\n", wtxIn.GetHash().ToString().substr(0,6).c_str(), fInsertedNew ? "new" : "update"); - + bool fUpdated = false; if (!fInsertedNew) { // Merge - bool fUpdated = false; if (wtxIn.hashBlock != 0 && wtxIn.hashBlock != wtx.hashBlock) { wtx.hashBlock = wtxIn.hashBlock; @@ -128,13 +125,15 @@ bool AddToWallet(const CWalletTx& wtxIn) wtx.fSpent = wtxIn.fSpent; fUpdated = true; } - if (!fUpdated) - return true; } + //// debug print + printf("AddToWallet %s %s%s\n", wtxIn.GetHash().ToString().substr(0,6).c_str(), (fInsertedNew ? "new" : ""), (fUpdated ? "update" : "")); + // Write to disk - if (!wtx.WriteToDisk()) - return false; + if (fInsertedNew || fUpdated) + if (!wtx.WriteToDisk()) + return false; // Notify UI vWalletUpdated.push_back(hash); @@ -820,7 +819,7 @@ bool CTransaction::ConnectInputs(CTxDB& txdb, map& mapTestPoo } if (prevout.n >= txPrev.vout.size() || prevout.n >= txindex.vSpent.size()) - return error("ConnectInputs() : %s prevout.n out of range %d %d %d", GetHash().ToString().substr(0,6).c_str(), prevout.n, txPrev.vout.size(), txindex.vSpent.size()); + return error("ConnectInputs() : %s prevout.n out of range %d %d %d prev tx %s\n%s", GetHash().ToString().substr(0,6).c_str(), prevout.n, txPrev.vout.size(), txindex.vSpent.size(), prevout.hash.ToString().substr(0,6).c_str(), txPrev.ToString().c_str()); // If prev is coinbase, check that it's matured if (txPrev.IsCoinBase()) @@ -1217,7 +1216,7 @@ bool CBlock::AcceptBlock() if (nTime <= pindexPrev->GetMedianTimePast()) return error("AcceptBlock() : block's timestamp is too early"); - // Check that all transactions are finalized (starting around 30 Nov 2009) + // Check that all transactions are finalized (starting around Dec 2009) if (nBestHeight > 31000) // 25620 + 5320 foreach(const CTransaction& tx, vtx) if (!tx.IsFinal(nTime)) @@ -1384,7 +1383,7 @@ FILE* OpenBlockFile(unsigned int nFile, unsigned int nBlockPos, const char* pszM { if (nFile == -1) return NULL; - FILE* file = fopen(strprintf("%s\\blk%04d.dat", GetDataDir().c_str(), nFile).c_str(), pszMode); + FILE* file = fopen(strprintf("%s/blk%04d.dat", GetDataDir().c_str(), nFile).c_str(), pszMode); if (!file) return NULL; if (nBlockPos != 0 && !strchr(pszMode, 'a') && !strchr(pszMode, 'w')) @@ -1718,6 +1717,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) + if (strCommand == "version") { // Each connection can only send one version message @@ -1765,6 +1765,10 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) pfrom->fSuccessfullyConnected = true; + // Update the last seen time + if (pfrom->fNetworkNode) + AddressCurrentlyConnected(pfrom->addr); + printf("version message: version %d\n", pfrom->nVersion); } @@ -1781,23 +1785,16 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vector vAddr; vRecv >> vAddr; - // Clear addrknown lists periodically to allow refresh broadcasts - static int64 nLastClearedAddrKnown; - if (nLastClearedAddrKnown < GetAdjustedTime() - 24 * 60 * 60) - { - nLastClearedAddrKnown = GetAdjustedTime(); - CRITICAL_BLOCK(cs_vNodes) - foreach(CNode* pnode, vNodes) - pnode->setAddrKnown.clear(); - } - // Store the new addresses CAddrDB addrdb; - foreach(const CAddress& addr, vAddr) + foreach(CAddress& addr, vAddr) { if (fShutdown) return true; - AddAddress(addrdb, addr); + addr.nTime = GetAdjustedTime(); + if (pfrom->fGetAddr) + addr.nTime -= 5 * 24 * 60 * 60; + AddAddress(addrdb, addr, false); pfrom->AddAddressKnown(addr); if (!pfrom->fGetAddr && addr.IsRoutable()) { @@ -1816,6 +1813,10 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vector vInv; vRecv >> vInv; + // Update the last seen time for this node's address + if (pfrom->fNetworkNode) + AddressCurrentlyConnected(pfrom->addr); + CTxDB txdb("r"); foreach(const CInv& inv, vInv) { @@ -2099,6 +2100,25 @@ bool SendMessages(CNode* pto) if (pto->nVersion == 0) return true; + // Address refresh broadcast + static int64 nLastRebroadcast; + if (nLastRebroadcast < GetTime() - 24 * 60 * 60) // every 24 hours + { + nLastRebroadcast = GetTime(); + CRITICAL_BLOCK(cs_vNodes) + { + foreach(CNode* pnode, vNodes) + { + // Periodically clear setAddrKnown to allow refresh broadcasts + pnode->setAddrKnown.clear(); + + // Rebroadcast our address + if (addrLocalHost.IsRoutable() && !fUseProxy) + pnode->PushAddress(addrLocalHost); + } + } + } + // // Message: addr @@ -2187,7 +2207,7 @@ void GenerateBitcoins(bool fGenerate) } if (fGenerateBitcoins) { - int nProcessors = atoi(getenv("NUMBER_OF_PROCESSORS")); + int nProcessors = wxThread::GetCPUCount(); printf("%d processors\n", nProcessors); if (nProcessors < 1) nProcessors = 1; diff --git a/main.h b/main.h index a0258cf2..6d8f0ed8 100644 --- a/main.h +++ b/main.h @@ -968,6 +968,14 @@ public: return error("CBlock::WriteToDisk() : ftell failed"); fileout << *this; + // Flush stdio buffers and commit to disk before returning + fflush(fileout); +#ifdef __WXMSW__ + _commit(_fileno(fileout)); +#else + fsync(fileno(fileout)); +#endif + return true; } diff --git a/makefile.unix b/makefile.unix new file mode 100644 index 00000000..c62efc00 --- /dev/null +++ b/makefile.unix @@ -0,0 +1,86 @@ +# Copyright (c) 2009 Satoshi Nakamoto +# Distributed under the MIT/X11 software license, see the accompanying +# file license.txt or http://www.opensource.org/licenses/mit-license.php. + + +ifneq "$(BUILD)" "debug" +ifneq "$(BUILD)" "release" +BUILD=debug +endif +endif +ifeq "$(BUILD)" "debug" +D=d +DEBUGFLAGS=-g -D__WXDEBUG__ +endif + + + +INCLUDEPATHS=-I"/usr/include" \ + -I"/usr/local/boost_1_40_0" \ + -I"/usr/local/db-4.7.25.NC/build_unix" \ + -I"/usr/local/include/wx-2.8" \ + -I"/usr/local/lib/wx/include/gtk2-ansi-debug-static-2.8" + +LIBPATHS=-L"/usr/lib" \ + -L"/usr/local/lib" \ + -L"/usr/local/db-4.7.25.NC/build_unix" + +LIBS= \ + -Wl,-Bstatic -l boost_thread -l boost_system -l boost_filesystem -Wl,-Bdynamic \ + -Wl,-Bstatic -l db_cxx -l wx_gtk2$(D)-2.8 -Wl,-Bdynamic \ + -l crypto \ + -l gtk-x11-2.0 -l gthread-2.0 -l SM +WXDEFS=-D__WXGTK__ -DNOPCH +CFLAGS=-O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h + + + +all: bitcoin + + +headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h + g++ -c $(CFLAGS) -o $@ $< + +obj/util.o: util.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/script.o: script.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/db.o: db.cpp $(HEADERS) market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/net.o: net.cpp $(HEADERS) net.h + g++ -c $(CFLAGS) -o $@ $< + +obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h + g++ -c $(CFLAGS) -o $@ $< + +obj/market.o: market.cpp $(HEADERS) market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/uibase.o: uibase.cpp uibase.h + g++ -c $(CFLAGS) -o $@ $< + +obj/sha.o: sha.cpp sha.h + g++ -c $(CFLAGS) -O3 -o $@ $< + +obj/irc.o: irc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + + + + +OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + +bitcoin: headers.h.gch $(OBJS) + g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) + +clean: + -rm obj/* + -rm headers.h.gch diff --git a/net.cpp b/net.cpp index 8ccf48b8..c14061e7 100644 --- a/net.cpp +++ b/net.cpp @@ -3,7 +3,6 @@ // file license.txt or http://www.opensource.org/licenses/mit-license.php. #include "headers.h" -#include void ThreadMessageHandler2(void* parg); void ThreadSocketHandler2(void* parg); @@ -201,12 +200,14 @@ bool GetMyExternalIP(unsigned int& ipRet) -bool AddAddress(CAddrDB& addrdb, const CAddress& addr) +bool AddAddress(CAddrDB& addrdb, CAddress addr, bool fCurrentlyOnline) { if (!addr.IsRoutable()) return false; if (addr.ip == addrLocalHost.ip) return false; + if (fCurrentlyOnline) + addr.nTime = GetAdjustedTime(); CRITICAL_BLOCK(cs_mapAddresses) { map, CAddress>::iterator it = mapAddresses.find(addr.GetKey()); @@ -219,24 +220,47 @@ bool AddAddress(CAddrDB& addrdb, const CAddress& addr) } else { + bool fUpdated = false; CAddress& addrFound = (*it).second; if ((addrFound.nServices | addr.nServices) != addrFound.nServices) { // Services have been added addrFound.nServices |= addr.nServices; - addrdb.WriteAddress(addrFound); - return true; + fUpdated = true; + } + int64 nUpdateInterval = (fCurrentlyOnline ? 60 * 60 : 24 * 60 * 60); + if (addrFound.nTime < addr.nTime - nUpdateInterval) + { + // Periodically update most recently seen time + addrFound.nTime = addr.nTime; + fUpdated = true; } - else if (addrFound.nTime < GetAdjustedTime() - 24 * 60 * 60) + if (fUpdated) + addrdb.WriteAddress(addrFound); + } + } + return false; +} + +void AddressCurrentlyConnected(const CAddress& addr) +{ + CRITICAL_BLOCK(cs_mapAddresses) + { + // Only if it's been published already + map, CAddress>::iterator it = mapAddresses.find(addr.GetKey()); + if (it != mapAddresses.end()) + { + CAddress& addrFound = (*it).second; + int64 nUpdateInterval = 60 * 60; + if (addrFound.nTime < GetAdjustedTime() - nUpdateInterval) { // Periodically update most recently seen time addrFound.nTime = GetAdjustedTime(); + CAddrDB addrdb; addrdb.WriteAddress(addrFound); - return false; } } } - return false; } @@ -398,9 +422,14 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) printf("connected %s\n", addrConnect.ToStringLog().c_str()); // Set to nonblocking - u_long nOne = 1; +#ifdef __WXMSW__ + u_long nOne = 1; if (ioctlsocket(hSocket, FIONBIO, &nOne) == SOCKET_ERROR) printf("ConnectSocket() : ioctlsocket nonblocking setting failed, error %d\n", WSAGetLastError()); +#else + if (fcntl(hSocket, F_SETFL, O_NONBLOCK) == SOCKET_ERROR) + printf("ConnectSocket() : fcntl nonblocking setting failed, error %d\n", errno); +#endif // Add node CNode* pnode = new CNode(hSocket, addrConnect, false); @@ -418,7 +447,7 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) else { CRITICAL_BLOCK(cs_mapAddresses) - mapAddresses[addrConnect.GetKey()].nLastFailed = GetTime(); + mapAddresses[addrConnect.GetKey()].nLastFailed = GetAdjustedTime(); return NULL; } } @@ -432,7 +461,7 @@ void CNode::DoDisconnect() // If outbound and never got version message, mark address as failed if (!fInbound && !fSuccessfullyConnected) CRITICAL_BLOCK(cs_mapAddresses) - mapAddresses[addr.GetKey()].nLastFailed = GetTime(); + mapAddresses[addr.GetKey()].nLastFailed = GetAdjustedTime(); // All of a nodes broadcasts and subscriptions are automatically torn down // when it goes down, so a node has to stay up to keep its broadcast going. @@ -549,8 +578,8 @@ void ThreadSocketHandler2(void* parg) timeout.tv_sec = 0; timeout.tv_usec = 50000; // frequency to poll pnode->vSend - struct fd_set fdsetRecv; - struct fd_set fdsetSend; + fd_set fdsetRecv; + fd_set fdsetSend; FD_ZERO(&fdsetRecv); FD_ZERO(&fdsetSend); SOCKET hSocketMax = 0; @@ -599,7 +628,11 @@ void ThreadSocketHandler2(void* parg) if (FD_ISSET(hListenSocket, &fdsetRecv)) { struct sockaddr_in sockaddr; +#ifdef __WXMSW__ int len = sizeof(sockaddr); +#else + socklen_t len = sizeof(sockaddr); +#endif SOCKET hSocket = accept(hListenSocket, (struct sockaddr*)&sockaddr, &len); CAddress addr(sockaddr); if (hSocket == INVALID_SOCKET) @@ -765,14 +798,12 @@ void ThreadOpenConnections2(void* parg) } // Initiate network connections - int nTry = 0; - bool fIRCOnly = false; - const int nMaxConnections = 15; loop { // Wait vnThreadsRunning[1]--; Sleep(500); + const int nMaxConnections = 15; while (vNodes.size() >= nMaxConnections || vNodes.size() >= mapAddresses.size()) { CheckForShutdown(1); @@ -781,93 +812,55 @@ void ThreadOpenConnections2(void* parg) vnThreadsRunning[1]++; CheckForShutdown(1); - // - // The IP selection process is designed to limit vulnerability to address flooding. - // Any class C (a.b.c.?) has an equal chance of being chosen, then an IP is - // chosen within the class C. An attacker may be able to allocate many IPs, but - // they would normally be concentrated in blocks of class C's. They can hog the - // attention within their class C, but not the whole IP address space overall. - // A lone node in a class C will get as much attention as someone holding all 255 - // IPs in another class C. + // Choose an address to connect to based on most recently seen // + CAddress addrConnect; + int64 nBestTime = 0; + int64 nDelay = ((60 * 60) << vNodes.size()); + if (vNodes.size() >= 3) + nDelay *= 4; + if (nGotIRCAddresses > 0) + nDelay *= 100; + + // Do this here so we don't have to critsect vNodes inside mapAddresses critsect + set setConnected; + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + setConnected.insert(pnode->addr.ip); - // Every other try is with IRC addresses only - fIRCOnly = !fIRCOnly; - if (mapIRCAddresses.empty()) - fIRCOnly = false; - else if (nTry++ < 30 && vNodes.size() < nMaxConnections/2) - fIRCOnly = true; - - // Make a list of unique class C's - unsigned char pchIPCMask[4] = { 0xff, 0xff, 0xff, 0x00 }; - unsigned int nIPCMask = *(unsigned int*)pchIPCMask; - vector vIPC; - CRITICAL_BLOCK(cs_mapIRCAddresses) CRITICAL_BLOCK(cs_mapAddresses) { - vIPC.reserve(mapAddresses.size()); - unsigned int nPrev = 0; foreach(const PAIRTYPE(vector, CAddress)& item, mapAddresses) { const CAddress& addr = item.second; - if (!addr.IsIPv4()) - continue; - if (fIRCOnly && !mapIRCAddresses.count(item.first)) + if (!addr.IsIPv4() || !addr.IsValid() || setConnected.count(addr.ip)) continue; - // Taking advantage of mapAddresses being in sorted order, - // with IPs of the same class C grouped together. - unsigned int ipC = addr.ip & nIPCMask; - if (ipC != nPrev) - vIPC.push_back(nPrev = ipC); - } - } - if (vIPC.empty()) - continue; - - // Choose a random class C - unsigned int ipC = vIPC[GetRand(vIPC.size())]; + // Limit retry frequency + if (GetAdjustedTime() < addr.nLastFailed + nDelay) + continue; - // Organize all addresses in the class C by IP - map > mapIP; - CRITICAL_BLOCK(cs_mapIRCAddresses) - CRITICAL_BLOCK(cs_mapAddresses) - { - int64 nDelay = ((30 * 60) << vNodes.size()); - if (!fIRCOnly) - { - nDelay *= 2; - if (vNodes.size() >= 3) - nDelay *= 4; - if (!mapIRCAddresses.empty()) - nDelay *= 100; - } + // Try again only after all addresses had a first attempt + int64 nTime = addr.nTime; + if (addr.nLastFailed > addr.nTime) + nTime -= 365 * 24 * 60 * 60; - for (map, CAddress>::iterator mi = mapAddresses.lower_bound(CAddress(ipC, 0).GetKey()); - mi != mapAddresses.upper_bound(CAddress(ipC | ~nIPCMask, 0xffff).GetKey()); - ++mi) - { - const CAddress& addr = (*mi).second; - if (fIRCOnly && !mapIRCAddresses.count((*mi).first)) - continue; + // Randomize the order a little, putting the standard port first + nTime += GetRand(1 * 60 * 60); + if (addr.port != DEFAULT_PORT) + nTime -= 1 * 60 * 60; - int64 nRandomizer = (addr.nLastFailed * addr.ip * 7777U) % 20000; - if (GetTime() - addr.nLastFailed > nDelay * nRandomizer / 10000) - mapIP[addr.ip].push_back(addr); + if (nTime > nBestTime) + { + nBestTime = nTime; + addrConnect = addr; + } } } - if (mapIP.empty()) - continue; - - // Choose a random IP in the class C - map >::iterator mi = mapIP.begin(); - advance(mi, GetRand(mapIP.size())); - // Once we've chosen an IP, we'll try every given port before moving on - foreach(const CAddress& addrConnect, (*mi).second) - if (OpenNetworkConnection(addrConnect)) - break; + if (addrConnect.IsValid()) + OpenNetworkConnection(addrConnect); } } @@ -989,6 +982,7 @@ bool StartNode(string& strError) pnodeLocalHost = new CNode(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); strError = ""; +#ifdef __WXMSW__ // Sockets startup WSADATA wsadata; int ret = WSAStartup(MAKEWORD(2,2), &wsadata); @@ -998,6 +992,7 @@ bool StartNode(string& strError) printf("%s\n", strError.c_str()); return false; } +#endif // Get local host ip char pszHostName[255]; @@ -1029,10 +1024,14 @@ bool StartNode(string& strError) } // Set to nonblocking, incoming connections will also inherit this +#ifdef __WXMSW__ u_long nOne = 1; if (ioctlsocket(hListenSocket, FIONBIO, &nOne) == SOCKET_ERROR) +#else + if (fcntl(hListenSocket, F_SETFL, O_NONBLOCK) == SOCKET_ERROR) +#endif { - strError = strprintf("Error: Couldn't set properties on socket for incoming connections (ioctlsocket returned error %d)", WSAGetLastError()); + strError = strprintf("Error: Couldn't set properties on socket for incoming connections (error %d)", WSAGetLastError()); printf("%s\n", strError.c_str()); return false; } @@ -1041,7 +1040,7 @@ bool StartNode(string& strError) // IP address, and port for the socket that is being bound int nRetryLimit = 15; struct sockaddr_in sockaddr = addrLocalHost.GetSockAddr(); - if (bind(hListenSocket, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) == SOCKET_ERROR) + if (::bind(hListenSocket, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) == SOCKET_ERROR) { int nErr = WSAGetLastError(); if (nErr == WSAEADDRINUSE) @@ -1131,7 +1130,9 @@ bool StopNode() Sleep(50); // Sockets shutdown +#ifdef __WXMSW__ WSACleanup(); +#endif return true; } diff --git a/net.h b/net.h index 7716426f..275a4cb8 100644 --- a/net.h +++ b/net.h @@ -22,7 +22,8 @@ enum bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet); bool GetMyExternalIP(unsigned int& ipRet); -bool AddAddress(CAddrDB& addrdb, const CAddress& addr); +bool AddAddress(CAddrDB& addrdb, CAddress addr, bool fCurrentlyOnline=true); +void AddressCurrentlyConnected(const CAddress& addr); CNode* FindNode(unsigned int ip); CNode* ConnectNode(CAddress addrConnect, int64 nTimeout=0); void AbandonRequests(void (*fn)(void*, CDataStream&), void* param1); diff --git a/ui.cpp b/ui.cpp index 9185b815..21254819 100644 --- a/ui.cpp +++ b/ui.cpp @@ -260,7 +260,8 @@ void AddPendingReplyEvent3(void* pevthandler, CDataStream& vRecv) CDataStream GetStreamFromEvent(const wxCommandEvent& event) { wxString strData = event.GetString(); - return CDataStream(strData.begin(), strData.begin() + event.GetInt(), SER_NETWORK); + const char* pszBegin = strData.c_str(); + return CDataStream(pszBegin, pszBegin + event.GetInt(), SER_NETWORK); } @@ -288,20 +289,6 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) SetIcon(wxICON(bitcoin)); ptaskbaricon = new CMyTaskBarIcon(); - // Init toolbar with transparency masked bitmaps - m_toolBar->ClearTools(); - - //// shouldn't have to do mask separately anymore, bitmap alpha support added in wx 2.8.9, - wxBitmap bmpSend(wxT("send20"), wxBITMAP_TYPE_RESOURCE); - bmpSend.SetMask(new wxMask(wxBitmap(wxT("send20mask"), wxBITMAP_TYPE_RESOURCE))); - m_toolBar->AddTool(wxID_BUTTONSEND, wxT("&Send Coins"), bmpSend, wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); - - wxBitmap bmpAddressBook(wxT("addressbook20"), wxBITMAP_TYPE_RESOURCE); - bmpAddressBook.SetMask(new wxMask(wxBitmap(wxT("addressbook20mask"), wxBITMAP_TYPE_RESOURCE))); - m_toolBar->AddTool(wxID_BUTTONRECEIVE, wxT("&Address Book"), bmpAddressBook, wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); - - m_toolBar->Realize(); - // Init column headers int nDateWidth = DateTimeStr(1229413914).size() * 6 + 8; if (!strstr(DateTimeStr(1229413914).c_str(), "2008")) @@ -909,15 +896,17 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) { TRY_CRITICAL_BLOCK(cs_mapWallet) { - bool fInserted = false; + string strTop; + if (m_listCtrl->GetItemCount()) + strTop = (string)m_listCtrl->GetItemText(0); foreach(uint256 hash, vWalletUpdated) { map::iterator mi = mapWallet.find(hash); if (mi != mapWallet.end()) - fInserted |= InsertTransaction((*mi).second, false); + InsertTransaction((*mi).second, false); } vWalletUpdated.clear(); - if (fInserted) + if (m_listCtrl->GetItemCount() && strTop != (string)m_listCtrl->GetItemText(0)) m_listCtrl->ScrollList(0, INT_MAX); } } @@ -954,7 +943,9 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, nTransactionCount); m_statusBar->SetStatusText(strStatus, 2); +#ifdef __WXMSW__ m_listCtrl->OnPaint(event); +#endif } @@ -1407,7 +1398,7 @@ COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) m_checkBoxLimitProcessors->SetValue(fLimitProcessors); m_spinCtrlLimitProcessors->Enable(fLimitProcessors); m_spinCtrlLimitProcessors->SetValue(nLimitProcessors); - int nProcessors = atoi(getenv("NUMBER_OF_PROCESSORS")); + int nProcessors = wxThread::GetCPUCount(); if (nProcessors < 1) nProcessors = 999; m_spinCtrlLimitProcessors->SetRange(1, nProcessors); @@ -1549,17 +1540,11 @@ void COptionsDialog::OnButtonApply(wxCommandEvent& event) walletdb.WriteSetting("fMinimizeOnClose", fMinimizeOnClose); } - if (fUseProxy != m_checkBoxUseProxy->GetValue()) - { - fUseProxy = m_checkBoxUseProxy->GetValue(); - walletdb.WriteSetting("fUseProxy", fUseProxy); - } + fUseProxy = m_checkBoxUseProxy->GetValue(); + walletdb.WriteSetting("fUseProxy", fUseProxy); - if (addrProxy != GetProxyAddr()) - { - addrProxy = GetProxyAddr(); - walletdb.WriteSetting("addrProxy", addrProxy); - } + addrProxy = GetProxyAddr(); + walletdb.WriteSetting("addrProxy", addrProxy); } @@ -1608,10 +1593,8 @@ CSendDialog::CSendDialog(wxWindow* parent, const wxString& strAddress) : CSendDi //// todo: should add a display of your balance for convenience // Set Icon - wxBitmap bmpSend(wxT("send16"), wxBITMAP_TYPE_RESOURCE); - bmpSend.SetMask(new wxMask(wxBitmap(wxT("send16masknoshadow"), wxBITMAP_TYPE_RESOURCE))); wxIcon iconSend; - iconSend.CopyFromBitmap(bmpSend); + iconSend.CopyFromBitmap(wxBitmap(send16noshadow_xpm)); SetIcon(iconSend); wxCommandEvent event; @@ -2231,10 +2214,8 @@ CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInit m_listCtrl->SetFocus(); // Set Icon - wxBitmap bmpAddressBook(wxT("addressbook16"), wxBITMAP_TYPE_RESOURCE); - bmpAddressBook.SetMask(new wxMask(wxBitmap(wxT("addressbook16mask"), wxBITMAP_TYPE_RESOURCE))); wxIcon iconAddressBook; - iconAddressBook.CopyFromBitmap(bmpAddressBook); + iconAddressBook.CopyFromBitmap(wxBitmap(addressbook16_xpm)); SetIcon(iconAddressBook); // Fill listctrl with address book data @@ -3345,7 +3326,7 @@ bool CMyApp::OnInit2() _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE); _CrtSetReportFile(_CRT_WARN, CreateFile("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); #endif -#ifdef __WXDEBUG__ +#if defined(__WXMSW__) && defined(__WXDEBUG__) // Disable malfunctioning wxWidgets debug assertion g_isPainting = 10000; #endif @@ -3362,8 +3343,7 @@ bool CMyApp::OnInit2() " -gen=0\t\t Don't generate coins\n" " -min\t\t Start minimized\n" " -datadir=\t Specify data directory\n" - " -proxy=\t Connect through socks4 proxy,\n" - " \t\t e.g. -proxy=127.0.0.1:9050 to use TOR\n" + " -proxy=\t Connect through socks4 proxy\n" " -addnode=\t Add a node to connect to\n" " -connect=\t Connect only to the specified node\n" " -?\t\t This help message\n"; @@ -3386,6 +3366,8 @@ bool CMyApp::OnInit2() unsigned int nStart = GetTime(); loop { + // TODO: find out how to do this in Linux, or replace with wxWidgets commands +#ifdef __WXMSW__ // Show the previous instance and exit HWND hwndPrev = FindWindow("wxWindowClassNR", "Bitcoin"); if (hwndPrev) @@ -3395,6 +3377,7 @@ bool CMyApp::OnInit2() SetForegroundWindow(hwndPrev); return false; } +#endif if (GetTime() > nStart + 60) return false; @@ -3421,7 +3404,7 @@ bool CMyApp::OnInit2() fPrintToDebugger = true; printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); - printf("Bitcoin version %d, Windows version %08x\n", VERSION, GetVersion()); + printf("Bitcoin version %d, OS version %s\n", VERSION, wxGetOsDescription().mb_str()); if (mapArgs.count("-dropmessages")) { @@ -3493,12 +3476,36 @@ bool CMyApp::OnInit2() return false; } + if (mapArgs.count("-printblock")) + { + string strMatch = mapArgs["-printblock"]; + int nFound = 0; + for (map::iterator mi = mapBlockIndex.begin(); mi != mapBlockIndex.end(); ++mi) + { + uint256 hash = (*mi).first; + if (strncmp(hash.ToString().c_str(), strMatch.c_str(), strMatch.size()) == 0) + { + CBlockIndex* pindex = (*mi).second; + CBlock block; + block.ReadFromDisk(pindex, true); + block.BuildMerkleTree(); + block.print(); + printf("\n"); + nFound++; + } + } + if (nFound == 0) + printf("No blocks matching %s were found\n", strMatch.c_str()); + OnExit(); + return false; + } + if (mapArgs.count("-gen")) { if (mapArgs["-gen"].empty()) fGenerateBitcoins = true; else - fGenerateBitcoins = atoi(mapArgs["-gen"].c_str()); + fGenerateBitcoins = (atoi(mapArgs["-gen"].c_str()) != 0); } if (mapArgs.count("-proxy")) @@ -3511,9 +3518,6 @@ bool CMyApp::OnInit2() OnExit(); return false; } - CWalletDB walletdb; - walletdb.WriteSetting("fUseProxy", fUseProxy); - walletdb.WriteSetting("addrProxy", addrProxy); } if (mapArgs.count("-addnode")) @@ -3522,6 +3526,7 @@ bool CMyApp::OnInit2() foreach(string strAddr, mapMultiArgs["-addnode"]) { CAddress addr(strAddr, NODE_NETWORK); + addr.nTime = 0; // so it won't relay unless successfully connected if (addr.IsValid()) AddAddress(addrdb, addr); } @@ -3559,7 +3564,11 @@ bool CMyApp::OnInit2() // // Tests // +#ifdef __WXMSW__ if (argc >= 2 && stricmp(argv[1], "-send") == 0) +#else + if (argc >= 2 && strcmp(argv[1], "-send") == 0) +#endif { int64 nValue = 1; if (argc >= 3) @@ -3646,7 +3655,8 @@ void CMyApp::OnFatalException() -typedef WINSHELLAPI BOOL WINAPI (*PSHGETSPECIALFOLDERPATHA)(HWND hwndOwner, LPSTR lpszPath, int nFolder, BOOL fCreate); +#ifdef __WXMSW__ +typedef WINSHELLAPI BOOL (WINAPI *PSHGETSPECIALFOLDERPATHA)(HWND hwndOwner, LPSTR lpszPath, int nFolder, BOOL fCreate); string MyGetSpecialFolderPath(int nFolder, bool fCreate) { @@ -3737,7 +3747,10 @@ void SetStartOnSystemStartup(bool fAutoStart) CoUninitialize(); } } - +#else +bool GetStartOnSystemStartup() { return false; } +void SetStartOnSystemStartup(bool fAutoStart) { } +#endif diff --git a/uibase.cpp b/uibase.cpp index b03e578b..bb564e99 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -7,6 +7,10 @@ #include "uibase.h" +#include "xpm/addressbook20.xpm" +#include "xpm/check.xpm" +#include "xpm/send20.xpm" + /////////////////////////////////////////////////////////////////////////// CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) @@ -60,8 +64,8 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_toolBar->SetToolSeparation( 1 ); m_toolBar->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); - m_toolBar->AddTool( wxID_BUTTONSEND, wxT("&Send Coins"), wxBitmap( wxT("send20"), wxBITMAP_TYPE_RESOURCE ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); - m_toolBar->AddTool( wxID_BUTTONRECEIVE, wxT("&Address Book"), wxBitmap( wxT("addressbook20"), wxBITMAP_TYPE_RESOURCE ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->AddTool( wxID_BUTTONSEND, wxT("&Send Coins"), wxBitmap( send20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->AddTool( wxID_BUTTONRECEIVE, wxT("&Address Book"), wxBitmap( addressbook20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); m_toolBar->Realize(); m_statusBar = this->CreateStatusBar( 1, wxST_SIZEGRIP, wxID_ANY ); @@ -666,7 +670,7 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin bSizer47->Add( 0, 0, 1, wxEXPAND, 5 ); - m_bitmapCheckMark = new wxStaticBitmap( this, wxID_ANY, wxICON( check ), wxDefaultPosition, wxSize( 16,16 ), 0 ); + m_bitmapCheckMark = new wxStaticBitmap( this, wxID_ANY, wxBitmap( check_xpm ), wxDefaultPosition, wxSize( 16,16 ), 0 ); bSizer47->Add( m_bitmapCheckMark, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); m_staticText36 = new wxStaticText( this, wxID_ANY, wxT("Pay &To:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); diff --git a/uiproject.fbp b/uiproject.fbp index 313e5aa6..7bce7349 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -225,7 +225,7 @@ - + 20,20 @@ -273,7 +273,7 @@ - send20; Load From Resource + xpm/send20.xpm; Load From File wxID_BUTTONSEND wxITEM_NORMAL &Send Coins @@ -287,7 +287,7 @@ - addressbook20; Load From Resource + xpm/addressbook20.xpm; Load From File wxID_BUTTONRECEIVE wxITEM_NORMAL &Address Book @@ -1685,7 +1685,7 @@ - + @@ -3699,7 +3699,7 @@ 0 - check; Load From Icon Resource [-1; -1] + xpm/check.xpm; Load From File 1 diff --git a/util.cpp b/util.cpp index 23b59f11..d2e624d6 100644 --- a/util.cpp +++ b/util.cpp @@ -38,8 +38,10 @@ public: ppmutexOpenSSL[i] = new wxMutex(); CRYPTO_set_locking_callback(locking_callback); +#ifdef __WXMSW__ // Seed random number generator with screen scrape and other hardware sources RAND_screen(); +#endif // Seed random number generator with performance counter RandAddSeed(); @@ -325,8 +327,8 @@ void ParseParameters(int argc, char* argv[]) pszValue = strchr(psz, '='); *pszValue++ = '\0'; } - strlwr(psz); #ifdef __WXMSW__ + _strlwr(psz); if (psz[0] == '/') psz[0] = '-'; #endif @@ -343,9 +345,13 @@ void ParseParameters(int argc, char* argv[]) void FormatException(char* pszMessage, std::exception* pex, const char* pszThread) { +#ifdef __WXMSW__ char pszModule[MAX_PATH]; pszModule[0] = '\0'; GetModuleFileName(NULL, pszModule, sizeof(pszModule)); +#else + const char* pszModule = wxStandardPaths::Get().GetExecutablePath().mb_str(); +#endif if (pex) snprintf(pszMessage, 1000, "EXCEPTION: %s \n%s \n%s in %s \n", typeid(*pex).name(), pex->what(), pszModule, pszThread); diff --git a/util.h b/util.h index 822a049c..3bc7c798 100644 --- a/util.h +++ b/util.h @@ -55,9 +55,34 @@ inline T& REF(const T& val) } #ifndef __WXMSW__ -#define closesocket(s) close(s) -#define INVALID_SOCKET (SOCKET)(~0) +#define _UI64_MAX UINT64_MAX +#define _I64_MAX INT64_MAX +#define WSAGetLastError() errno +#define WSAEWOULDBLOCK EWOULDBLOCK +#define WSAEMSGSIZE EMSGSIZE +#define WSAEINTR EINTR +#define WSAEINPROGRESS EINPROGRESS +#define WSAEADDRINUSE EADDRINUSE +#define closesocket(s) close(s) +#define INVALID_SOCKET (SOCKET)(~0) +#define SOCKET_ERROR -1 typedef u_int SOCKET; +#define _vsnprintf(a,b,c,d) vsnprintf(a,b,c,d) +#define strlwr(psz) to_lower(psz) +#define _strlwr(psz) to_lower(psz) +#define _mkdir(psz) filesystem::create_directory(psz) +#define MAX_PATH 1024 +#define Sleep(n) wxMilliSleep(n) +#define Beep(n1,n2) (0) +inline int _beginthread(void(*pfn)(void*), unsigned nStack, void* parg) { thread(bind(pfn, parg)); return 0; } +inline void _endthread() { pthread_exit(NULL); } +inline int GetCurrentThread() { return 0; } +// threads are processes on linux, so setpriority affects just the one thread +inline void SetThreadPriority(int nThread, int nPriority) { setpriority(PRIO_PROCESS, getpid(), nPriority); } +#define THREAD_PRIORITY_LOWEST PRIO_MIN +#define THREAD_PRIORITY_BELOW_NORMAL 2 +#define THREAD_PRIORITY_NORMAL 0 +#define THREAD_PRIORITY_ABOVE_NORMAL 0 #endif @@ -120,7 +145,7 @@ public: protected: wxMutex mutex; public: - explicit CCriticalSection() { } + explicit CCriticalSection() : mutex(wxMUTEX_RECURSIVE) { } ~CCriticalSection() { } void Enter() { mutex.Lock(); } void Leave() { mutex.Unlock(); } @@ -183,7 +208,7 @@ inline int OutputDebugStringF(const char* pszFormat, ...) // print to debug.log char pszFile[MAX_PATH+100]; GetDataDir(pszFile); - strlcat(pszFile, "\\debug.log", sizeof(pszFile)); + strlcat(pszFile, "/debug.log", sizeof(pszFile)); FILE* fileout = fopen(pszFile, "a"); if (fileout) { @@ -354,13 +379,6 @@ inline int64 GetTimeMillis() return wxGetLocalTimeMillis().GetValue(); } -#ifndef __WXMSW__ -inline void Sleep(unsigned int nMilliseconds) -{ - wxMilliSleep(nMilliseconds); -} -#endif - diff --git a/xpm/addressbook16.xpm b/xpm/addressbook16.xpm new file mode 100644 index 00000000..471f700c --- /dev/null +++ b/xpm/addressbook16.xpm @@ -0,0 +1,278 @@ +/* XPM */ +static char * addressbook16_xpm[] = { +/* columns rows colors chars-per-pixel */ +"16 16 256 2", +" c #FFFFFF", +". c #F7FFFF", +"X c #F7F7FF", +"o c #EFF7FF", +"O c #E6EFF7", +"+ c #E6E6F7", +"@ c #CEE6F7", +"# c #DEDEEF", +"$ c #D6DEEF", +"% c #D6DEE6", +"& c #CEDEF7", +"* c #CEDEEF", +"= c #EFF708", +"- c #C5DEF7", +"; c #CED6EF", +": c None", +"> c #C5D6E6", +", c #BDD6F7", +"< c #BDD6EF", +"1 c #D6CECE", +"2 c #BDCEE6", +"3 c #BDC5E6", +"4 c #B5C5DE", +"5 c #BDD631", +"6 c #ADBDDE", +"7 c #B5B5BD", +"8 c #A5B5D6", +"9 c #00FFFF", +"0 c #9CB5CE", +"q c #9CADD6", +"w c #94A5D6", +"e c #8CA5D6", +"r c #8CA5CE", +"t c #8CA5C5", +"y c #849CC5", +"u c #7B9CD6", +"i c #7B9CCE", +"p c #31BDCE", +"a c #6B9CD6", +"s c #00F708", +"d c #8494AD", +"f c #7B94B5", +"g c #6B94D6", +"h c #6B9C84", +"j c #7B8CAD", +"k c #738CAD", +"l c #638CC5", +"z c #10CE42", +"x c #638CBD", +"c c #7B849C", +"v c #73849C", +"b c #6B84A5", +"n c #7B7BA5", +"m c #6B849C", +"M c #7B8C42", +"N c #5A84C5", +"B c #29AD6B", +"V c #F74A4A", +"C c #6384A5", +"Z c #5284C5", +"A c #637BA5", +"S c #637B9C", +"D c #9C637B", +"F c #6B7B5A", +"G c #637394", +"H c #52739C", +"J c #5A7384", +"K c #526B94", +"L c #426B94", +"P c #52638C", +"I c #426B7B", +"U c #5A5A8C", +"Y c #524A7B", +"T c #425273", +"R c #21636B", +"E c #106394", +"W c #106B52", +"Q c #3A4273", +"! c #31426B", +"~ c #523163", +"^ c #29426B", +"/ c #293A63", +"( c #213A63", +") c #193A63", +"_ c #193163", +"` c #19315A", +"' c #212963", +"] c #10315A", +"[ c #082952", +"{ c #FFCC33", +"} c #33FF33", +"| c #66FF33", +" . c #99FF33", +".. c #CCFF33", +"X. c #FFFF33", +"o. c #000066", +"O. c #330066", +"+. c #660066", +"@. c #990066", +"#. c #CC0066", +"$. c #FF0066", +"%. c #003366", +"&. c #333366", +"*. c #663366", +"=. c #993366", +"-. c #CC3366", +";. c #FF3366", +":. c #006666", +">. c #336666", +",. c #666666", +"<. c #996666", +"1. c #CC6666", +"2. c #009966", +"3. c #339966", +"4. c #669966", +"5. c #999966", +"6. c #CC9966", +"7. c #FF9966", +"8. c #00CC66", +"9. c #33CC66", +"0. c #99CC66", +"q. c #CCCC66", +"w. c #FFCC66", +"e. c #00FF66", +"r. c #33FF66", +"t. c #99FF66", +"y. c #CCFF66", +"u. c #FF00CC", +"i. c #CC00FF", +"p. c #009999", +"a. c #993399", +"s. c #990099", +"d. c #CC0099", +"f. c #000099", +"g. c #333399", +"h. c #660099", +"j. c #CC3399", +"k. c #FF0099", +"l. c #006699", +"z. c #336699", +"x. c #663399", +"c. c #996699", +"v. c #CC6699", +"b. c #FF3399", +"n. c #339999", +"m. c #669999", +"M. c #999999", +"N. c #CC9999", +"B. c #FF9999", +"V. c #00CC99", +"C. c #33CC99", +"Z. c #66CC66", +"A. c #99CC99", +"S. c #CCCC99", +"D. c #FFCC99", +"F. c #00FF99", +"G. c #33FF99", +"H. c #66CC99", +"J. c #99FF99", +"K. c #CCFF99", +"L. c #FFFF99", +"P. c #0000CC", +"I. c #330099", +"U. c #6600CC", +"Y. c #9900CC", +"T. c #CC00CC", +"R. c #003399", +"E. c #3333CC", +"W. c #6633CC", +"Q. c #9933CC", +"!. c #CC33CC", +"~. c #FF33CC", +"^. c #0066CC", +"/. c #3366CC", +"(. c #666699", +"). c #9966CC", +"_. c #CC66CC", +"`. c #FF6699", +"'. c #0099CC", +"]. c #3399CC", +"[. c #6699CC", +"{. c #9999CC", +"}. c #CC99CC", +"|. c #FF99CC", +" X c #00CCCC", +".X c #33CCCC", +"XX c #66CCCC", +"oX c #99CCCC", +"OX c #CCCCCC", +"+X c #FFCCCC", +"@X c #00FFCC", +"#X c #33FFCC", +"$X c #66FF99", +"%X c #99FFCC", +"&X c #CCFFCC", +"*X c #FFFFCC", +"=X c #3300CC", +"-X c #6600FF", +";X c #9900FF", +":X c #0033CC", +">X c #3333FF", +",X c #6633FF", +" g , S z R : ", +"n * c * r r y g , 6 r q S s W : ", +"n * c X 4 N u + m B I : ", +"n * c X ; a - S 5 F : ", +"n * c * r r r g - S = M : ", +"n * c X 4 N - m h J : ", +"n * c X ; a - A 9 E : ", +"n * ( ] ` ^ P l y T / / ( p L : ", +"n O > 0 f ) ! t 8 % n : ", +"U U U U U U U ' Q U U U U U U : ", +": : : : : : : : : : : : : : : : " +}; diff --git a/xpm/addressbook20.xpm b/xpm/addressbook20.xpm new file mode 100644 index 00000000..48df12d8 --- /dev/null +++ b/xpm/addressbook20.xpm @@ -0,0 +1,282 @@ +/* XPM */ +static char * addressbook20_xpm[] = { +/* columns rows colors chars-per-pixel */ +"20 20 256 2", +" c #FFFFFF", +". c #F7FFFF", +"X c #F7F7FF", +"o c #EFF7FF", +"O c #EFF7F7", +"+ c #E6EFFF", +"@ c #E6EFF7", +"# c #DEEFFF", +"$ c #DEE6F7", +"% c #DEE6EF", +"& c #D6E6F7", +"* c #FFFF00", +"= c #DEDEE6", +"- c #D6DEE6", +"; c #D6D6DE", +": c #CED6E6", +"> c None", +", c #C5D6E6", +"< c #C5CEE6", +"1 c #B5CEEF", +"2 c #C5C5C5", +"3 c #C5DE31", +"4 c #B5C5DE", +"5 c #BDC5C5", +"6 c #ADC5EF", +"7 c #B5C5CE", +"8 c #BDBDBD", +"9 c #B5BDCE", +"0 c #ADBDDE", +"q c #ADBDD6", +"w c #B5CE52", +"e c #ADB5C5", +"r c #00FFFF", +"t c #A5B5C5", +"y c #9CB5CE", +"u c #94B5DE", +"i c #9CADD6", +"p c #A5ADB5", +"a c #94ADDE", +"s c #94ADD6", +"d c #9CADBD", +"f c #8CADDE", +"g c #BD9CA5", +"h c #9CA5BD", +"j c #9CA5B5", +"k c #29D6E6", +"l c #8CA5CE", +"z c #849CCE", +"x c #6BA5C5", +"c c #739CDE", +"v c #00FF00", +"b c #739CD6", +"n c #7B94CE", +"m c #8494AD", +"M c #7394CE", +"N c #7B94B5", +"B c #4AB584", +"V c #848CB5", +"C c #6B94CE", +"Z c #6394D6", +"A c #6394CE", +"S c #7B8CAD", +"D c #6B8CC5", +"F c #738CAD", +"G c #5294B5", +"H c #6B84C5", +"J c #7384A5", +"K c #73849C", +"L c #738494", +"P c #FF4A4A", +"I c #FF4A42", +"U c #737B8C", +"Y c #637BAD", +"T c #527BBD", +"R c #637394", +"E c #637352", +"W c #5A6B8C", +"Q c #526B9C", +"! c #63638C", +"~ c #5A734A", +"^ c #4A6B9C", +"/ c #526B63", +"( c #0884A5", +") c #526384", +"_ c #52637B", +"` c #4A6B5A", +"' c #52636B", +"] c #525A8C", +"[ c #525A7B", +"{ c #426363", +"} c #4A5A7B", +"| c #425A8C", +" . c #196B94", +".. c #3A5A8C", +"X. c #3A5A84", +"o. c #087B4A", +"O. c #21636B", +"+. c #634263", +"@. c #3A527B", +"#. c #424A84", +"$. c #315284", +"%. c #295284", +"&. c #3A4A6B", +"*. c #42427B", +"=. c #424273", +"-. c #294A84", +";. c #3A3A73", +":. c #194284", +">. c #104A63", +",. c #213A6B", +"<. c #31316B", +"1. c #21315A", +"2. c #212163", +"3. c #08295A", +"4. c #082152", +"5. c #101952", +"6. c #CC9966", +"7. c #FF9966", +"8. c #00CC66", +"9. c #33CC66", +"0. c #99CC66", +"q. c #CCCC66", +"w. c #FFCC66", +"e. c #00FF66", +"r. c #33FF66", +"t. c #99FF66", +"y. c #CCFF66", +"u. c #FF00CC", +"i. c #CC00FF", +"p. c #009999", +"a. c #993399", +"s. c #990099", +"d. c #CC0099", +"f. c #000099", +"g. c #333399", +"h. c #660099", +"j. c #CC3399", +"k. c #FF0099", +"l. c #006699", +"z. c #336699", +"x. c #663399", +"c. c #996699", +"v. c #CC6699", +"b. c #FF3399", +"n. c #339999", +"m. c #669999", +"M. c #999999", +"N. c #CC9999", +"B. c #FF9999", +"V. c #00CC99", +"C. c #33CC99", +"Z. c #66CC66", +"A. c #99CC99", +"S. c #CCCC99", +"D. c #FFCC99", +"F. c #00FF99", +"G. c #33FF99", +"H. c #66CC99", +"J. c #99FF99", +"K. c #CCFF99", +"L. c #FFFF99", +"P. c #0000CC", +"I. c #330099", +"U. c #6600CC", +"Y. c #9900CC", +"T. c #CC00CC", +"R. c #003399", +"E. c #3333CC", +"W. c #6633CC", +"Q. c #9933CC", +"!. c #CC33CC", +"~. c #FF33CC", +"^. c #0066CC", +"/. c #3366CC", +"(. c #666699", +"). c #9966CC", +"_. c #CC66CC", +"`. c #FF6699", +"'. c #0099CC", +"]. c #3399CC", +"[. c #6699CC", +"{. c #9999CC", +"}. c #CC99CC", +"|. c #FF99CC", +" X c #00CCCC", +".X c #33CCCC", +"XX c #66CCCC", +"oX c #99CCCC", +"OX c #CCCCCC", +"+X c #FFCCCC", +"@X c #00FFCC", +"#X c #33FFCC", +"$X c #66FF99", +"%X c #99FFCC", +"&X c #CCFFCC", +"*X c #FFFFCC", +"=X c #3300CC", +"-X c #6600FF", +";X c #9900FF", +":X c #0033CC", +">X c #3333FF", +",X c #6633FF", +" > > > > > > > > > > > > > > > > > > > ", +"> > > > > > > > > > > > > > > > > > > > ", +"> > U $.| | ^ S 2 > p W | | @.L > > > > ", +"8 5 R - < Y j S O - ) g e > > ", +"! V K - % a Q # - +.P <.> > ", +"! & K - 0 z n D C b f n n z q +.P <.> > ", +"! & K - % M A 1 - %.G #.> > ", +"! & K - % u b # - o.v >.> > ", +"! & K - 0 z n H M b 6 z n z q o.v >.> > ", +"! & K - X - M A a - O.B @.> > ", +"! & K - X % u b # - ` 3 / > > ", +"! & K - 0 l i 4 u b # - ~ * E > > ", +"! & K - X o $ s T b # - { w ' > > ", +"! & K - % f b # - .k -.> > ", +"! & K m d t 7 , u b # ; 9 9 h ( r :.> > ", +"! & h _ _ [ &.4.$.A ,.1.} _ _ F x ] > > ", +"! @ , y N _ 3._ N y , @ ! > > ", +"*.*.*.*.*.*.*.*.;.5.*.*.*.*.*.*.*.2.> > ", +"> > > > > > > > > > > > > > > > > > > > ", +"> > > > > > > > > > > > > > > > > > > > " +}; diff --git a/xpm/bitcoin.xpm b/xpm/bitcoin.xpm new file mode 100644 index 00000000..166d5aa6 --- /dev/null +++ b/xpm/bitcoin.xpm @@ -0,0 +1,304 @@ +/* XPM */ +static char * bitcoin_xpm[] = { +/* columns rows colors chars-per-pixel */ +"48 48 250 2", +" c #725203", +". c #795603", +"X c #7D5903", +"o c #76560B", +"O c #77590E", +"+ c #795A0D", +"@ c #7B5D14", +"# c #7C5F18", +"$ c #7D6019", +"% c #825D05", +"& c #856007", +"* c #86620B", +"= c #8B660B", +"- c #8E690E", +"; c #906A0F", +": c #8F6B17", +"> c #83641C", +", c #8D6C1E", +"< c #926C11", +"1 c #967014", +"2 c #997215", +"3 c #9C761B", +"4 c #9E791D", +"5 c #A37C1E", +"6 c #816520", +"7 c #876A25", +"8 c #8E6E22", +"9 c #866A29", +"0 c #896E2C", +"q c #8E7020", +"w c #937324", +"e c #997722", +"r c #9E7B25", +"t c #94762B", +"y c #967828", +"u c #9A7B2D", +"i c #8B7131", +"p c #9E7E31", +"a c #947839", +"s c #A37D22", +"d c #A68125", +"f c #AA8325", +"g c #AE8827", +"h c #A6832D", +"j c #AA852B", +"k c #AD892B", +"l c #B08727", +"z c #B28827", +"x c #B08729", +"c c #B38B2C", +"v c #B88E2F", +"b c #B8902D", +"n c #A38334", +"m c #A98632", +"M c #AB8A34", +"N c #A4873C", +"B c #A78A3D", +"V c #AC8B3C", +"C c #B38D32", +"Z c #BA8F30", +"A c #B28E3C", +"S c #B69332", +"D c #BC9433", +"F c #BF9832", +"G c #B4923C", +"H c #BA963D", +"J c #B7993E", +"K c #BE9A3B", +"L c #C1932F", +"P c #C39732", +"I c #C49935", +"U c #C59C3A", +"Y c #C99E3D", +"T c #C2A13F", +"R c #CDA23F", +"E c #9D8342", +"W c #AB8C43", +"Q c #B28E40", +"! c #AE9144", +"~ c #AE914A", +"^ c #B49445", +"/ c #BC9B44", +"( c #B3964D", +") c #B5994C", +"_ c #BD9B4A", +"` c #A98F50", +"' c #B19553", +"] c #B59A54", +"[ c #BD9F51", +"{ c #B59B5C", +"} c #B89D5C", +"| c #BEA155", +" . c #BDA35D", +".. c #B59C61", +"X. c #B99F66", +"o. c #BCA363", +"O. c #BDA56C", +"+. c #BCA571", +"@. c #BDA873", +"#. c #BFAA78", +"$. c #C49D43", +"%. c #C99F45", +"&. c #C29E4B", +"*. c #C5A144", +"=. c #CCA244", +"-. c #C5A44B", +";. c #CAA54B", +":. c #C8A84C", +">. c #D0A644", +",. c #D3AA44", +"<. c #D3AC4C", +"1. c #D8AD4D", +"2. c #DAB046", +"3. c #DCB24E", +"4. c #C3A454", +"5. c #CBA751", +"6. c #CCAA53", +"7. c #C1A65B", +"8. c #C8A75A", +"9. c #CBAC5B", +"0. c #D0A650", +"q. c #D2AC53", +"w. c #DAAD54", +"e. c #D3AD5C", +"r. c #CFB259", +"t. c #D4B156", +"y. c #DDB454", +"u. c #D4B25C", +"i. c #DAB65A", +"p. c #D7B95F", +"a. c #DEBA5E", +"s. c #E2B555", +"d. c #E5BA53", +"f. c #E1B55A", +"g. c #E5BC5C", +"h. c #EABF5D", +"j. c #C1A761", +"k. c #C4AA63", +"l. c #CBAE63", +"z. c #CBB166", +"x. c #CBB26C", +"c. c #D4B263", +"v. c #DAB462", +"b. c #D6B864", +"n. c #DCB965", +"m. c #D3B669", +"M. c #DCB768", +"N. c #D4BA6E", +"B. c #DCBB6C", +"V. c #CDB672", +"C. c #D2B972", +"Z. c #DBBE72", +"A. c #E4BC62", +"S. c #E9BE62", +"D. c #E2BD6C", +"F. c #E0BF72", +"G. c #E6C05E", +"H. c #EFC05D", +"J. c #F0C15B", +"K. c #DFC167", +"L. c #D7C069", +"P. c #DDC36D", +"I. c #DBC376", +"U. c #D4C17B", +"Y. c #DAC17B", +"T. c #D8C878", +"R. c #E4C362", +"E. c #EBC364", +"W. c #E3C865", +"Q. c #EDC866", +"!. c #E4C36A", +"~. c #E9C66B", +"^. c #ECCA6B", +"/. c #F1C564", +"(. c #F8C765", +"). c #F5CB66", +"_. c #F8CC67", +"`. c #F6CC6A", +"'. c #F9CD6B", +"]. c #EED26A", +"[. c #F2D06F", +"{. c #FBD26D", +"}. c #E4C374", +"|. c #EBC474", +" X c #E1C972", +".X c #EDCD72", +"XX c #E4C57A", +"oX c #E9C67C", +"OX c #E5C87C", +"+X c #EACA7D", +"@X c #F2CC74", +"#X c #FBCF71", +"$X c #EED174", +"%X c #ECD37B", +"&X c #F4D274", +"*X c #FDD473", +"=X c #FFD975", +"-X c #F4D57C", +";X c #FCD57A", +":X c #F3DA7C", +">X c #FEDB7C", +",X c #FFE37D", +"X=XQ.s.=.v 5 1 < E HXHXHXHXHXHXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHXHXHX' g f F d.).).{.{.=X=X=X{.{.{.`.`.`.).g.U f 2 * a HXHXHXHXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHXHXV f b G.J.{.{.{.*X=X,X=X*X{.`.).`.).).{.`.{./.U 5 ; + HXHXHXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHX} h g 1.)._._.{.,X*X=X,X{.{.)._.).).`.`.`.{.*X*X*X`.y.g 2 & $ HXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHX{ j c G.).{.{.{.=X,X{.{.J.d.2.R 2.,.3.g.`.&X;X;X;X&X[.{.`.I 3 & + HXHXHXHXHXHXHXHX", +"HXHXHXHXHXHX{ d D /.{.{.*X=X=X*X).3.R I I I P F F U $.;.n.-XrXrX;X&X;X>XdX:.4 X o HXHXHXHXHXHXHX", +"HXHXHXHXHX..j v /.*X=X=X=X=X`.1.R R R R I I I P K U *.e.D.|.}.+XrXqXhXhXdXfX:.4 X o HXHXHXHXHXHX", +"HXHXHXHXHXh v `.{.>X,X*X{.g.>.Y R I I I I I I U U ;.t.D.|.oXB.z.F.kXvXcXjXjXjX/ 1 . + HXHXHXHXHX", +"HXHXHXHXV g Q.=X=X>X>X'.>.Y Y U R I I I P P I U U <.n.~.}.F.XXXX}.%XbXbXcXcXcXsXc = # HXHXHXHX", +"HXHXHX} j t.>X>X>X*X'.>.U U I U P U U I P P I U T 6.M.D.oX5XwXeXeXqX0XvXbXbXcXjXW.5 % HXHXHXHX", +"HXHXHXM G hXhXqX>X*X<.U U U I I I I I I D D U T T -.9.B.3XpXpXzXgXqX:X:XbXbXcXjXfXT < o HXHXHX", +"HXHX} k XlXkXkXrXA.$.D Z Z Z v v v b b v D U U *.-.9.B.OX2XOXI.P.L.K.W.$XbXcXjX,X].d % 9 HXHX", +"HXHXV J xXxXxXxXrX5.&.A M m m m h h n s 5 g S K *.:.8.4.| k.x.C.N.z.7.) :.$XjXfX,X,XT ; o HXHX", +"HXHXM L.vXxXxXxXF._ MXCXCXCXCXCXCXCXCXmXY.h g K *./ ^ Y.mXCXCXCXCXCXCXVXZ.4.hXfX,X,XW.4 X HXHX", +"HX] k gXxXxXxXgXe.V MXCXCXCXCXVXCXCXCXCXCXyXh D G [ mXCXCXCXCXCXCXCXCXCX4XG ~.fX,X,X,Xg & $ HX", +"HXV J vXxXxXxX6Xe.V MXCXCXCXCXk.N VXCXCXCXVX| h ^ MXCXCXCXCXuX( n V l.mX4XA y.fX,X,X=XT ; HX", +"HXk r.xXxXxXxX|.v.V MXCXCXCXCXo.> 4XCXCXCXCXx.w tXCXCXCXCXnXn V / / M V m.&.t.=X,X,X=Xy.2 o HX", +"HXk P.xXvXxXxX|.M.Q nXCXCXCXCXj.w X>X=XH.5 X $ ", +"o.k %XvXbXBXkX|.D.Q nXCXCXCXCXj., 7XCXCXCXCX~ k.CXCXCXCXCXV &.n.R.g.G.g.S.S.S.(.qX*X=X`.5 X $ ", +" .k 0XvXvXvXrX@XD.^ VXCXCXCXCXx.~ VXCXCXCXX=X*X*Xd X $ ", +"{ k sXcXvXBXeX@XD.( nXCXCXCXCXCXCXCXCXVXo., u T.CXCXCXVXmXn t.E.g.g.h.S.g.S.f.A.>X;X*X*Xf & o ", +"{ k sXjXlXcX0X~.n.^ MXCXCXCXCXCXCXCXCXCXCXU.t U.CXCXCXCXmXM p.~.W.g.s.s.s.s.f.A.>X*X*X*Xj % ", +"o.k :XjXlXlX-XD.v.A MXCXCXCXCXj.t mXCXCXCXCX .x.CXCXCXCXVXV p.$X^.E.g.s.w.w.w.A.9X;X*X*Xf X ", +"o.g ].dXjXhX-Xn.e.V MXCXCXCXCXj.8 X*X#X5 X O ", +"HXj K.dXdXhX9Xv.9.M MXCXCXCXCXk.a Z.CXCXCXCX7Xu CXCXCXCXCXV./ !.$X~.f.5.%.0.q.S.>X*X*XE.5 X # ", +"HXj t.dX,XdXdXi.6.N MXCXCXCXCXo.q XqXqXqX!.6.m MXCXCXCXCXV.' VXCXCXZXCXk.! ] VXCXCXCXCXVXC.[ 7.Z.VX2Xx %.#X#X'.'.%.- o 6 ", +"HX( g &X>X>XdX-X5.M MXCXCXCXCXCXCXCXCXCXCX7X) m.9. .MXCXCXCXCXCXCXCXCXCX2Xs 1.'.`.'.'.x % HX", +"HXO.j y.>X>X>X>X6.! zXMXMXMXMXMXMXMXMXuXx.( N.8X6Xz.) C.uXCXCXCXCXCXVX7X4.c h.'.(.(.s.5 X HX", +"HXHXj H &X=X:X>X~./ V h y u n n n N W ( 7.Z.8XpXpX+Xm.| V V ^ ) ( m e 3 s R (.(.'.'.Y ; . > HX", +"HXHX} f G.&X&X&X:Xt._ / ) _ 4.8.l.m.B.Z.2XwXpXeXwX6X+XP.c.8.-./ C x x z P J.(.'.(./.5 % HXHX", +"HXHXHXh D &X&X&X&X@X:.4.5.9.c.m.F.OX+XwXwXwXwX6X3XOX}.D.D.v.w.%.Y I P P d.(.(.'.'.=.< . + HXHX", +"HXHXHX] d y.[.&X&X&X~.:.4.9.e.M.B.}.oX3X5X3X3X+X}.F.M.e.0.0.0.%.%.Y Y s.#X#X#X#XS.s % HXHXHX", +"HXHXHXHXm g `.@X&X&X&X~.5.6.e.b.B.}.XX+X3XOX}.I.F.F.D.e.e.e.e.e.q.0.A.;X;X#X-X@XZ = o + HXHXHX", +"HXHXHXHX..j D @X&X&X&X&X@Xp.u.M.D.}.XXOXOXZ.Z.XX1XOXoXoXF.F.F.M.D.6XrXqX9X9X9X%.1 . HXHXHXHX", +"HXHXHXHXHX' f $.&X&X&X>XqXqX XB.D.!.XXXXZ.XXOX5XwXwXwXwXiXwXwXnXVXZXBXxXzXxXb.r X $ HXHXHXHX", +"HXHXHXHXHXHX~ j ;.qXqXqXqXrXkXrX+XD.Z.Z.XX1X2X5X5X5XwXwXiXnXCXGXGXFXFXSXAXT.s % @ HXHXHXHXHX", +"HXHXHXHXHXHXHX~ j -.0XrXzXxXzXzXzXzXwX3XXXXXOX1X2X5XpXmXAXFXGXGXGXFXFXSXL.r % O HXHXHXHXHXHX", +"HXHXHXHXHXHXHXHX! j / gXSXSXZXxXzXzXkXxXzXBXBXBXZXCXAXAXAXAXSXSXSXSXNX| 3 & O HXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHX} n V U.DXSXBXzXkXkXkXxXxXxXBXxXBXBXCXZXZXZXAXAXU.M < . @ HXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHXHXHXV d G Z.pXzXkXzXjXkXkXxXzXxXxXxXBXBXZXNXT.G 3 * . 9 HXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHXHXHXHX@.u s k -.K.6XhXjXhXkXlXzXzXzXeXOX9.k 3 = X O HXHXHXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHXHXHXHXHXHX..w < s j k K -.;.:.-./ C j 4 < & . O HXHXHXHXHXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXE * = - - < < - = & X . 0 HXHXHXHXHXHXHXHXHXHXHXHXHXHXHX", +"HXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXi 7 7 @ o o O > 0 i HXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHX" +}; diff --git a/xpm/check.xpm b/xpm/check.xpm new file mode 100644 index 00000000..8f0b9d28 --- /dev/null +++ b/xpm/check.xpm @@ -0,0 +1,41 @@ +/* XPM */ +static char * check_xpm[] = { +/* columns rows colors chars-per-pixel */ +"32 32 3 1", +" c #008000", +". c #00FF00", +"X c None", +/* pixels */ +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXX XXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXX . XXXXXXXXXXX", +"XXXXXXXXXXXXXXXX .. XXXXXXXXXXXX", +"XXXXXXXXXXXXXXXX . XXXXXXXXXXXX", +"XXXXXXXXXXXXXXX .. XXXXXXXXXXXXX", +"XXXXXXXXXXX XX . XXXXXXXXXXXXX", +"XXXXXXXXXXX . .. XXXXXXXXXXXXXX", +"XXXXXXXXXXX .. . XXXXXXXXXXXXXX", +"XXXXXXXXXXXX ... XXXXXXXXXXXXXXX", +"XXXXXXXXXXXXX . XXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXX XXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX", +"XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX" +}; diff --git a/xpm/send16.xpm b/xpm/send16.xpm new file mode 100644 index 00000000..1eeceb4e --- /dev/null +++ b/xpm/send16.xpm @@ -0,0 +1,278 @@ +/* XPM */ +static char * send16_xpm[] = { +/* columns rows colors chars-per-pixel */ +"16 16 256 2", +" c #ADF7AD", +". c #9CFF9C", +"X c None", +"o c #ADEFAD", +"O c #94FF94", +"+ c #D6CECE", +"@ c #8CFF8C", +"# c #CECECE", +"$ c #CECEC5", +"% c #84FF84", +"& c #CEC5C5", +"* c #73FF73", +"= c #C5C5C5", +"- c #6BFF6B", +"; c #73F773", +": c #C5BDBD", +"> c #6BF76B", +", c #BDBDBD", +"< c #63F763", +"1 c #B5B5B5", +"2 c #52F752", +"3 c #42FF42", +"4 c #3AFF3A", +"5 c #ADADAD", +"6 c #ADADA5", +"7 c #4AEF4A", +"8 c #29FF29", +"9 c #A5A5A5", +"0 c #42E642", +"q c #9CA59C", +"w c #3AE63A", +"e c #10FF10", +"r c #08FF08", +"t c #949C94", +"y c #00FF00", +"u c #00F700", +"i c #8C948C", +"p c #00EF00", +"a c #08E608", +"s c #10DE10", +"d c #00E600", +"f c #00DE00", +"g c #19C519", +"h c #00CE00", +"j c #00C500", +"k c #008C00", +"l c #008400", +"z c #669900", +"x c #999900", +"c c #CC9900", +"v c #FF9900", +"b c #00CC00", +"n c #33CC00", +"m c #66CC00", +"M c #99CC00", +"N c #CCCC00", +"B c #FFCC00", +"V c #66FF00", +"C c #99FF00", +"Z c #CCFF00", +"A c #000033", +"S c #330033", +"D c #660033", +"F c #990033", +"G c #CC0033", +"H c #FF0033", +"J c #003333", +"K c #333333", +"L c #663333", +"P c #993333", +"I c #CC3333", +"U c #FF3333", +"Y c #006633", +"T c #336633", +"R c #666633", +"E c #996633", +"W c #CC6633", +"Q c #FF6633", +"! c #009933", +"~ c #339933", +"^ c #669933", +"/ c #999933", +"( c #CC9933", +") c #FF9933", +"_ c #00CC33", +"` c #33CC33", +"' c #66CC33", +"] c #99CC33", +"[ c #CCCC33", +"{ c #FFCC33", +"} c #33FF33", +"| c #66FF33", +" . c #99FF33", +".. c #CCFF33", +"X. c #FFFF33", +"o. c #000066", +"O. c #330066", +"+. c #660066", +"@. c #990066", +"#. c #CC0066", +"$. c #FF0066", +"%. c #003366", +"&. c #333366", +"*. c #663366", +"=. c #993366", +"-. c #CC3366", +";. c #FF3366", +":. c #006666", +">. c #336666", +",. c #666666", +"<. c #996666", +"1. c #CC6666", +"2. c #009966", +"3. c #339966", +"4. c #669966", +"5. c #999966", +"6. c #CC9966", +"7. c #FF9966", +"8. c #00CC66", +"9. c #33CC66", +"0. c #99CC66", +"q. c #CCCC66", +"w. c #FFCC66", +"e. c #00FF66", +"r. c #33FF66", +"t. c #99FF66", +"y. c #CCFF66", +"u. c #FF00CC", +"i. c #CC00FF", +"p. c #009999", +"a. c #993399", +"s. c #990099", +"d. c #CC0099", +"f. c #000099", +"g. c #333399", +"h. c #660099", +"j. c #CC3399", +"k. c #FF0099", +"l. c #006699", +"z. c #336699", +"x. c #663399", +"c. c #996699", +"v. c #CC6699", +"b. c #FF3399", +"n. c #339999", +"m. c #669999", +"M. c #999999", +"N. c #CC9999", +"B. c #FF9999", +"V. c #00CC99", +"C. c #33CC99", +"Z. c #66CC66", +"A. c #99CC99", +"S. c #CCCC99", +"D. c #FFCC99", +"F. c #00FF99", +"G. c #33FF99", +"H. c #66CC99", +"J. c #99FF99", +"K. c #CCFF99", +"L. c #FFFF99", +"P. c #0000CC", +"I. c #330099", +"U. c #6600CC", +"Y. c #9900CC", +"T. c #CC00CC", +"R. c #003399", +"E. c #3333CC", +"W. c #6633CC", +"Q. c #9933CC", +"!. c #CC33CC", +"~. c #FF33CC", +"^. c #0066CC", +"/. c #3366CC", +"(. c #666699", +"). c #9966CC", +"_. c #CC66CC", +"`. c #FF6699", +"'. c #0099CC", +"]. c #3399CC", +"[. c #6699CC", +"{. c #9999CC", +"}. c #CC99CC", +"|. c #FF99CC", +" X c #00CCCC", +".X c #33CCCC", +"XX c #66CCCC", +"oX c #99CCCC", +"OX c #CCCCCC", +"+X c #FFCCCC", +"@X c #00FFCC", +"#X c #33FFCC", +"$X c #66FF99", +"%X c #99FFCC", +"&X c #CCFFCC", +"*X c #FFFFCC", +"=X c #3300CC", +"-X c #6600FF", +";X c #9900FF", +":X c #0033CC", +">X c #3333FF", +",X c #6633FF", +" c #6BF76B", +", c #BDBDBD", +"< c #63F763", +"1 c #B5B5B5", +"2 c #52F752", +"3 c #42FF42", +"4 c #3AFF3A", +"5 c #ADADAD", +"6 c #ADADA5", +"7 c #4AEF4A", +"8 c #29FF29", +"9 c #A5A5A5", +"0 c #42E642", +"q c #9CA59C", +"w c #3AE63A", +"e c #10FF10", +"r c #08FF08", +"t c #949C94", +"y c #00FF00", +"u c #00F700", +"i c #8C948C", +"p c #00EF00", +"a c #08E608", +"s c #10DE10", +"d c #00E600", +"f c #00DE00", +"g c #19C519", +"h c #00CE00", +"j c #00C500", +"k c #008C00", +"l c #008400", +"z c #669900", +"x c #999900", +"c c #CC9900", +"v c #FF9900", +"b c #00CC00", +"n c #33CC00", +"m c #66CC00", +"M c #99CC00", +"N c #CCCC00", +"B c #FFCC00", +"V c #66FF00", +"C c #99FF00", +"Z c #CCFF00", +"A c #000033", +"S c #330033", +"D c #660033", +"F c #990033", +"G c #CC0033", +"H c #FF0033", +"J c #003333", +"K c #333333", +"L c #663333", +"P c #993333", +"I c #CC3333", +"U c #FF3333", +"Y c #006633", +"T c #336633", +"R c #666633", +"E c #996633", +"W c #CC6633", +"Q c #FF6633", +"! c #009933", +"~ c #339933", +"^ c #669933", +"/ c #999933", +"( c #CC9933", +") c #FF9933", +"_ c #00CC33", +"` c #33CC33", +"' c #66CC33", +"] c #99CC33", +"[ c #CCCC33", +"{ c #FFCC33", +"} c #33FF33", +"| c #66FF33", +" . c #99FF33", +".. c #CCFF33", +"X. c #FFFF33", +"o. c #000066", +"O. c #330066", +"+. c #660066", +"@. c #990066", +"#. c #CC0066", +"$. c #FF0066", +"%. c #003366", +"&. c #333366", +"*. c #663366", +"=. c #993366", +"-. c #CC3366", +";. c #FF3366", +":. c #006666", +">. c #336666", +",. c #666666", +"<. c #996666", +"1. c #CC6666", +"2. c #009966", +"3. c #339966", +"4. c #669966", +"5. c #999966", +"6. c #CC9966", +"7. c #FF9966", +"8. c #00CC66", +"9. c #33CC66", +"0. c #99CC66", +"q. c #CCCC66", +"w. c #FFCC66", +"e. c #00FF66", +"r. c #33FF66", +"t. c #99FF66", +"y. c #CCFF66", +"u. c #FF00CC", +"i. c #CC00FF", +"p. c #009999", +"a. c #993399", +"s. c #990099", +"d. c #CC0099", +"f. c #000099", +"g. c #333399", +"h. c #660099", +"j. c #CC3399", +"k. c #FF0099", +"l. c #006699", +"z. c #336699", +"x. c #663399", +"c. c #996699", +"v. c #CC6699", +"b. c #FF3399", +"n. c #339999", +"m. c #669999", +"M. c #999999", +"N. c #CC9999", +"B. c #FF9999", +"V. c #00CC99", +"C. c #33CC99", +"Z. c #66CC66", +"A. c #99CC99", +"S. c #CCCC99", +"D. c #FFCC99", +"F. c #00FF99", +"G. c #33FF99", +"H. c #66CC99", +"J. c #99FF99", +"K. c #CCFF99", +"L. c #FFFF99", +"P. c #0000CC", +"I. c #330099", +"U. c #6600CC", +"Y. c #9900CC", +"T. c #CC00CC", +"R. c #003399", +"E. c #3333CC", +"W. c #6633CC", +"Q. c #9933CC", +"!. c #CC33CC", +"~. c #FF33CC", +"^. c #0066CC", +"/. c #3366CC", +"(. c #666699", +"). c #9966CC", +"_. c #CC66CC", +"`. c #FF6699", +"'. c #0099CC", +"]. c #3399CC", +"[. c #6699CC", +"{. c #9999CC", +"}. c #CC99CC", +"|. c #FF99CC", +" X c #00CCCC", +".X c #33CCCC", +"XX c #66CCCC", +"oX c #99CCCC", +"OX c #CCCCCC", +"+X c #FFCCCC", +"@X c #00FFCC", +"#X c #33FFCC", +"$X c #66FF99", +"%X c #99FFCC", +"&X c #CCFFCC", +"*X c #FFFFCC", +"=X c #3300CC", +"-X c #6600FF", +";X c #9900FF", +":X c #0033CC", +">X c #3333FF", +",X c #6633FF", +" c #73FF73", +", c #C5C5C5", +"< c #C5C5BD", +"1 c #6BFF6B", +"2 c #BDC5B5", +"3 c #63FF63", +"4 c #6BF76B", +"5 c #BDBDBD", +"6 c #BDBDB5", +"7 c #5AFF5A", +"8 c #63F763", +"9 c #B5BDB5", +"0 c #B5BDAD", +"q c #52FF52", +"w c #BDB5B5", +"e c #5AF75A", +"r c #B5B5B5", +"t c #B5B5AD", +"y c #52F752", +"u c #42FF42", +"i c #52EF52", +"p c #ADADAD", +"a c #ADADA5", +"s c #4AEF4A", +"d c #31FF31", +"f c #29FF29", +"g c #A5A5A5", +"h c #21FF21", +"j c #5AD65A", +"k c #42E642", +"l c #94AD94", +"z c #4ADE4A", +"x c #3AE63A", +"c c #5ACE5A", +"v c #10FF10", +"b c #9C9C9C", +"n c #31E631", +"m c #08FF08", +"M c #949C94", +"N c #84A584", +"B c #00FF00", +"V c #3AD63A", +"C c #52C552", +"Z c #00F700", +"A c #8C948C", +"S c #849484", +"D c #00EF00", +"F c #739C73", +"G c #08E608", +"H c #4AB54A", +"J c #31C531", +"K c #00E600", +"L c #739473", +"P c #00DE00", +"I c #63945A", +"U c #6B8C6B", +"Y c #00D600", +"T c #42A542", +"R c #638C63", +"E c #00CE00", +"W c #21B521", +"Q c #5A8C5A", +"! c #00C500", +"~ c #528C52", +"^ c #3A9C3A", +"/ c #4A8C4A", +"( c #00BD00", +") c #319431", +"_ c #219C21", +"` c #318C31", +"' c #3A843A", +"] c #219421", +"[ c #298C29", +"{ c #318431", +"} c #218C21", +"| c #218C19", +" . c #198C19", +".. c #218421", +"X. c #297B29", +"o. c #198419", +"O. c #217B21", +"+. c #108410", +"@. c #197B19", +"#. c #CC0066", +"$. c #FF0066", +"%. c #003366", +"&. c #333366", +"*. c #663366", +"=. c #993366", +"-. c #CC3366", +";. c #FF3366", +":. c #006666", +">. c #336666", +",. c #666666", +"<. c #996666", +"1. c #CC6666", +"2. c #009966", +"3. c #339966", +"4. c #669966", +"5. c #999966", +"6. c #CC9966", +"7. c #FF9966", +"8. c #00CC66", +"9. c #33CC66", +"0. c #99CC66", +"q. c #CCCC66", +"w. c #FFCC66", +"e. c #00FF66", +"r. c #33FF66", +"t. c #99FF66", +"y. c #CCFF66", +"u. c #FF00CC", +"i. c #CC00FF", +"p. c #009999", +"a. c #993399", +"s. c #990099", +"d. c #CC0099", +"f. c #000099", +"g. c #333399", +"h. c #660099", +"j. c #CC3399", +"k. c #FF0099", +"l. c #006699", +"z. c #336699", +"x. c #663399", +"c. c #996699", +"v. c #CC6699", +"b. c #FF3399", +"n. c #339999", +"m. c #669999", +"M. c #999999", +"N. c #CC9999", +"B. c #FF9999", +"V. c #00CC99", +"C. c #33CC99", +"Z. c #66CC66", +"A. c #99CC99", +"S. c #CCCC99", +"D. c #FFCC99", +"F. c #00FF99", +"G. c #33FF99", +"H. c #66CC99", +"J. c #99FF99", +"K. c #CCFF99", +"L. c #FFFF99", +"P. c #0000CC", +"I. c #330099", +"U. c #6600CC", +"Y. c #9900CC", +"T. c #CC00CC", +"R. c #003399", +"E. c #3333CC", +"W. c #6633CC", +"Q. c #9933CC", +"!. c #CC33CC", +"~. c #FF33CC", +"^. c #0066CC", +"/. c #3366CC", +"(. c #666699", +"). c #9966CC", +"_. c #CC66CC", +"`. c #FF6699", +"'. c #0099CC", +"]. c #3399CC", +"[. c #6699CC", +"{. c #9999CC", +"}. c #CC99CC", +"|. c #FF99CC", +" X c #00CCCC", +".X c #33CCCC", +"XX c #66CCCC", +"oX c #99CCCC", +"OX c #CCCCCC", +"+X c #FFCCCC", +"@X c #00FFCC", +"#X c #33FFCC", +"$X c #66FF99", +"%X c #99FFCC", +"&X c #CCFFCC", +"*X c #FFFFCC", +"=X c #3300CC", +"-X c #6600FF", +";X c #9900FF", +":X c #0033CC", +">X c #3333FF", +",X c #6633FF", +" Date: Fri, 6 Nov 2009 05:50:05 +0000 Subject: [PATCH 022/133] got rid of CheckForShutdown, replaced some thread-unsafe wxWidgets calls, Linux fixes, socket send MSG_NOSIGNAL, bind INADDR_ANY, works reliably on Linux now except if wxMessageBox is used in a thread other than the GUI thread git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@33 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 2 +- irc.cpp | 9 ++- main.cpp | 67 +++++++++++------- main.h | 2 +- net.cpp | 193 ++++++++++++++++++++++++++------------------------ net.h | 2 +- ui.cpp | 63 ++++++++++++---- ui.h | 3 +- uibase.cpp | 60 ++-------------- uiproject.fbp | 68 +++++++++--------- util.cpp | 12 ++-- util.h | 14 +++- 12 files changed, 260 insertions(+), 235 deletions(-) diff --git a/db.cpp b/db.cpp index b702b0cb..12355926 100644 --- a/db.cpp +++ b/db.cpp @@ -550,7 +550,7 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) //printf("LoadWallet %s\n", wtx.GetHash().ToString().c_str()); //printf(" %12I64d %s %s %s\n", // wtx.vout[0].nValue, - // DateTimeStr(wtx.nTime).c_str(), + // DateTimeStrFormat("%x %H:%M:%S", wtx.nTime).c_str(), // wtx.hashBlock.ToString().substr(0,14).c_str(), // wtx.mapValue["message"].c_str()); } diff --git a/irc.cpp b/irc.cpp index 4d4ed0f4..3b232cae 100644 --- a/irc.cpp +++ b/irc.cpp @@ -167,9 +167,12 @@ void ThreadIRCSeed(void* parg) while (!fShutdown) { CAddress addrConnect("216.155.130.130:6667"); - struct hostent* phostent = gethostbyname("chat.freenode.net"); - if (phostent && phostent->h_addr_list && phostent->h_addr_list[0]) - addrConnect = CAddress(*(u_long*)phostent->h_addr_list[0], htons(6667)); + if (!(fUseProxy && addrProxy.port == htons(9050))) + { + struct hostent* phostent = gethostbyname("chat.freenode.net"); + if (phostent && phostent->h_addr_list && phostent->h_addr_list[0]) + addrConnect = CAddress(*(u_long*)phostent->h_addr_list[0], htons(6667)); + } SOCKET hSocket; if (!ConnectSocket(addrConnect, hSocket)) diff --git a/main.cpp b/main.cpp index 2119495e..cc22bad6 100644 --- a/main.cpp +++ b/main.cpp @@ -1361,15 +1361,24 @@ bool ScanMessageStart(Stream& s) bool CheckDiskSpace(int64 nAdditionalBytes) { - wxLongLong nFreeBytesAvailable = 0; - if (!wxGetDiskSpace(GetDataDir(), NULL, &nFreeBytesAvailable)) - { - printf("ERROR: wxGetDiskSpace() failed\n"); +#ifdef __WXMSW__ + uint64 nFreeBytesAvailable = 0; // bytes available to caller + uint64 nTotalNumberOfBytes = 0; // bytes on disk + uint64 nTotalNumberOfFreeBytes = 0; // free bytes on disk + if (!GetDiskFreeSpaceEx(GetDataDir().c_str(), + (PULARGE_INTEGER)&nFreeBytesAvailable, + (PULARGE_INTEGER)&nTotalNumberOfBytes, + (PULARGE_INTEGER)&nTotalNumberOfFreeBytes)) + { + printf("ERROR: GetDiskFreeSpaceEx() failed\n"); return true; } +#else + uint64 nFreeBytesAvailable = filesystem::space(GetDataDir()).available; +#endif // Check for 15MB because database could create another 10MB log file at any time - if (nFreeBytesAvailable.GetValue() < (int64)15000000 + nAdditionalBytes) + if (nFreeBytesAvailable < (int64)15000000 + nAdditionalBytes) { fShutdown = true; wxMessageBox("Warning: Your disk space is low ", "Bitcoin", wxICON_EXCLAMATION); @@ -1546,7 +1555,7 @@ void PrintBlockTree() pindex->nFile, pindex->nBlockPos, block.GetHash().ToString().substr(0,14).c_str(), - DateTimeStr(block.nTime).c_str(), + DateTimeStrFormat("%x %H:%M:%S", block.nTime).c_str(), block.vtx.size()); CRITICAL_BLOCK(cs_mapWallet) @@ -1673,20 +1682,24 @@ bool ProcessMessages(CNode* pfrom) bool fRet = false; try { - CheckForShutdown(2); CRITICAL_BLOCK(cs_main) fRet = ProcessMessage(pfrom, strCommand, vMsg); - CheckForShutdown(2); + if (fShutdown) + return true; } - catch (std::ios_base::failure& e) { + catch (std::ios_base::failure& e) + { if (strstr(e.what(), "CDataStream::read() : end of data")) { // Allow exceptions from underlength message on vRecv printf("ProcessMessage(%s, %d bytes) : Exception '%s' caught, normally caused by a message being shorter than its stated length\n", strCommand.c_str(), nMessageSize, e.what()); } else + { PrintException(&e, "ProcessMessage()"); - } catch (std::exception& e) { + } + } + catch (std::exception& e) { PrintException(&e, "ProcessMessage()"); } catch (...) { PrintException(NULL, "ProcessMessage()"); @@ -2093,7 +2106,6 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) bool SendMessages(CNode* pto) { - CheckForShutdown(2); CRITICAL_BLOCK(cs_main) { // Don't send anything until we get their version message @@ -2223,12 +2235,10 @@ void GenerateBitcoins(bool fGenerate) void ThreadBitcoinMiner(void* parg) { - vnThreadsRunning[3]++; - CheckForShutdown(3); try { - bool fRet = BitcoinMiner(); - printf("BitcoinMiner returned %s\n", fRet ? "true" : "false"); + vnThreadsRunning[3]++; + BitcoinMiner(); vnThreadsRunning[3]--; } catch (std::exception& e) { @@ -2238,6 +2248,8 @@ void ThreadBitcoinMiner(void* parg) vnThreadsRunning[3]--; PrintException(NULL, "ThreadBitcoinMiner()"); } + + printf("ThreadBitcoinMiner exiting, %d threads remaining\n", vnThreadsRunning[3]); } int FormatHashBlocks(void* pbuffer, unsigned int len) @@ -2285,7 +2297,7 @@ void BlockSHA256(const void* pin, unsigned int nBlocks, void* pout) } -bool BitcoinMiner() +void BitcoinMiner() { printf("BitcoinMiner started\n"); @@ -2296,11 +2308,13 @@ bool BitcoinMiner() { SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); Sleep(50); - CheckForShutdown(3); + if (fShutdown) + return; while (vNodes.empty()) { Sleep(1000); - CheckForShutdown(3); + if (fShutdown) + return; } unsigned int nTransactionsUpdatedLast = nTransactionsUpdated; @@ -2324,7 +2338,7 @@ bool BitcoinMiner() // auto_ptr pblock(new CBlock()); if (!pblock.get()) - return false; + return; // Add our coinbase tx as first transaction pblock->vtx.push_back(txNew); @@ -2433,7 +2447,7 @@ bool BitcoinMiner() { // Save key if (!AddKey(key)) - return false; + return; key.MakeNewKey(); // Process this block the same as if we had received it from another node @@ -2450,7 +2464,12 @@ bool BitcoinMiner() // Update nTime every few seconds if ((++tmp.block.nNonce & 0xffff) == 0) { - CheckForShutdown(3); + if (fShutdown) + return; + if (!fGenerateBitcoins) + return; + if (fLimitProcessors && vnThreadsRunning[3] > nLimitProcessors) + return; if (tmp.block.nNonce == 0) break; if (pindexPrev != pindexBest) @@ -2459,16 +2478,10 @@ bool BitcoinMiner() break; if (vNodes.empty()) break; - if (!fGenerateBitcoins) - return true; - if (fLimitProcessors && vnThreadsRunning[3] > nLimitProcessors) - return true; tmp.block.nTime = pblock->nTime = max(pindexPrev->GetMedianTimePast()+1, GetAdjustedTime()); } } } - - return true; } diff --git a/main.h b/main.h index 6d8f0ed8..16b8c6a3 100644 --- a/main.h +++ b/main.h @@ -68,7 +68,7 @@ bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key); bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew); void GenerateBitcoins(bool fGenerate); void ThreadBitcoinMiner(void* parg); -bool BitcoinMiner(); +void BitcoinMiner(); diff --git a/net.cpp b/net.cpp index c14061e7..44a75a17 100644 --- a/net.cpp +++ b/net.cpp @@ -48,6 +48,10 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) SOCKET hSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (hSocket == INVALID_SOCKET) return false; +#if defined(__BSD__) || defined(__WXOSX__) + int set = 1; + setsockopt(hSocket, SOL_SOCKET, SO_NOSIGPIPE, (void*)&set, sizeof(int)); +#endif bool fRoutable = !(addrConnect.GetByte(3) == 10 || (addrConnect.GetByte(3) == 192 && addrConnect.GetByte(2) == 168)); bool fProxy = (fUseProxy && fRoutable); @@ -68,7 +72,7 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) char* pszSocks4 = pszSocks4IP; int nSize = sizeof(pszSocks4IP); - int ret = send(hSocket, pszSocks4, nSize, 0); + int ret = send(hSocket, pszSocks4, nSize, MSG_NOSIGNAL); if (ret != nSize) { closesocket(hSocket); @@ -100,7 +104,7 @@ bool GetMyExternalIP2(const CAddress& addrConnect, const char* pszGet, const cha if (!ConnectSocket(addrConnect, hSocket)) return error("GetMyExternalIP() : connection to %s failed", addrConnect.ToString().c_str()); - send(hSocket, pszGet, strlen(pszGet), 0); + send(hSocket, pszGet, strlen(pszGet), MSG_NOSIGNAL); string strLine; while (RecvLine(hSocket, strLine)) @@ -124,7 +128,8 @@ bool GetMyExternalIP2(const CAddress& addrConnect, const char* pszGet, const cha if (strLine.find("<")) strLine = strLine.substr(0, strLine.find("<")); strLine = strLine.substr(strspn(strLine.c_str(), " \t\n\r")); - strLine = wxString(strLine).Trim(); + while (strLine.size() > 0 && isspace(strLine[strLine.size()-1])) + strLine.resize(strLine.size()-1); CAddress addr(strLine.c_str()); printf("GetMyExternalIP() received [%s] %s\n", strLine.c_str(), addr.ToString().c_str()); if (addr.ip == 0 || addr.ip == INADDR_NONE || !addr.IsRoutable()) @@ -492,24 +497,26 @@ void ThreadSocketHandler(void* parg) { IMPLEMENT_RANDOMIZE_STACK(ThreadSocketHandler(parg)); - loop + try { vnThreadsRunning[0]++; - CheckForShutdown(0); - try - { - ThreadSocketHandler2(parg); - vnThreadsRunning[0]--; - } - catch (std::exception& e) { - vnThreadsRunning[0]--; - PrintException(&e, "ThreadSocketHandler()"); - } catch (...) { - vnThreadsRunning[0]--; - PrintException(NULL, "ThreadSocketHandler()"); - } - Sleep(5000); + ThreadSocketHandler2(parg); + vnThreadsRunning[0]--; } + catch (std::exception& e) { + vnThreadsRunning[0]--; + PrintException(&e, "ThreadSocketHandler()"); + } catch (...) { + vnThreadsRunning[0]--; + PrintException(NULL, "ThreadSocketHandler()"); + } + + foreach(CNode* pnode, vNodes) + closesocket(pnode->hSocket); + if (closesocket(hListenSocket) == SOCKET_ERROR) + printf("closesocket(hListenSocket) failed with error %d\n", WSAGetLastError()); + + printf("ThreadSocketHandler exiting\n"); } void ThreadSocketHandler2(void* parg) @@ -600,7 +607,8 @@ void ThreadSocketHandler2(void* parg) vnThreadsRunning[0]--; int nSelect = select(hSocketMax + 1, &fdsetRecv, &fdsetSend, NULL, &timeout); vnThreadsRunning[0]++; - CheckForShutdown(0); + if (fShutdown) + return; if (nSelect == SOCKET_ERROR) { int nErr = WSAGetLastError(); @@ -659,7 +667,8 @@ void ThreadSocketHandler2(void* parg) vNodesCopy = vNodes; foreach(CNode* pnode, vNodesCopy) { - CheckForShutdown(0); + if (fShutdown) + return; SOCKET hSocket = pnode->hSocket; // @@ -708,7 +717,7 @@ void ThreadSocketHandler2(void* parg) CDataStream& vSend = pnode->vSend; if (!vSend.empty()) { - int nBytes = send(hSocket, &vSend[0], vSend.size(), 0); + int nBytes = send(hSocket, &vSend[0], vSend.size(), MSG_NOSIGNAL); if (nBytes > 0) { vSend.erase(vSend.begin(), vSend.begin() + nBytes); @@ -747,24 +756,21 @@ void ThreadOpenConnections(void* parg) { IMPLEMENT_RANDOMIZE_STACK(ThreadOpenConnections(parg)); - loop + try { vnThreadsRunning[1]++; - CheckForShutdown(1); - try - { - ThreadOpenConnections2(parg); - vnThreadsRunning[1]--; - } - catch (std::exception& e) { - vnThreadsRunning[1]--; - PrintException(&e, "ThreadOpenConnections()"); - } catch (...) { - vnThreadsRunning[1]--; - PrintException(NULL, "ThreadOpenConnections()"); - } - Sleep(5000); + ThreadOpenConnections2(parg); + vnThreadsRunning[1]--; + } + catch (std::exception& e) { + vnThreadsRunning[1]--; + PrintException(&e, "ThreadOpenConnections()"); + } catch (...) { + vnThreadsRunning[1]--; + PrintException(NULL, "ThreadOpenConnections()"); } + + printf("ThreadOpenConnections exiting\n"); } void ThreadOpenConnections2(void* parg) @@ -778,7 +784,8 @@ void ThreadOpenConnections2(void* parg) for (int i = 0; i < 10; i++) { Sleep(1000); - CheckForShutdown(1); + if (fShutdown) + return; } } @@ -792,7 +799,8 @@ void ThreadOpenConnections2(void* parg) { OpenNetworkConnection(addr); Sleep(1000); - CheckForShutdown(1); + if (fShutdown) + return; } } } @@ -806,11 +814,13 @@ void ThreadOpenConnections2(void* parg) const int nMaxConnections = 15; while (vNodes.size() >= nMaxConnections || vNodes.size() >= mapAddresses.size()) { - CheckForShutdown(1); + if (fShutdown) + return; Sleep(2000); } vnThreadsRunning[1]++; - CheckForShutdown(1); + if (fShutdown) + return; // // Choose an address to connect to based on most recently seen @@ -869,14 +879,16 @@ bool OpenNetworkConnection(const CAddress& addrConnect) // // Initiate outbound network connection // - CheckForShutdown(1); + if (fShutdown) + return false; if (addrConnect.ip == addrLocalHost.ip || !addrConnect.IsIPv4() || FindNode(addrConnect.ip)) return false; vnThreadsRunning[1]--; CNode* pnode = ConnectNode(addrConnect); vnThreadsRunning[1]++; - CheckForShutdown(1); + if (fShutdown) + return false; if (!pnode) return false; pnode->fNetworkNode = true; @@ -914,24 +926,21 @@ void ThreadMessageHandler(void* parg) { IMPLEMENT_RANDOMIZE_STACK(ThreadMessageHandler(parg)); - loop + try { vnThreadsRunning[2]++; - CheckForShutdown(2); - try - { - ThreadMessageHandler2(parg); - vnThreadsRunning[2]--; - } - catch (std::exception& e) { - vnThreadsRunning[2]--; - PrintException(&e, "ThreadMessageHandler()"); - } catch (...) { - vnThreadsRunning[2]--; - PrintException(NULL, "ThreadMessageHandler()"); - } - Sleep(5000); + ThreadMessageHandler2(parg); + vnThreadsRunning[2]--; + } + catch (std::exception& e) { + vnThreadsRunning[2]--; + PrintException(&e, "ThreadMessageHandler()"); + } catch (...) { + vnThreadsRunning[2]--; + PrintException(NULL, "ThreadMessageHandler()"); } + + printf("ThreadMessageHandler exiting\n"); } void ThreadMessageHandler2(void* parg) @@ -951,10 +960,14 @@ void ThreadMessageHandler2(void* parg) // Receive messages TRY_CRITICAL_BLOCK(pnode->cs_vRecv) ProcessMessages(pnode); + if (fShutdown) + return; // Send messages TRY_CRITICAL_BLOCK(pnode->cs_vSend) SendMessages(pnode); + if (fShutdown) + return; pnode->Release(); } @@ -963,7 +976,8 @@ void ThreadMessageHandler2(void* parg) vnThreadsRunning[2]--; Sleep(100); vnThreadsRunning[2]++; - CheckForShutdown(2); + if (fShutdown) + return; } } @@ -996,7 +1010,7 @@ bool StartNode(string& strError) // Get local host ip char pszHostName[255]; - if (gethostname(pszHostName, 255) == SOCKET_ERROR) + if (gethostname(pszHostName, sizeof(pszHostName)) == SOCKET_ERROR) { strError = strprintf("Error: Unable to get IP address of this computer (gethostname returned error %d)", WSAGetLastError()); printf("%s\n", strError.c_str()); @@ -1009,9 +1023,16 @@ bool StartNode(string& strError) printf("%s\n", strError.c_str()); return false; } - addrLocalHost = CAddress(*(long*)(phostent->h_addr_list[0]), - DEFAULT_PORT, - nLocalServices); + + // Take the first IP that isn't loopback 127.x.x.x + for (int i = 0; phostent->h_addr_list[i] != NULL; i++) + printf("host ip %d: %s\n", i, CAddress(*(unsigned int*)phostent->h_addr_list[i]).ToStringIP().c_str()); + for (int i = 0; phostent->h_addr_list[i] != NULL; i++) + { + addrLocalHost = CAddress(*(unsigned int*)phostent->h_addr_list[i], DEFAULT_PORT, nLocalServices); + if (addrLocalHost.IsValid() && addrLocalHost.GetByte(3) != 127) + break; + } printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); // Create socket for listening for incoming connections @@ -1022,6 +1043,10 @@ bool StartNode(string& strError) printf("%s\n", strError.c_str()); return false; } +#if defined(__BSD__) || defined(__WXOSX__) + int set = 1; + setsockopt(hSocket, SOL_SOCKET, SO_NOSIGPIPE, (void*)&set, sizeof(int)); +#endif // Set to nonblocking, incoming connections will also inherit this #ifdef __WXMSW__ @@ -1038,19 +1063,22 @@ bool StartNode(string& strError) // The sockaddr_in structure specifies the address family, // IP address, and port for the socket that is being bound - int nRetryLimit = 15; - struct sockaddr_in sockaddr = addrLocalHost.GetSockAddr(); + struct sockaddr_in sockaddr; + memset(&sockaddr, 0, sizeof(sockaddr)); + sockaddr.sin_family = AF_INET; + sockaddr.sin_addr.s_addr = INADDR_ANY; // bind to all IPs on this computer + sockaddr.sin_port = DEFAULT_PORT; if (::bind(hListenSocket, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) == SOCKET_ERROR) { int nErr = WSAGetLastError(); if (nErr == WSAEADDRINUSE) - strError = strprintf("Error: Unable to bind to port %s on this computer. The program is probably already running.", addrLocalHost.ToString().c_str()); + strError = strprintf("Error: Unable to bind to port %d on this computer. The program is probably already running.", ntohs(sockaddr.sin_port)); else - strError = strprintf("Error: Unable to bind to port %s on this computer (bind returned error %d)", addrLocalHost.ToString().c_str(), nErr); + strError = strprintf("Error: Unable to bind to port %d on this computer (bind returned error %d)", ntohs(sockaddr.sin_port), nErr); printf("%s\n", strError.c_str()); return false; } - printf("bound to addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); + printf("bound to port %d\n", ntohs(sockaddr.sin_port)); // Listen for incoming connections if (listen(hListenSocket, SOMAXCONN) == SOCKET_ERROR) @@ -1065,6 +1093,7 @@ bool StartNode(string& strError) { // Proxies can't take incoming connections addrLocalHost.ip = CAddress("0.0.0.0").ip; + printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); } else { @@ -1115,17 +1144,17 @@ bool StopNode() fShutdown = true; nTransactionsUpdated++; int64 nStart = GetTime(); - while (vnThreadsRunning[0] || vnThreadsRunning[2] || vnThreadsRunning[3]) + while (vnThreadsRunning[0] > 0 || vnThreadsRunning[2] > 0 || vnThreadsRunning[3] > 0) { if (GetTime() - nStart > 15) break; Sleep(20); } - if (vnThreadsRunning[0]) printf("ThreadSocketHandler still running\n"); - if (vnThreadsRunning[1]) printf("ThreadOpenConnections still running\n"); - if (vnThreadsRunning[2]) printf("ThreadMessageHandler still running\n"); - if (vnThreadsRunning[3]) printf("ThreadBitcoinMiner still running\n"); - while (vnThreadsRunning[2]) + if (vnThreadsRunning[0] > 0) printf("ThreadSocketHandler still running\n"); + if (vnThreadsRunning[1] > 0) printf("ThreadOpenConnections still running\n"); + if (vnThreadsRunning[2] > 0) printf("ThreadMessageHandler still running\n"); + if (vnThreadsRunning[3] > 0) printf("ThreadBitcoinMiner still running\n"); + while (vnThreadsRunning[2] > 0) Sleep(20); Sleep(50); @@ -1135,21 +1164,3 @@ bool StopNode() #endif return true; } - -void CheckForShutdown(int n) -{ - if (fShutdown) - { - if (n != -1) - if (--vnThreadsRunning[n] < 0) - vnThreadsRunning[n] = 0; - if (n == 0) - { - foreach(CNode* pnode, vNodes) - closesocket(pnode->hSocket); - closesocket(hListenSocket); - } - printf("Thread %d exiting\n", n); - _endthread(); - } -} diff --git a/net.h b/net.h index 275a4cb8..024e7336 100644 --- a/net.h +++ b/net.h @@ -30,7 +30,6 @@ void AbandonRequests(void (*fn)(void*, CDataStream&), void* param1); bool AnySubscribed(unsigned int nChannel); bool StartNode(string& strError=REF(string())); bool StopNode(); -void CheckForShutdown(int n); @@ -268,6 +267,7 @@ public: struct sockaddr_in GetSockAddr() const { struct sockaddr_in sockaddr; + memset(&sockaddr, 0, sizeof(sockaddr)); sockaddr.sin_family = AF_INET; sockaddr.sin_addr.s_addr = ip; sockaddr.sin_port = port; diff --git a/ui.cpp b/ui.cpp index 21254819..92e1d59c 100644 --- a/ui.cpp +++ b/ui.cpp @@ -65,11 +65,13 @@ bool Is24HourTime() string DateStr(int64 nTime) { + // Can only be used safely here in the UI return (string)wxDateTime((time_t)nTime).FormatDate(); } string DateTimeStr(int64 nTime) { + // Can only be used safely here in the UI wxDateTime datetime((time_t)nTime); if (Is24HourTime()) return (string)datetime.Format("%x %H:%M"); @@ -283,6 +285,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) fRefreshListCtrl = false; fRefreshListCtrlRunning = false; fOnSetFocusAddress = false; + fRefresh = false; m_choiceFilter->SetSelection(0); m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); @@ -350,7 +353,7 @@ void Shutdown(void* parg) StopNode(); DBFlush(true); - printf("Bitcoin exiting\n"); + printf("Bitcoin exiting\n\n"); exit(0); } } @@ -391,6 +394,30 @@ void CMainFrame::OnListColBeginDrag(wxListEvent& event) event.Veto(); } +int CMainFrame::GetSortIndex(const string& strSort) +{ +#ifdef __WXMSW__ + return 0; +#else + // The wx generic listctrl implementation used on GTK doesn't sort, + // so we have to do it ourselves. Remember, we sort in reverse order. + // In the wx generic implementation, they store the list of items + // in a vector, so indexed lookups are fast, but inserts are slower + // the closer they are to the top. + int low = 0; + int high = m_listCtrl->GetItemCount(); + while (low < high) + { + int mid = low + ((high - low) / 2); + if (strSort.compare(m_listCtrl->GetItemText(mid).c_str()) >= 0) + high = mid; + else + low = mid + 1; + } + return low; +#endif +} + void CMainFrame::InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5, const wxString& str6) { string str0 = strSort; @@ -407,7 +434,7 @@ void CMainFrame::InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSo // fNew is for blind insert, only use if you're sure it's new if (fNew || nIndex == -1) { - nIndex = m_listCtrl->InsertItem(0, str0); + nIndex = m_listCtrl->InsertItem(GetSortIndex(strSort), str0); } else { @@ -415,7 +442,7 @@ void CMainFrame::InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSo if (GetItemText(m_listCtrl, nIndex, 0) != str0) { m_listCtrl->DeleteItem(nIndex); - nIndex = m_listCtrl->InsertItem(0, str0); + nIndex = m_listCtrl->InsertItem(GetSortIndex(strSort), str0); } } @@ -826,6 +853,11 @@ void CMainFrame::RefreshStatusColumn() void CMainFrame::OnPaint(wxPaintEvent& event) { + if (fRefresh) + { + fRefresh = false; + Refresh(); + } event.Skip(); } @@ -846,7 +878,7 @@ void ThreadDelayedRepaint(void* parg) { printf("DelayedRepaint\n"); wxPaintEvent event; - pframeMain->Refresh(); + pframeMain->fRefresh = true; pframeMain->AddPendingEvent(event); } } @@ -871,7 +903,7 @@ void MainFrameRepaint() printf("MainFrameRepaint\n"); wxPaintEvent event; - pframeMain->Refresh(); + pframeMain->fRefresh = true; pframeMain->AddPendingEvent(event); } } @@ -907,7 +939,7 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) } vWalletUpdated.clear(); if (m_listCtrl->GetItemCount() && strTop != (string)m_listCtrl->GetItemText(0)) - m_listCtrl->ScrollList(0, INT_MAX); + m_listCtrl->ScrollList(0, INT_MIN/2); } } @@ -943,9 +975,10 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, nTransactionCount); m_statusBar->SetStatusText(strStatus, 2); -#ifdef __WXMSW__ - m_listCtrl->OnPaint(event); -#endif + // Pass through to listctrl to actually do the paint, we're just hooking the message + m_listCtrl->Disconnect(wxEVT_PAINT, (wxObjectEventFunction)NULL, NULL, this); + m_listCtrl->GetEventHandler()->ProcessEvent(event); + m_listCtrl->Connect(wxEVT_PAINT, wxPaintEventHandler(CMainFrame::OnPaintListCtrl), NULL, this); } @@ -3331,7 +3364,11 @@ bool CMyApp::OnInit2() g_isPainting = 10000; #endif wxImage::AddHandler(new wxPNGHandler); +#ifdef __WXMSW__ SetAppName("Bitcoin"); +#else + SetAppName("bitcoin"); +#endif ParseParameters(argc, argv); if (mapArgs.count("-?") || mapArgs.count("--help")) @@ -3355,7 +3392,10 @@ bool CMyApp::OnInit2() // Limit to single instance per user // Required to protect the database files if we're going to keep deleting log.* // - wxString strMutexName = wxString("Bitcoin.") + getenv("HOMEPATH"); +#ifdef __WXMSW__ + // todo: wxSingleInstanceChecker wasn't working on Linux, never deleted its lock file + // maybe should go by whether successfully bind port 8333 instead + wxString strMutexName = wxString("bitcoin_running.") + getenv("HOMEPATH"); for (int i = 0; i < strMutexName.size(); i++) if (!isalnum(strMutexName[i])) strMutexName[i] = '.'; @@ -3367,7 +3407,6 @@ bool CMyApp::OnInit2() loop { // TODO: find out how to do this in Linux, or replace with wxWidgets commands -#ifdef __WXMSW__ // Show the previous instance and exit HWND hwndPrev = FindWindow("wxWindowClassNR", "Bitcoin"); if (hwndPrev) @@ -3377,7 +3416,6 @@ bool CMyApp::OnInit2() SetForegroundWindow(hwndPrev); return false; } -#endif if (GetTime() > nStart + 60) return false; @@ -3390,6 +3428,7 @@ bool CMyApp::OnInit2() break; } } +#endif // // Parameters diff --git a/ui.h b/ui.h index 47839e81..1d0491f1 100644 --- a/ui.h +++ b/ui.h @@ -32,7 +32,6 @@ extern int fMinimizeOnClose; extern void HandleCtrlA(wxKeyEvent& event); -extern string DateTimeStr(int64 nTime); extern string FormatTxStatus(const CWalletTx& wtx); extern void CrossThreadCall(int nID, void* pdata); extern void MainFrameRepaint(); @@ -84,8 +83,10 @@ public: bool fRefreshListCtrlRunning; bool fOnSetFocusAddress; unsigned int nListViewUpdated; + bool fRefresh; void OnCrossThreadCall(wxCommandEvent& event); + int GetSortIndex(const string& strSort); void InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5); bool DeleteLine(uint256 hashKey); bool InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex=-1); diff --git a/uibase.cpp b/uibase.cpp index bb564e99..9a619995 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -84,7 +84,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_staticText32->Wrap( -1 ); bSizer85->Add( m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxLEFT, 5 ); - m_textCtrlAddress = new wxTextCtrl( this, wxID_TEXTCTRLADDRESS, wxEmptyString, wxDefaultPosition, wxSize( 250,-1 ), wxTE_READONLY ); + m_textCtrlAddress = new wxTextCtrl( this, wxID_TEXTCTRLADDRESS, wxEmptyString, wxDefaultPosition, wxSize( 340,-1 ), wxTE_READONLY ); m_textCtrlAddress->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); bSizer85->Add( m_textCtrlAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); @@ -143,7 +143,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& wxBoxSizer* bSizer157; bSizer157 = new wxBoxSizer( wxVERTICAL ); - m_listCtrl = new wxListCtrl( m_panel7, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxALWAYS_SHOW_SB ); + m_listCtrl = new wxListCtrl( m_panel7, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); bSizer157->Add( m_listCtrl, 1, wxEXPAND|wxALL, 5 ); m_panel7->SetSizer( bSizer157 ); @@ -343,7 +343,7 @@ CTxDetailsDialogBase::CTxDetailsDialogBase( wxWindow* parent, wxWindowID id, con wxBoxSizer* bSizer65; bSizer65 = new wxBoxSizer( wxVERTICAL ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( 85,25 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer65->Add( m_buttonOK, 0, wxALL, 5 ); bSizer64->Add( bSizer65, 0, wxALIGN_RIGHT, 5 ); @@ -520,17 +520,13 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w wxBoxSizer* bSizer58; bSizer58 = new wxBoxSizer( wxHORIZONTAL ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( 85,25 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer58->Add( m_buttonOK, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer58->Add( m_buttonCancel, 0, wxALL, 5 ); m_buttonApply = new wxButton( this, wxID_APPLY, wxT("&Apply"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonApply->SetMinSize( wxSize( 85,25 ) ); - bSizer58->Add( m_buttonApply, 0, wxALL, 5 ); bSizer55->Add( bSizer58, 0, wxALIGN_RIGHT, 5 ); @@ -622,7 +618,7 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr bSizer61->Add( 0, 0, 1, wxEXPAND, 5 ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( 85,25 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer61->Add( m_buttonOK, 0, wxALL, 5 ); bSizer60->Add( bSizer61, 0, wxALIGN_RIGHT|wxEXPAND, 5 ); @@ -767,13 +763,10 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin m_buttonSend = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); m_buttonSend->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); - m_buttonSend->SetMinSize( wxSize( 85,25 ) ); bSizer23->Add( m_buttonSend, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer23->Add( m_buttonCancel, 0, wxALL, 5 ); bSizer21->Add( bSizer23, 0, wxEXPAND, 5 ); @@ -833,13 +826,10 @@ CSendingDialogBase::CSendingDialogBase( wxWindow* parent, wxWindowID id, const w m_buttonOK = new wxButton( this, wxID_ANY, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); m_buttonOK->Enable( false ); - m_buttonOK->SetMinSize( wxSize( 85,25 ) ); bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); @@ -887,28 +877,19 @@ CYourAddressDialogBase::CYourAddressDialogBase( wxWindow* parent, wxWindowID id, bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); m_buttonRename = new wxButton( this, wxID_BUTTONRENAME, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonRename->SetMinSize( wxSize( 85,25 ) ); - bSizer69->Add( m_buttonRename, 0, wxALL, 5 ); m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonNew->SetMinSize( wxSize( 110,25 ) ); - bSizer69->Add( m_buttonNew, 0, wxALL, 5 ); m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonCopy->SetMinSize( wxSize( 120,25 ) ); - bSizer69->Add( m_buttonCopy, 0, wxALL, 5 ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonOK->SetMinSize( wxSize( 85,25 ) ); - bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); m_buttonCancel->Hide(); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); @@ -969,28 +950,18 @@ CAddressBookDialogBase::CAddressBookDialogBase( wxWindow* parent, wxWindowID id, bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); m_buttonEdit = new wxButton( this, wxID_BUTTONEDIT, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonEdit->SetMinSize( wxSize( 85,25 ) ); - bSizer69->Add( m_buttonEdit, 0, wxALL, 5 ); m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonNew->SetMinSize( wxSize( 110,25 ) ); - bSizer69->Add( m_buttonNew, 0, wxALL, 5 ); m_buttonDelete = new wxButton( this, wxID_BUTTONDELETE, wxT("&Delete"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonDelete->SetMinSize( wxSize( 85,25 ) ); - bSizer69->Add( m_buttonDelete, 0, wxALL, 5 ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonOK->SetMinSize( wxSize( 85,25 ) ); - bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); @@ -1389,18 +1360,12 @@ CEditProductDialogBase::CEditProductDialogBase( wxWindow* parent, wxWindowID id, bSizer26 = new wxBoxSizer( wxHORIZONTAL ); m_buttonOK = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonOK->SetMinSize( wxSize( 85,25 ) ); - bSizer26->Add( m_buttonOK, 0, wxALL, 5 ); m_buttonPreview = new wxButton( this, wxID_BUTTONPREVIEW, wxT("&Preview"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonPreview->SetMinSize( wxSize( 85,25 ) ); - bSizer26->Add( m_buttonPreview, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer26->Add( m_buttonCancel, 0, wxALL, 5 ); bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); @@ -1605,18 +1570,13 @@ CViewProductDialogBase::CViewProductDialogBase( wxWindow* parent, wxWindowID id, m_buttonBack = new wxButton( this, wxID_BUTTONBACK, wxT("< &Back "), wxDefaultPosition, wxDefaultSize, 0 ); m_buttonBack->Enable( false ); - m_buttonBack->SetMinSize( wxSize( 85,25 ) ); bSizer26->Add( m_buttonBack, 0, wxALL, 5 ); m_buttonNext = new wxButton( this, wxID_BUTTONNEXT, wxT(" &Next >"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonNext->SetMinSize( wxSize( 85,25 ) ); - bSizer26->Add( m_buttonNext, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer26->Add( m_buttonCancel, 0, wxALL, 5 ); bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); @@ -1662,8 +1622,6 @@ CViewOrderDialogBase::CViewOrderDialogBase( wxWindow* parent, wxWindowID id, con bSizer26 = new wxBoxSizer( wxHORIZONTAL ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonOK->SetMinSize( wxSize( 85,25 ) ); - bSizer26->Add( m_buttonOK, 0, wxALL, 5 ); bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); @@ -1720,13 +1678,9 @@ CEditReviewDialogBase::CEditReviewDialogBase( wxWindow* parent, wxWindowID id, c bSizer113 = new wxBoxSizer( wxHORIZONTAL ); m_buttonSubmit = new wxButton( this, wxID_SUBMIT, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonSubmit->SetMinSize( wxSize( 85,25 ) ); - bSizer113->Add( m_buttonSubmit, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer113->Add( m_buttonCancel, 0, wxALL, 5 ); bSizer112->Add( bSizer113, 0, wxALIGN_RIGHT, 5 ); @@ -1951,13 +1905,9 @@ CGetTextFromUserDialogBase::CGetTextFromUserDialogBase( wxWindow* parent, wxWind bSizer80->Add( 0, 0, 1, wxEXPAND, 5 ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - m_buttonOK->SetMinSize( wxSize( 85,25 ) ); - bSizer80->Add( m_buttonOK, 0, wxALL, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonCancel->SetMinSize( wxSize( 85,25 ) ); - bSizer80->Add( m_buttonCancel, 0, wxALL, 5 ); bSizer79->Add( bSizer80, 0, wxEXPAND, 5 ); diff --git a/uiproject.fbp b/uiproject.fbp index 7bce7349..17656b56 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -70,7 +70,7 @@ - + 240,240,240 1 @@ -225,7 +225,7 @@ - + 20,20 @@ -439,7 +439,7 @@ m_textCtrlAddress protected - 250,-1 + 340,-1 wxTE_READONLY @@ -944,7 +944,7 @@ - wxALWAYS_SHOW_SB + wxVSCROLL @@ -1648,7 +1648,7 @@ m_buttonOK protected - 85,25 + -1,-1 @@ -3031,7 +3031,7 @@ m_buttonOK protected - 85,25 + -1,-1 @@ -3079,7 +3079,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected @@ -3131,7 +3131,7 @@ wxID_APPLY &Apply - 85,25 + -1,-1 m_buttonApply protected @@ -3493,7 +3493,7 @@ m_buttonOK protected - 85,25 + -1,-1 @@ -4485,7 +4485,7 @@ wxID_BUTTONSEND &Send - 85,25 + -1,-1 m_buttonSend protected @@ -4537,7 +4537,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected @@ -4775,7 +4775,7 @@ wxID_ANY OK - 85,25 + -1,-1 m_buttonOK protected @@ -4827,7 +4827,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected @@ -5089,7 +5089,7 @@ wxID_BUTTONRENAME &Edit... - 85,25 + -1,-1 m_buttonRename protected @@ -5141,7 +5141,7 @@ wxID_BUTTONNEW &New Address... - 110,25 + -1,-1 m_buttonNew protected @@ -5193,7 +5193,7 @@ wxID_BUTTONCOPY &Copy to Clipboard - 120,25 + -1,-1 m_buttonCopy protected @@ -5245,7 +5245,7 @@ wxID_OK OK - 85,25 + -1,-1 m_buttonOK protected @@ -5297,7 +5297,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected @@ -5559,7 +5559,7 @@ wxID_BUTTONEDIT &Edit... - 85,25 + -1,-1 m_buttonEdit protected @@ -5611,7 +5611,7 @@ wxID_BUTTONNEW &New Address... - 110,25 + -1,-1 m_buttonNew protected @@ -5663,7 +5663,7 @@ wxID_BUTTONDELETE &Delete - 85,25 + -1,-1 m_buttonDelete protected @@ -5715,7 +5715,7 @@ wxID_OK OK - 85,25 + -1,-1 m_buttonOK protected @@ -5767,7 +5767,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel public @@ -10225,7 +10225,7 @@ wxID_BUTTONSEND &Send - 85,25 + -1,-1 m_buttonOK protected @@ -10277,7 +10277,7 @@ wxID_BUTTONPREVIEW &Preview - 85,25 + -1,-1 m_buttonPreview protected @@ -10329,7 +10329,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected @@ -10798,7 +10798,7 @@ wxID_BUTTONBACK < &Back - 85,25 + -1,-1 m_buttonBack protected @@ -10850,7 +10850,7 @@ wxID_BUTTONNEXT &Next > - 85,25 + -1,-1 m_buttonNext protected @@ -10902,7 +10902,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected @@ -11087,7 +11087,7 @@ wxID_OK OK - 85,25 + -1,-1 m_buttonOK protected @@ -11488,7 +11488,7 @@ wxID_SUBMIT &Submit - 85,25 + -1,-1 m_buttonSubmit protected @@ -11540,7 +11540,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected @@ -12907,7 +12907,7 @@ wxID_OK OK - 85,25 + -1,-1 m_buttonOK protected @@ -12959,7 +12959,7 @@ wxID_CANCEL Cancel - 85,25 + -1,-1 m_buttonCancel protected diff --git a/util.cpp b/util.cpp index d2e624d6..5efb579b 100644 --- a/util.cpp +++ b/util.cpp @@ -96,12 +96,7 @@ void RandAddSeedPerfmon() hash = 0; memset(pdata, 0, nSize); - time_t nTime; - time(&nTime); - struct tm* ptmTime = gmtime(&nTime); - char pszTime[200]; - strftime(pszTime, sizeof(pszTime), "%x %H:%M:%S", ptmTime); - printf("%s RandAddSeed() %d bytes\n", pszTime, nSize); + printf("%s RandAddSeed() %d bytes\n", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str(), nSize); } #endif } @@ -350,7 +345,9 @@ void FormatException(char* pszMessage, std::exception* pex, const char* pszThrea pszModule[0] = '\0'; GetModuleFileName(NULL, pszModule, sizeof(pszModule)); #else - const char* pszModule = wxStandardPaths::Get().GetExecutablePath().mb_str(); + // might not be thread safe, uses wxString + //const char* pszModule = wxStandardPaths::Get().GetExecutablePath().mb_str(); + const char* pszModule = "bitcoin"; #endif if (pex) snprintf(pszMessage, 1000, @@ -425,7 +422,6 @@ void GetDataDir(char* pszDir) } strlcpy(pszDir, pszCachedDir, MAX_PATH); } - } string GetDataDir() diff --git a/util.h b/util.h index 3bc7c798..8fcfcd0d 100644 --- a/util.h +++ b/util.h @@ -84,6 +84,10 @@ inline void SetThreadPriority(int nThread, int nPriority) { setpriority(PRIO_PRO #define THREAD_PRIORITY_NORMAL 0 #define THREAD_PRIORITY_ABOVE_NORMAL 0 #endif +#ifndef MSG_NOSIGNAL +#define MSG_NOSIGNAL 0 +#endif + @@ -379,6 +383,14 @@ inline int64 GetTimeMillis() return wxGetLocalTimeMillis().GetValue(); } +inline string DateTimeStrFormat(const char* pszFormat, int64 nTime) +{ + time_t n = nTime; + struct tm* ptmTime = gmtime(&n); + char pszTime[200]; + strftime(pszTime, sizeof(pszTime), pszFormat, ptmTime); + return pszTime; +} @@ -400,7 +412,7 @@ inline void heapchk() { \ static char nLoops; \ if (nLoops <= 0) \ - nLoops = GetRand(50) + 1; \ + nLoops = GetRand(20) + 1; \ if (nLoops-- > 1) \ { \ ThreadFn; \ From 8acda009d921a65dab87bd0a40f63981a1586561 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 6 Nov 2009 18:53:26 +0000 Subject: [PATCH 023/133] UIThreadCall, ThreadSafeMessageBox git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@34 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 2 + main.cpp | 2 +- ui.cpp | 68 ++-- ui.h | 22 +- uibase.cpp | 160 -------- uibase.h | 82 +---- uiproject.fbp | 992 -------------------------------------------------- 7 files changed, 44 insertions(+), 1284 deletions(-) diff --git a/headers.h b/headers.h index c7f3cd85..6cd2da07 100644 --- a/headers.h +++ b/headers.h @@ -53,6 +53,8 @@ #include #include #include +#include +#include #ifdef __WXMSW__ #include diff --git a/main.cpp b/main.cpp index cc22bad6..e4f1deb8 100644 --- a/main.cpp +++ b/main.cpp @@ -1381,7 +1381,7 @@ bool CheckDiskSpace(int64 nAdditionalBytes) if (nFreeBytesAvailable < (int64)15000000 + nAdditionalBytes) { fShutdown = true; - wxMessageBox("Warning: Your disk space is low ", "Bitcoin", wxICON_EXCLAMATION); + ThreadSafeMessageBox("Warning: Your disk space is low ", "Bitcoin", wxOK | wxICON_EXCLAMATION); _beginthread(Shutdown, 0, NULL); return false; } diff --git a/ui.cpp b/ui.cpp index 92e1d59c..d3302313 100644 --- a/ui.cpp +++ b/ui.cpp @@ -14,13 +14,10 @@ void SetStartOnSystemStartup(bool fAutoStart); -DEFINE_EVENT_TYPE(wxEVT_CROSSTHREADCALL) +DEFINE_EVENT_TYPE(wxEVT_UITHREADCALL) DEFINE_EVENT_TYPE(wxEVT_REPLY1) DEFINE_EVENT_TYPE(wxEVT_REPLY2) DEFINE_EVENT_TYPE(wxEVT_REPLY3) -DEFINE_EVENT_TYPE(wxEVT_TABLEADDED) -DEFINE_EVENT_TYPE(wxEVT_TABLEUPDATED) -DEFINE_EVENT_TYPE(wxEVT_TABLEDELETED) CMainFrame* pframeMain = NULL; CMyTaskBarIcon* ptaskbaricon = NULL; @@ -184,6 +181,24 @@ void AddToMyProducts(CProduct product) ""); } +void StringMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) +{ + wxMessageBox(message, caption, style, parent, x, y); +} + +int ThreadSafeMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) +{ +#ifdef __WXMSW__ + wxMessageBox(message, caption, style, parent, x, y); +#else + UIThreadCall(bind(StringMessageBox, message, caption, style, parent, x, y)); +#endif +} + + + + + @@ -193,6 +208,7 @@ void AddToMyProducts(CProduct product) // // Custom events // +// If this code gets used again, it should be replaced with something like UIThreadCall set setCallbackAvailable; CCriticalSection cs_setCallbackAvailable; @@ -279,7 +295,7 @@ CDataStream GetStreamFromEvent(const wxCommandEvent& event) CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) { - Connect(wxEVT_CROSSTHREADCALL, wxCommandEventHandler(CMainFrame::OnCrossThreadCall), NULL, this); + Connect(wxEVT_UITHREADCALL, wxCommandEventHandler(CMainFrame::OnUIThreadCall), NULL, this); // Init fRefreshListCtrl = false; @@ -982,36 +998,27 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) } -void CrossThreadCall(wxCommandEvent& event) +void UIThreadCall(boost::function fn) { + // Call this with a function object created with bind. + // bind needs all parameters to match the function's expected types + // and all default parameters specified. Some examples: + // UIThreadCall(bind(wxBell)); + // UIThreadCall(bind(wxMessageBox, wxT("Message"), wxT("Title"), wxOK, (wxWindow*)NULL, -1, -1)); + // UIThreadCall(bind(&CMainFrame::OnMenuHelpAbout, pframeMain, event)); if (pframeMain) + { + wxCommandEvent event(wxEVT_UITHREADCALL); + event.SetClientData((void*)new boost::function(fn)); pframeMain->GetEventHandler()->AddPendingEvent(event); + } } -void CrossThreadCall(int nID, void* pdata) -{ - wxCommandEvent event; - event.SetInt(nID); - event.SetClientData(pdata); - if (pframeMain) - pframeMain->GetEventHandler()->AddPendingEvent(event); -} - -void CMainFrame::OnCrossThreadCall(wxCommandEvent& event) +void CMainFrame::OnUIThreadCall(wxCommandEvent& event) { - void* pdata = event.GetClientData(); - switch (event.GetInt()) - { - case UICALL_ADDORDER: - { - break; - } - - case UICALL_UPDATEORDER: - { - break; - } - } + boost::function* pfn = (boost::function*)event.GetClientData(); + (*pfn)(); + delete pfn; } void CMainFrame::OnMenuFileExit(wxCommandEvent& event) @@ -3305,9 +3312,6 @@ wxMenu* CMyTaskBarIcon::CreatePopupMenu() - - - ////////////////////////////////////////////////////////////////////////////// // // CMyApp diff --git a/ui.h b/ui.h index 1d0491f1..1db40997 100644 --- a/ui.h +++ b/ui.h @@ -5,19 +5,10 @@ -DECLARE_EVENT_TYPE(wxEVT_CROSSTHREADCALL, -1) +DECLARE_EVENT_TYPE(wxEVT_UITHREADCALL, -1) DECLARE_EVENT_TYPE(wxEVT_REPLY1, -1) DECLARE_EVENT_TYPE(wxEVT_REPLY2, -1) DECLARE_EVENT_TYPE(wxEVT_REPLY3, -1) -DECLARE_EVENT_TYPE(wxEVT_TABLEADDED, -1) -DECLARE_EVENT_TYPE(wxEVT_TABLEUPDATED, -1) -DECLARE_EVENT_TYPE(wxEVT_TABLEDELETED, -1) - -enum -{ - UICALL_ADDORDER = 1, - UICALL_UPDATEORDER, -}; @@ -33,9 +24,10 @@ extern int fMinimizeOnClose; extern void HandleCtrlA(wxKeyEvent& event); extern string FormatTxStatus(const CWalletTx& wtx); -extern void CrossThreadCall(int nID, void* pdata); +extern void UIThreadCall(boost::function); extern void MainFrameRepaint(); extern void Shutdown(void* parg); +extern int ThreadSafeMessageBox(const string& message, const string& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1); @@ -85,7 +77,7 @@ public: unsigned int nListViewUpdated; bool fRefresh; - void OnCrossThreadCall(wxCommandEvent& event); + void OnUIThreadCall(wxCommandEvent& event); int GetSortIndex(const string& strSort); void InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str1, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5); bool DeleteLine(uint256 hashKey); @@ -473,9 +465,3 @@ public: DECLARE_EVENT_TABLE() }; - - - - - - diff --git a/uibase.cpp b/uibase.cpp index 9a619995..6a280cda 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -1702,166 +1702,6 @@ CEditReviewDialogBase::~CEditReviewDialogBase() m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonCancel ), NULL, this ); } -CPokerLobbyDialogBase::CPokerLobbyDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) -{ - this->SetSizeHints( wxDefaultSize, wxDefaultSize ); - this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE ) ); - - wxBoxSizer* bSizer156; - bSizer156 = new wxBoxSizer( wxHORIZONTAL ); - - m_treeCtrl = new wxTreeCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT ); - m_treeCtrl->SetMinSize( wxSize( 130,-1 ) ); - - bSizer156->Add( m_treeCtrl, 0, wxEXPAND|wxTOP|wxBOTTOM|wxLEFT, 5 ); - - wxBoxSizer* bSizer172; - bSizer172 = new wxBoxSizer( wxVERTICAL ); - - m_listCtrl = new wxListCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); - bSizer172->Add( m_listCtrl, 1, wxEXPAND|wxALL, 5 ); - - m_buttonNewTable = new wxButton( this, wxID_OPENNEWTABLE, wxT("&Open New Table"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer172->Add( m_buttonNewTable, 0, wxALL, 5 ); - - bSizer156->Add( bSizer172, 1, wxEXPAND, 5 ); - - this->SetSizer( bSizer156 ); - this->Layout(); - - // Connect Events - m_treeCtrl->Connect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( CPokerLobbyDialogBase::OnTreeSelChanged ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemActivated ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemSelected ), NULL, this ); - m_buttonNewTable->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerLobbyDialogBase::OnButtonNewTable ), NULL, this ); -} - -CPokerLobbyDialogBase::~CPokerLobbyDialogBase() -{ - // Disconnect Events - m_treeCtrl->Disconnect( wxEVT_COMMAND_TREE_SEL_CHANGED, wxTreeEventHandler( CPokerLobbyDialogBase::OnTreeSelChanged ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemActivated ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CPokerLobbyDialogBase::OnListItemSelected ), NULL, this ); - m_buttonNewTable->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerLobbyDialogBase::OnButtonNewTable ), NULL, this ); -} - -CPokerDialogBase::CPokerDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) -{ - this->SetSizeHints( wxDefaultSize, wxDefaultSize ); - - wxBoxSizer* bSizer174; - bSizer174 = new wxBoxSizer( wxVERTICAL ); - - m_checkSitOut = new wxCheckBox( this, wxID_ANY, wxT("Deal Me Out"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer174->Add( m_checkSitOut, 0, wxALL, 5 ); - - m_buttonDealHand = new wxButton( this, wxID_DEALHAND, wxT("&Deal Hand"), wxDefaultPosition, wxSize( 150,25 ), 0 ); - bSizer174->Add( m_buttonDealHand, 0, wxALL, 5 ); - - m_buttonFold = new wxButton( this, wxID_FOLD, wxT("&Fold"), wxDefaultPosition, wxSize( 80,25 ), 0 ); - bSizer174->Add( m_buttonFold, 0, wxALL, 5 ); - - m_buttonCall = new wxButton( this, wxID_CALL, wxT("&Call"), wxDefaultPosition, wxSize( 80,25 ), 0 ); - bSizer174->Add( m_buttonCall, 0, wxALL, 5 ); - - m_buttonRaise = new wxButton( this, wxID_RAISE, wxT("&Raise"), wxDefaultPosition, wxSize( 80,25 ), 0 ); - bSizer174->Add( m_buttonRaise, 0, wxALL, 5 ); - - m_buttonLeaveTable = new wxButton( this, wxID_LEAVETABLE, wxT("&Leave Table"), wxDefaultPosition, wxSize( 90,25 ), 0 ); - bSizer174->Add( m_buttonLeaveTable, 0, wxALL, 5 ); - - m_textDitchPlayer = new wxTextCtrl( this, wxID_DITCHPLAYER, wxEmptyString, wxDefaultPosition, wxSize( 45,-1 ), wxTE_PROCESS_ENTER ); - bSizer174->Add( m_textDitchPlayer, 0, wxALL, 5 ); - - m_checkPreFold = new wxCheckBox( this, wxID_ANY, wxT("FOLD"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); - - bSizer174->Add( m_checkPreFold, 0, wxALL, 5 ); - - m_checkPreCall = new wxCheckBox( this, wxID_ANY, wxT("CALL"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); - - bSizer174->Add( m_checkPreCall, 0, wxALL, 5 ); - - m_checkPreCallAny = new wxCheckBox( this, wxID_ANY, wxT("CALL ANY"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); - - bSizer174->Add( m_checkPreCallAny, 0, wxALL, 5 ); - - m_checkPreRaise = new wxCheckBox( this, wxID_ANY, wxT("RAISE"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); - - bSizer174->Add( m_checkPreRaise, 0, wxALL, 5 ); - - m_checkPreRaiseAny = new wxCheckBox( this, wxID_ANY, wxT("RAISE ANY"), wxDefaultPosition, wxSize( 100,-1 ), 0 ); - - bSizer174->Add( m_checkPreRaiseAny, 0, wxALL, 5 ); - - this->SetSizer( bSizer174 ); - this->Layout(); - m_statusBar = this->CreateStatusBar( 1, wxST_SIZEGRIP, wxID_ANY ); - - // Connect Events - this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CPokerDialogBase::OnClose ) ); - this->Connect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_LEFT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_RIGHT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_MOTION, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Connect( wxEVT_PAINT, wxPaintEventHandler( CPokerDialogBase::OnPaint ) ); - this->Connect( wxEVT_SIZE, wxSizeEventHandler( CPokerDialogBase::OnSize ) ); - m_checkSitOut->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckSitOut ), NULL, this ); - m_buttonDealHand->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonDealHand ), NULL, this ); - m_buttonFold->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonFold ), NULL, this ); - m_buttonCall->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonCall ), NULL, this ); - m_buttonRaise->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonRaise ), NULL, this ); - m_buttonLeaveTable->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonLeaveTable ), NULL, this ); - m_textDitchPlayer->Connect( wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler( CPokerDialogBase::OnDitchPlayer ), NULL, this ); - m_checkPreFold->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreFold ), NULL, this ); - m_checkPreCall->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCall ), NULL, this ); - m_checkPreCallAny->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCallAny ), NULL, this ); - m_checkPreRaise->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaise ), NULL, this ); - m_checkPreRaiseAny->Connect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaiseAny ), NULL, this ); -} - -CPokerDialogBase::~CPokerDialogBase() -{ - // Disconnect Events - this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CPokerDialogBase::OnClose ) ); - this->Disconnect( wxEVT_LEFT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_LEFT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_MIDDLE_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_MIDDLE_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_RIGHT_DOWN, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_RIGHT_UP, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_MOTION, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_LEFT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_MIDDLE_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_RIGHT_DCLICK, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_LEAVE_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CPokerDialogBase::OnMouseEvents ) ); - this->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CPokerDialogBase::OnPaint ) ); - this->Disconnect( wxEVT_SIZE, wxSizeEventHandler( CPokerDialogBase::OnSize ) ); - m_checkSitOut->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckSitOut ), NULL, this ); - m_buttonDealHand->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonDealHand ), NULL, this ); - m_buttonFold->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonFold ), NULL, this ); - m_buttonCall->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonCall ), NULL, this ); - m_buttonRaise->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonRaise ), NULL, this ); - m_buttonLeaveTable->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnButtonLeaveTable ), NULL, this ); - m_textDitchPlayer->Disconnect( wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler( CPokerDialogBase::OnDitchPlayer ), NULL, this ); - m_checkPreFold->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreFold ), NULL, this ); - m_checkPreCall->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCall ), NULL, this ); - m_checkPreCallAny->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreCallAny ), NULL, this ); - m_checkPreRaise->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaise ), NULL, this ); - m_checkPreRaiseAny->Disconnect( wxEVT_COMMAND_CHECKBOX_CLICKED, wxCommandEventHandler( CPokerDialogBase::OnCheckPreRaiseAny ), NULL, this ); -} - CGetTextFromUserDialogBase::CGetTextFromUserDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { this->SetSizeHints( wxDefaultSize, wxDefaultSize ); diff --git a/uibase.h b/uibase.h index d52158f1..6332b931 100644 --- a/uibase.h +++ b/uibase.h @@ -37,7 +37,6 @@ #include #include #include -#include /////////////////////////////////////////////////////////////////////////// @@ -88,14 +87,7 @@ #define wxID_BUTTONBACK 1044 #define wxID_BUTTONNEXT 1045 #define wxID_SUBMIT 1046 -#define wxID_OPENNEWTABLE 1047 -#define wxID_DEALHAND 1048 -#define wxID_FOLD 1049 -#define wxID_CALL 1050 -#define wxID_RAISE 1051 -#define wxID_LEAVETABLE 1052 -#define wxID_DITCHPLAYER 1053 -#define wxID_TEXTCTRL 1054 +#define wxID_TEXTCTRL 1047 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase @@ -652,78 +644,6 @@ class CEditReviewDialogBase : public wxFrame }; -/////////////////////////////////////////////////////////////////////////////// -/// Class CPokerLobbyDialogBase -/////////////////////////////////////////////////////////////////////////////// -class CPokerLobbyDialogBase : public wxFrame -{ - private: - - protected: - wxTreeCtrl* m_treeCtrl; - wxListCtrl* m_listCtrl; - wxButton* m_buttonNewTable; - - // Virtual event handlers, overide them in your derived class - virtual void OnTreeSelChanged( wxTreeEvent& event ){ event.Skip(); } - virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } - virtual void OnListItemSelected( wxListEvent& event ){ event.Skip(); } - virtual void OnButtonNewTable( wxCommandEvent& event ){ event.Skip(); } - - - public: - CPokerLobbyDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker Lobby"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 586,457 ), long style = wxDEFAULT_FRAME_STYLE|wxTAB_TRAVERSAL ); - ~CPokerLobbyDialogBase(); - -}; - -/////////////////////////////////////////////////////////////////////////////// -/// Class CPokerDialogBase -/////////////////////////////////////////////////////////////////////////////// -class CPokerDialogBase : public wxFrame -{ - private: - - protected: - wxButton* m_buttonDealHand; - wxButton* m_buttonFold; - wxButton* m_buttonCall; - wxButton* m_buttonRaise; - wxButton* m_buttonLeaveTable; - wxTextCtrl* m_textDitchPlayer; - - // Virtual event handlers, overide them in your derived class - virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } - virtual void OnMouseEvents( wxMouseEvent& event ){ event.Skip(); } - virtual void OnPaint( wxPaintEvent& event ){ event.Skip(); } - virtual void OnSize( wxSizeEvent& event ){ event.Skip(); } - virtual void OnCheckSitOut( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDealHand( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonFold( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonCall( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonRaise( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonLeaveTable( wxCommandEvent& event ){ event.Skip(); } - virtual void OnDitchPlayer( wxCommandEvent& event ){ event.Skip(); } - virtual void OnCheckPreFold( wxCommandEvent& event ){ event.Skip(); } - virtual void OnCheckPreCall( wxCommandEvent& event ){ event.Skip(); } - virtual void OnCheckPreCallAny( wxCommandEvent& event ){ event.Skip(); } - virtual void OnCheckPreRaise( wxCommandEvent& event ){ event.Skip(); } - virtual void OnCheckPreRaiseAny( wxCommandEvent& event ){ event.Skip(); } - - - public: - wxCheckBox* m_checkSitOut; - wxCheckBox* m_checkPreFold; - wxCheckBox* m_checkPreCall; - wxCheckBox* m_checkPreCallAny; - wxCheckBox* m_checkPreRaise; - wxCheckBox* m_checkPreRaiseAny; - wxStatusBar* m_statusBar; - CPokerDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Poker"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 806,550 ), long style = wxDEFAULT_FRAME_STYLE|wxFRAME_NO_TASKBAR|wxFULL_REPAINT_ON_RESIZE|wxTAB_TRAVERSAL ); - ~CPokerDialogBase(); - -}; - /////////////////////////////////////////////////////////////////////////////// /// Class CGetTextFromUserDialogBase /////////////////////////////////////////////////////////////////////////////// diff --git a/uiproject.fbp b/uiproject.fbp index 17656b56..7c5bb24f 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -11581,998 +11581,6 @@ - - wxSYS_COLOUR_BTNFACE - - - 1 - - - - 0 - wxID_ANY - - - CPokerLobbyDialogBase - - 586,457 - wxDEFAULT_FRAME_STYLE - - Poker Lobby - - - - wxTAB_TRAVERSAL - 1 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer156 - wxHORIZONTAL - none - - 5 - wxEXPAND|wxTOP|wxBOTTOM|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - - 130,-1 - m_treeCtrl - protected - - - wxTR_HAS_BUTTONS|wxTR_HIDE_ROOT|wxTR_LINES_AT_ROOT - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - OnTreeSelChanged - - - - - - - - 5 - wxEXPAND - 1 - - - bSizer172 - wxVERTICAL - none - - 5 - wxEXPAND|wxALL - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_listCtrl - protected - - - wxLC_NO_SORT_HEADER|wxLC_REPORT - - - - - - - - - - - - - - - - - - - - - - - - - - - - - OnListItemActivated - - - - - OnListItemSelected - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 0 - 1 - - - 0 - wxID_OPENNEWTABLE - &Open New Table - - - m_buttonNewTable - protected - - - - - - - - - OnButtonNewTable - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 1 - - - - 0 - wxID_ANY - - - CPokerDialogBase - - 806,550 - wxDEFAULT_FRAME_STYLE|wxFRAME_NO_TASKBAR - - Poker - - - - wxFULL_REPAINT_ON_RESIZE|wxTAB_TRAVERSAL - 1 - - - - OnClose - - - - - - - - - - - - - - - - - OnMouseEvents - - OnPaint - - - - - OnSize - - - - bSizer174 - wxVERTICAL - none - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - Deal Me Out - - - m_checkSitOut - public - - - - - - - - - - OnCheckSitOut - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 0 - 1 - - - 0 - wxID_DEALHAND - &Deal Hand - - - m_buttonDealHand - protected - - 150,25 - - - - - - - OnButtonDealHand - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 0 - 1 - - - 0 - wxID_FOLD - &Fold - - - m_buttonFold - protected - - 80,25 - - - - - - - OnButtonFold - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 0 - 1 - - - 0 - wxID_CALL - &Call - - - m_buttonCall - protected - - 80,25 - - - - - - - OnButtonCall - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 0 - 1 - - - 0 - wxID_RAISE - &Raise - - - m_buttonRaise - protected - - 80,25 - - - - - - - OnButtonRaise - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 0 - 1 - - - 0 - wxID_LEAVETABLE - &Leave Table - - - m_buttonLeaveTable - protected - - 90,25 - - - - - - - OnButtonLeaveTable - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 1 - - - 0 - wxID_DITCHPLAYER - - 0 - - m_textDitchPlayer - protected - - 45,-1 - wxTE_PROCESS_ENTER - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - OnDitchPlayer - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - FOLD - - - m_checkPreFold - public - - 100,-1 - - - - - - - - OnCheckPreFold - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - CALL - - - m_checkPreCall - public - - 100,-1 - - - - - - - - OnCheckPreCall - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - CALL ANY - - - m_checkPreCallAny - public - - 100,-1 - - - - - - - - OnCheckPreCallAny - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - RAISE - - - m_checkPreRaise - public - - 100,-1 - - - - - - - - OnCheckPreRaise - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - RAISE ANY - - - m_checkPreRaiseAny - public - - 100,-1 - - - - - - - - OnCheckPreRaiseAny - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 1 - - 1 - - 0 - wxID_ANY - - - m_statusBar - public - - - wxST_SIZEGRIP - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - From 8b4cefd3245c349cad14baaf2ba42380dcc88112 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sat, 7 Nov 2009 05:05:03 +0000 Subject: [PATCH 024/133] UI tweaks, use BindListenPort to detect instance already running, setsockopt(SO_REUSEADDR) so can bind during TIME_WAIT after exit and restart git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@35 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 2 +- main.cpp | 8 +- main.h | 1 - net.cpp | 98 ++++++++--------- net.h | 6 +- ui.cpp | 122 +++++++++++++-------- ui.h | 2 +- uibase.cpp | 91 ++++++++-------- uibase.h | 2 +- uiproject.fbp | 287 ++++++++++++++++++++++++++------------------------ util.cpp | 11 ++ 11 files changed, 346 insertions(+), 284 deletions(-) diff --git a/db.cpp b/db.cpp index 12355926..61d60253 100644 --- a/db.cpp +++ b/db.cpp @@ -139,7 +139,7 @@ void DBFlush(bool fShutdown) { // Flush log data to the actual data file // on all files that are not in use - printf("DBFlush(%s)\n", fShutdown ? "true" : "false"); + printf("DBFlush(%s)%s\n", fShutdown ? "true" : "false", fDbEnvInit ? "" : " db not started"); if (!fDbEnvInit) return; CRITICAL_BLOCK(cs_db) diff --git a/main.cpp b/main.cpp index e4f1deb8..fe213c09 100644 --- a/main.cpp +++ b/main.cpp @@ -42,8 +42,6 @@ map > mapPubKeys; CCriticalSection cs_mapKeys; CKey keyUser; -int nDropMessagesTest = 0; - // Settings int fGenerateBitcoins = false; int64 nTransactionFee = 0; @@ -1721,9 +1719,9 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) static map > mapReuseKey; RandAddSeedPerfmon(); printf("received: %s (%d bytes)\n", strCommand.c_str(), vRecv.size()); - if (nDropMessagesTest > 0 && GetRand(nDropMessagesTest) == 0) + if (mapArgs.count("-dropmessagestest") && GetRand(atoi(mapArgs["-dropmessagestest"])) == 0) { - printf("dropmessages DROPPING RECV MESSAGE\n"); + printf("dropmessagestest DROPPING RECV MESSAGE\n"); return true; } @@ -2315,6 +2313,8 @@ void BitcoinMiner() Sleep(1000); if (fShutdown) return; + if (!fGenerateBitcoins) + return; } unsigned int nTransactionsUpdatedLast = nTransactionsUpdated; diff --git a/main.h b/main.h index 16b8c6a3..853fdfa5 100644 --- a/main.h +++ b/main.h @@ -34,7 +34,6 @@ extern int nBestHeight; extern uint256 hashBestChain; extern CBlockIndex* pindexBest; extern unsigned int nTransactionsUpdated; -extern int nDropMessagesTest; // Settings extern int fGenerateBitcoins; diff --git a/net.cpp b/net.cpp index 44a75a17..b4df35cb 100644 --- a/net.cpp +++ b/net.cpp @@ -511,11 +511,6 @@ void ThreadSocketHandler(void* parg) PrintException(NULL, "ThreadSocketHandler()"); } - foreach(CNode* pnode, vNodes) - closesocket(pnode->hSocket); - if (closesocket(hListenSocket) == SOCKET_ERROR) - printf("closesocket(hListenSocket) failed with error %d\n", WSAGetLastError()); - printf("ThreadSocketHandler exiting\n"); } @@ -989,15 +984,13 @@ void ThreadMessageHandler2(void* parg) - -bool StartNode(string& strError) +bool BindListenPort(string& strError) { - if (pnodeLocalHost == NULL) - pnodeLocalHost = new CNode(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); strError = ""; + int nOne = 1; #ifdef __WXMSW__ - // Sockets startup + // Initialize Windows Sockets WSADATA wsadata; int ret = WSAStartup(MAKEWORD(2,2), &wsadata); if (ret != NO_ERROR) @@ -1008,33 +1001,6 @@ bool StartNode(string& strError) } #endif - // Get local host ip - char pszHostName[255]; - if (gethostname(pszHostName, sizeof(pszHostName)) == SOCKET_ERROR) - { - strError = strprintf("Error: Unable to get IP address of this computer (gethostname returned error %d)", WSAGetLastError()); - printf("%s\n", strError.c_str()); - return false; - } - struct hostent* phostent = gethostbyname(pszHostName); - if (!phostent) - { - strError = strprintf("Error: Unable to get IP address of this computer (gethostbyname returned error %d)", WSAGetLastError()); - printf("%s\n", strError.c_str()); - return false; - } - - // Take the first IP that isn't loopback 127.x.x.x - for (int i = 0; phostent->h_addr_list[i] != NULL; i++) - printf("host ip %d: %s\n", i, CAddress(*(unsigned int*)phostent->h_addr_list[i]).ToStringIP().c_str()); - for (int i = 0; phostent->h_addr_list[i] != NULL; i++) - { - addrLocalHost = CAddress(*(unsigned int*)phostent->h_addr_list[i], DEFAULT_PORT, nLocalServices); - if (addrLocalHost.IsValid() && addrLocalHost.GetByte(3) != 127) - break; - } - printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); - // Create socket for listening for incoming connections hListenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (hListenSocket == INVALID_SOCKET) @@ -1043,15 +1009,21 @@ bool StartNode(string& strError) printf("%s\n", strError.c_str()); return false; } + #if defined(__BSD__) || defined(__WXOSX__) - int set = 1; - setsockopt(hSocket, SOL_SOCKET, SO_NOSIGPIPE, (void*)&set, sizeof(int)); + // Different way of disabling SIGPIPE on BSD + setsockopt(hListenSocket, SOL_SOCKET, SO_NOSIGPIPE, (void*)&nOne, sizeof(int)); +#endif + +#ifndef __WXMSW__ + // Allow binding if the port is still in TIME_WAIT state after + // the program was closed and restarted. Not an issue on windows. + setsockopt(hListenSocket, SOL_SOCKET, SO_REUSEADDR, (void*)&nOne, sizeof(int)); #endif - // Set to nonblocking, incoming connections will also inherit this #ifdef __WXMSW__ - u_long nOne = 1; - if (ioctlsocket(hListenSocket, FIONBIO, &nOne) == SOCKET_ERROR) + // Set to nonblocking, incoming connections will also inherit this + if (ioctlsocket(hListenSocket, FIONBIO, (u_long*)&nOne) == SOCKET_ERROR) #else if (fcntl(hListenSocket, F_SETFL, O_NONBLOCK) == SOCKET_ERROR) #endif @@ -1072,7 +1044,7 @@ bool StartNode(string& strError) { int nErr = WSAGetLastError(); if (nErr == WSAEADDRINUSE) - strError = strprintf("Error: Unable to bind to port %d on this computer. The program is probably already running.", ntohs(sockaddr.sin_port)); + strError = strprintf("Unable to bind to port %d on this computer. Bitcoin may be running already.", ntohs(sockaddr.sin_port)); else strError = strprintf("Error: Unable to bind to port %d on this computer (bind returned error %d)", ntohs(sockaddr.sin_port), nErr); printf("%s\n", strError.c_str()); @@ -1088,6 +1060,42 @@ bool StartNode(string& strError) return false; } + return true; +} + +bool StartNode(string& strError) +{ + strError = ""; + if (pnodeLocalHost == NULL) + pnodeLocalHost = new CNode(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); + + // Get local host ip + char pszHostName[255]; + if (gethostname(pszHostName, sizeof(pszHostName)) == SOCKET_ERROR) + { + strError = strprintf("Error: Unable to get IP address of this computer (gethostname returned error %d)", WSAGetLastError()); + printf("%s\n", strError.c_str()); + return false; + } + struct hostent* phostent = gethostbyname(pszHostName); + if (!phostent) + { + strError = strprintf("Error: Unable to get IP address of this computer (gethostbyname returned error %d)", WSAGetLastError()); + printf("%s\n", strError.c_str()); + return false; + } + + // Take the first IP that isn't loopback 127.x.x.x + for (int i = 0; phostent->h_addr_list[i] != NULL; i++) + printf("host ip %d: %s\n", i, CAddress(*(unsigned int*)phostent->h_addr_list[i]).ToStringIP().c_str()); + for (int i = 0; phostent->h_addr_list[i] != NULL; i++) + { + addrLocalHost = CAddress(*(unsigned int*)phostent->h_addr_list[i], DEFAULT_PORT, nLocalServices); + if (addrLocalHost.IsValid() && addrLocalHost.GetByte(3) != 127) + break; + } + printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); + // Get our external IP address for incoming connections if (fUseProxy) { @@ -1158,9 +1166,5 @@ bool StopNode() Sleep(20); Sleep(50); - // Sockets shutdown -#ifdef __WXMSW__ - WSACleanup(); -#endif return true; } diff --git a/net.h b/net.h index 024e7336..7b83d462 100644 --- a/net.h +++ b/net.h @@ -28,6 +28,7 @@ CNode* FindNode(unsigned int ip); CNode* ConnectNode(CAddress addrConnect, int64 nTimeout=0); void AbandonRequests(void (*fn)(void*, CDataStream&), void* param1); bool AnySubscribed(unsigned int nChannel); +bool BindListenPort(string& strError=REF(string())); bool StartNode(string& strError=REF(string())); bool StopNode(); @@ -456,6 +457,8 @@ extern CNode* pnodeLocalHost; extern uint64 nLocalHostNonce; extern bool fShutdown; extern array vnThreadsRunning; +extern SOCKET hListenSocket; + extern vector vNodes; extern CCriticalSection cs_vNodes; extern map, CAddress> mapAddresses; @@ -647,8 +650,7 @@ public: void EndMessage() { - extern int nDropMessagesTest; - if (nDropMessagesTest > 0 && GetRand(nDropMessagesTest) == 0) + if (mapArgs.count("-dropmessagestest") && GetRand(atoi(mapArgs["-dropmessagestest"])) == 0) { printf("dropmessages DROPPING SEND MESSAGE\n"); AbortMessage(); diff --git a/ui.cpp b/ui.cpp index d3302313..bfb0ad29 100644 --- a/ui.cpp +++ b/ui.cpp @@ -181,17 +181,30 @@ void AddToMyProducts(CProduct product) ""); } -void StringMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) +void CalledMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y, int* pnRet, bool* pfDone) { - wxMessageBox(message, caption, style, parent, x, y); + *pnRet = wxMessageBox(message, caption, style, parent, x, y); + *pfDone = true; } int ThreadSafeMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) { #ifdef __WXMSW__ - wxMessageBox(message, caption, style, parent, x, y); + return wxMessageBox(message, caption, style, parent, x, y); #else - UIThreadCall(bind(StringMessageBox, message, caption, style, parent, x, y)); + if (wxThread::IsMain()) + { + return wxMessageBox(message, caption, style, parent, x, y); + } + else + { + int nRet = 0; + bool fDone = false; + UIThreadCall(bind(CalledMessageBox, message, caption, style, parent, x, y, &nRet, &fDone)); + while (!fDone) + Sleep(100); + return nRet; + } #endif } @@ -303,6 +316,18 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) fOnSetFocusAddress = false; fRefresh = false; m_choiceFilter->SetSelection(0); +#ifndef __WXMSW__ + wxFont fontTmp = m_staticTextBalance->GetFont(); + fontTmp.SetPointSize(10); + fontTmp.SetFamily(wxFONTFAMILY_TELETYPE); + m_staticTextBalance->SetFont(fontTmp); + m_staticTextBalance->SetSize(140, 17); + // ampersand underlines aren't working on gtk + m_toolBar->ClearTools(); + m_toolBar->AddTool(wxID_BUTTONSEND, "Send Coins", wxBitmap(send20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + m_toolBar->AddTool(wxID_BUTTONRECEIVE, "Address Book", wxBitmap(addressbook20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + m_toolBar->Realize(); +#endif m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); SetIcon(wxICON(bitcoin)); @@ -998,7 +1023,7 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) } -void UIThreadCall(boost::function fn) +void UIThreadCall(boost::function0 fn) { // Call this with a function object created with bind. // bind needs all parameters to match the function's expected types @@ -1009,14 +1034,14 @@ void UIThreadCall(boost::function fn) if (pframeMain) { wxCommandEvent event(wxEVT_UITHREADCALL); - event.SetClientData((void*)new boost::function(fn)); + event.SetClientData((void*)new boost::function0(fn)); pframeMain->GetEventHandler()->AddPendingEvent(event); } } void CMainFrame::OnUIThreadCall(wxCommandEvent& event) { - boost::function* pfn = (boost::function*)event.GetClientData(); + boost::function0* pfn = (boost::function0*)event.GetClientData(); (*pfn)(); delete pfn; } @@ -1630,7 +1655,14 @@ CSendDialog::CSendDialog(wxWindow* parent, const wxString& strAddress) : CSendDi m_choiceTransferType->SetSelection(0); m_bitmapCheckMark->Show(false); fEnabledPrev = true; + m_textCtrlAddress->SetFocus(); //// todo: should add a display of your balance for convenience +#ifndef __WXMSW__ + wxFont fontTmp = m_staticTextInstructions->GetFont(); + fontTmp.SetPointSize(fontTmp.GetPointSize()-1); + m_staticTextInstructions->SetFont(fontTmp); + SetSize(725, wxDefaultCoord); +#endif // Set Icon wxIcon iconSend; @@ -1801,7 +1833,7 @@ CSendingDialog::CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 n fUIDone = false; fWorkDone = false; - SetTitle(strprintf("Sending %s to %s...", FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); + SetTitle(strprintf("Sending %s to %s", FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); m_textCtrlStatus->SetValue(""); _beginthread(SendingDialogStartTransfer, 0, this); @@ -3344,16 +3376,19 @@ IMPLEMENT_APP(CMyApp) bool CMyApp::OnInit() { + bool fRet = false; try { - return OnInit2(); + fRet = OnInit2(); } catch (std::exception& e) { PrintException(&e, "OnInit()"); } catch (...) { PrintException(NULL, "OnInit()"); } - return false; + if (!fRet) + Shutdown(NULL); + return fRet; } bool CMyApp::OnInit2() @@ -3374,6 +3409,9 @@ bool CMyApp::OnInit2() SetAppName("bitcoin"); #endif + // + // Parameters + // ParseParameters(argc, argv); if (mapArgs.count("-?") || mapArgs.count("--help")) { @@ -3389,7 +3427,27 @@ bool CMyApp::OnInit2() " -connect=\t Connect only to the specified node\n" " -?\t\t This help message\n"; wxMessageBox(strUsage, "Bitcoin", wxOK); - exit(0); + return false; + } + + if (mapArgs.count("-datadir")) + strlcpy(pszSetDataDir, mapArgs["-datadir"].c_str(), sizeof(pszSetDataDir)); + + if (mapArgs.count("-debug")) + fDebug = true; + + if (mapArgs.count("-printtodebugger")) + fPrintToDebugger = true; + + printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); + printf("Bitcoin version %d, OS version %s\n", VERSION, wxGetOsDescription().mb_str()); + + if (mapArgs.count("-loadblockindextest")) + { + CTxDB txdb("r"); + txdb.LoadBlockIndex(); + PrintBlockTree(); + return false; } // @@ -3434,41 +3492,20 @@ bool CMyApp::OnInit2() } #endif - // - // Parameters - // - if (mapArgs.count("-datadir")) - strlcpy(pszSetDataDir, mapArgs["-datadir"].c_str(), sizeof(pszSetDataDir)); - - if (mapArgs.count("-debug")) - fDebug = true; - - if (mapArgs.count("-printtodebugger")) - fPrintToDebugger = true; - - printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); - printf("Bitcoin version %d, OS version %s\n", VERSION, wxGetOsDescription().mb_str()); - - if (mapArgs.count("-dropmessages")) - { - nDropMessagesTest = atoi(mapArgs["-dropmessages"]); - if (nDropMessagesTest == 0) - nDropMessagesTest = 20; - } - - if (mapArgs.count("-loadblockindextest")) + // Bind to the port early so we can tell if another instance is already running. + // This is a backup to wxSingleInstanceChecker, which doesn't work on Linux. + string strErrors; + if (!BindListenPort(strErrors)) { - CTxDB txdb("r"); - txdb.LoadBlockIndex(); - PrintBlockTree(); - exit(0); + wxMessageBox(strErrors, "Bitcoin"); + return false; } // // Load data files // bool fFirstRun; - string strErrors; + strErrors = ""; int64 nStart; printf("Loading addresses...\n"); @@ -3502,7 +3539,6 @@ bool CMyApp::OnInit2() if (!strErrors.empty()) { wxMessageBox(strErrors, "Bitcoin"); - OnExit(); return false; } @@ -3515,7 +3551,6 @@ bool CMyApp::OnInit2() if (mapArgs.count("-printblockindex") || mapArgs.count("-printblocktree")) { PrintBlockTree(); - OnExit(); return false; } @@ -3539,7 +3574,6 @@ bool CMyApp::OnInit2() } if (nFound == 0) printf("No blocks matching %s were found\n", strMatch.c_str()); - OnExit(); return false; } @@ -3558,7 +3592,6 @@ bool CMyApp::OnInit2() if (!addrProxy.IsValid()) { wxMessageBox("Invalid -proxy address", "Bitcoin"); - OnExit(); return false; } } @@ -3588,10 +3621,7 @@ bool CMyApp::OnInit2() _beginthread(ThreadDelayedRepaint, 0, NULL); if (!CheckDiskSpace()) - { - OnExit(); return false; - } RandAddSeedPerfmon(); diff --git a/ui.h b/ui.h index 1db40997..a919c366 100644 --- a/ui.h +++ b/ui.h @@ -24,7 +24,7 @@ extern int fMinimizeOnClose; extern void HandleCtrlA(wxKeyEvent& event); extern string FormatTxStatus(const CWalletTx& wtx); -extern void UIThreadCall(boost::function); +extern void UIThreadCall(boost::function0); extern void MainFrameRepaint(); extern void Shutdown(void* parg); extern int ThreadSafeMessageBox(const string& message, const string& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1); diff --git a/uibase.cpp b/uibase.cpp index 6a280cda..7bc8081f 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -89,7 +89,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& bSizer85->Add( m_textCtrlAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT ); + m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT(" &Copy to Clipboard "), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT ); bSizer85->Add( m_buttonCopy, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); m_button91 = new wxButton( this, wxID_BUTTONCHANGE, wxT("C&hange..."), wxDefaultPosition, wxDefaultSize, 0 ); @@ -116,7 +116,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_staticTextBalance = new wxStaticText( m_panel14, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( 120,15 ), wxALIGN_RIGHT|wxST_NO_AUTORESIZE ); m_staticTextBalance->Wrap( -1 ); m_staticTextBalance->SetFont( wxFont( 8, 70, 90, 90, false, wxEmptyString ) ); - m_staticTextBalance->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW ) ); + m_staticTextBalance->SetBackgroundColour( wxColour( 255, 255, 255 ) ); bSizer66->Add( m_staticTextBalance, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); @@ -341,10 +341,10 @@ CTxDetailsDialogBase::CTxDetailsDialogBase( wxWindow* parent, wxWindowID id, con bSizer64->Add( bSizer66, 1, wxEXPAND, 5 ); wxBoxSizer* bSizer65; - bSizer65 = new wxBoxSizer( wxVERTICAL ); + bSizer65 = new wxBoxSizer( wxHORIZONTAL ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer65->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer65->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer64->Add( bSizer65, 0, wxALIGN_RIGHT, 5 ); @@ -521,13 +521,13 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer58 = new wxBoxSizer( wxHORIZONTAL ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer58->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer58->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer58->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer58->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonApply = new wxButton( this, wxID_APPLY, wxT("&Apply"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer58->Add( m_buttonApply, 0, wxALL, 5 ); + bSizer58->Add( m_buttonApply, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer55->Add( bSizer58, 0, wxALIGN_RIGHT, 5 ); @@ -619,7 +619,7 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr bSizer61->Add( 0, 0, 1, wxEXPAND, 5 ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer61->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer61->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer60->Add( bSizer61, 0, wxALIGN_RIGHT|wxEXPAND, 5 ); @@ -655,9 +655,9 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin fgSizer1->Add( 0, 0, 0, wxEXPAND, 5 ); - m_staticText14 = new wxStaticText( this, wxID_ANY, wxT("Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, \nor Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online."), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText14->Wrap( -1 ); - fgSizer1->Add( m_staticText14, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); + m_staticTextInstructions = new wxStaticText( this, wxID_ANY, wxT("Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, \nor Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextInstructions->Wrap( -1 ); + fgSizer1->Add( m_staticTextInstructions, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); wxBoxSizer* bSizer47; bSizer47 = new wxBoxSizer( wxHORIZONTAL ); @@ -681,11 +681,16 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin m_textCtrlAddress = new wxTextCtrl( this, wxID_TEXTCTRLPAYTO, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); bSizer19->Add( m_textCtrlAddress, 1, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); + wxBoxSizer* bSizer66; + bSizer66 = new wxBoxSizer( wxHORIZONTAL ); + m_buttonPaste = new wxButton( this, wxID_BUTTONPASTE, wxT("&Paste"), wxDefaultPosition, wxSize( -1,-1 ), wxBU_EXACTFIT ); - bSizer19->Add( m_buttonPaste, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); + bSizer66->Add( m_buttonPaste, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxEXPAND, 5 ); m_buttonAddress = new wxButton( this, wxID_BUTTONADDRESSBOOK, wxT(" Address &Book..."), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer19->Add( m_buttonAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); + bSizer66->Add( m_buttonAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxEXPAND, 5 ); + + bSizer19->Add( bSizer66, 0, wxALIGN_CENTER_VERTICAL, 5 ); fgSizer1->Add( bSizer19, 1, wxEXPAND|wxRIGHT, 5 ); @@ -764,10 +769,10 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin m_buttonSend = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); m_buttonSend->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); - bSizer23->Add( m_buttonSend, 0, wxALL, 5 ); + bSizer23->Add( m_buttonSend, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer23->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer23->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer21->Add( bSizer23, 0, wxEXPAND, 5 ); @@ -827,10 +832,10 @@ CSendingDialogBase::CSendingDialogBase( wxWindow* parent, wxWindowID id, const w m_buttonOK = new wxButton( this, wxID_ANY, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); m_buttonOK->Enable( false ); - bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer69->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer69->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); @@ -877,21 +882,21 @@ CYourAddressDialogBase::CYourAddressDialogBase( wxWindow* parent, wxWindowID id, bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); m_buttonRename = new wxButton( this, wxID_BUTTONRENAME, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer69->Add( m_buttonRename, 0, wxALL, 5 ); + bSizer69->Add( m_buttonRename, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer69->Add( m_buttonNew, 0, wxALL, 5 ); + m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT(" &New Address... "), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + bSizer69->Add( m_buttonNew, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT("&Copy to Clipboard"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer69->Add( m_buttonCopy, 0, wxALL, 5 ); + m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT(" &Copy to Clipboard "), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + bSizer69->Add( m_buttonCopy, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer69->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); m_buttonCancel->Hide(); - bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer69->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); @@ -950,19 +955,19 @@ CAddressBookDialogBase::CAddressBookDialogBase( wxWindow* parent, wxWindowID id, bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); m_buttonEdit = new wxButton( this, wxID_BUTTONEDIT, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer69->Add( m_buttonEdit, 0, wxALL, 5 ); + bSizer69->Add( m_buttonEdit, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT("&New Address..."), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer69->Add( m_buttonNew, 0, wxALL, 5 ); + m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT(" &New Address... "), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer69->Add( m_buttonNew, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonDelete = new wxButton( this, wxID_BUTTONDELETE, wxT("&Delete"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer69->Add( m_buttonDelete, 0, wxALL, 5 ); + bSizer69->Add( m_buttonDelete, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer69->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer69->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer69->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer69->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); @@ -1360,13 +1365,13 @@ CEditProductDialogBase::CEditProductDialogBase( wxWindow* parent, wxWindowID id, bSizer26 = new wxBoxSizer( wxHORIZONTAL ); m_buttonOK = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer26->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonPreview = new wxButton( this, wxID_BUTTONPREVIEW, wxT("&Preview"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonPreview, 0, wxALL, 5 ); + bSizer26->Add( m_buttonPreview, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer26->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); @@ -1551,10 +1556,10 @@ CViewProductDialogBase::CViewProductDialogBase( wxWindow* parent, wxWindowID id, bSizer25 = new wxBoxSizer( wxHORIZONTAL ); m_buttonSubmitForm = new wxButton( m_scrolledWindow, wxID_BUTTONSAMPLE, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer25->Add( m_buttonSubmitForm, 0, wxALL, 5 ); + bSizer25->Add( m_buttonSubmitForm, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancelForm = new wxButton( m_scrolledWindow, wxID_CANCEL2, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer25->Add( m_buttonCancelForm, 0, wxALL, 5 ); + bSizer25->Add( m_buttonCancelForm, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer21->Add( bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5 ); @@ -1571,13 +1576,13 @@ CViewProductDialogBase::CViewProductDialogBase( wxWindow* parent, wxWindowID id, m_buttonBack = new wxButton( this, wxID_BUTTONBACK, wxT("< &Back "), wxDefaultPosition, wxDefaultSize, 0 ); m_buttonBack->Enable( false ); - bSizer26->Add( m_buttonBack, 0, wxALL, 5 ); + bSizer26->Add( m_buttonBack, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonNext = new wxButton( this, wxID_BUTTONNEXT, wxT(" &Next >"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonNext, 0, wxALL, 5 ); + bSizer26->Add( m_buttonNext, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer26->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); @@ -1622,7 +1627,7 @@ CViewOrderDialogBase::CViewOrderDialogBase( wxWindow* parent, wxWindowID id, con bSizer26 = new wxBoxSizer( wxHORIZONTAL ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer26->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); @@ -1678,10 +1683,10 @@ CEditReviewDialogBase::CEditReviewDialogBase( wxWindow* parent, wxWindowID id, c bSizer113 = new wxBoxSizer( wxHORIZONTAL ); m_buttonSubmit = new wxButton( this, wxID_SUBMIT, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer113->Add( m_buttonSubmit, 0, wxALL, 5 ); + bSizer113->Add( m_buttonSubmit, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer113->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer113->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer112->Add( bSizer113, 0, wxALIGN_RIGHT, 5 ); @@ -1745,10 +1750,10 @@ CGetTextFromUserDialogBase::CGetTextFromUserDialogBase( wxWindow* parent, wxWind bSizer80->Add( 0, 0, 1, wxEXPAND, 5 ); m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); - bSizer80->Add( m_buttonOK, 0, wxALL, 5 ); + bSizer80->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer80->Add( m_buttonCancel, 0, wxALL, 5 ); + bSizer80->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer79->Add( bSizer80, 0, wxEXPAND, 5 ); diff --git a/uibase.h b/uibase.h index 6332b931..2cb99e9a 100644 --- a/uibase.h +++ b/uibase.h @@ -277,7 +277,7 @@ class CSendDialogBase : public wxDialog protected: - wxStaticText* m_staticText14; + wxStaticText* m_staticTextInstructions; wxStaticBitmap* m_bitmapCheckMark; wxStaticText* m_staticText36; diff --git a/uiproject.fbp b/uiproject.fbp index 7c5bb24f..3aa1c86b 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -225,7 +225,7 @@ - + 20,20 @@ -489,7 +489,7 @@ 0 wxID_BUTTONCOPY - &Copy to Clipboard + &Copy to Clipboard m_buttonCopy @@ -708,7 +708,7 @@ wxALIGN_CENTER_VERTICAL|wxALL 0 - wxSYS_COLOUR_WINDOW + 255,255,255 1 @@ -1627,11 +1627,11 @@ bSizer65 - wxVERTICAL + wxHORIZONTAL none 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -3014,7 +3014,7 @@ none 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -3066,7 +3066,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -3118,7 +3118,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -3476,7 +3476,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -3638,7 +3638,7 @@ Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, or Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online. - m_staticText14 + m_staticTextInstructions protected @@ -3861,106 +3861,117 @@ 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT + wxALIGN_CENTER_VERTICAL 0 - - - - 0 - 1 - - - 0 - wxID_BUTTONPASTE - &Paste - - - m_buttonPaste - protected - - -1,-1 - wxBU_EXACTFIT - - - - - - OnButtonPaste - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT - 0 - - - - 0 - 1 - - - 0 - wxID_BUTTONADDRESSBOOK - Address &Book... - + - m_buttonAddress - protected - - - - - - - - - OnButtonAddressBook - - - - - - - - - - - - - - - - - - - - - - - + bSizer66 + wxHORIZONTAL + none + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxEXPAND + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONPASTE + &Paste + + + m_buttonPaste + protected + + -1,-1 + wxBU_EXACTFIT + + + + + + OnButtonPaste + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_CENTER_VERTICAL|wxRIGHT|wxEXPAND + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONADDRESSBOOK + Address &Book... + + + m_buttonAddress + protected + + + + + + + + + OnButtonAddressBook + + + + + + + + + + + + + + + + + + + + + + + + + @@ -4472,7 +4483,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -4524,7 +4535,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -4762,7 +4773,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -4814,7 +4825,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5076,7 +5087,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5128,7 +5139,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5139,7 +5150,7 @@ 0 wxID_BUTTONNEW - &New Address... + &New Address... -1,-1 m_buttonNew @@ -5180,7 +5191,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5191,7 +5202,7 @@ 0 wxID_BUTTONCOPY - &Copy to Clipboard + &Copy to Clipboard -1,-1 m_buttonCopy @@ -5232,7 +5243,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5284,7 +5295,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5546,7 +5557,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5598,7 +5609,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5609,7 +5620,7 @@ 0 wxID_BUTTONNEW - &New Address... + &New Address... -1,-1 m_buttonNew @@ -5650,7 +5661,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5702,7 +5713,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -5754,7 +5765,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10212,7 +10223,7 @@ none 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10264,7 +10275,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10316,7 +10327,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10665,7 +10676,7 @@ none 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10717,7 +10728,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10785,7 +10796,7 @@ none 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10837,7 +10848,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -10889,7 +10900,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -11074,7 +11085,7 @@ none 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -11475,7 +11486,7 @@ none 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -11527,7 +11538,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -11902,7 +11913,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 @@ -11954,7 +11965,7 @@ 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 diff --git a/util.cpp b/util.cpp index 5efb579b..4a5b983d 100644 --- a/util.cpp +++ b/util.cpp @@ -53,6 +53,17 @@ public: for (int i = 0; i < CRYPTO_num_locks(); i++) delete ppmutexOpenSSL[i]; OPENSSL_free(ppmutexOpenSSL); + + // Close sockets + foreach(CNode* pnode, vNodes) + closesocket(pnode->hSocket); + if (closesocket(hListenSocket) == SOCKET_ERROR) + printf("closesocket(hListenSocket) failed with error %d\n", WSAGetLastError()); + +#ifdef __WXMSW__ + // Shutdown Windows Sockets + WSACleanup(); +#endif } } instance_of_cinit; From b4e235f9f23b492539eeeddbbd1e7a7ef1b6793e Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sat, 7 Nov 2009 18:51:41 +0000 Subject: [PATCH 025/133] ui tweaks, multiple xpm icon sizes git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@36 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 5 +- net.cpp | 4 +- ui.cpp | 18 ++- xpm/bitcoin.xpm | 304 ---------------------------------------------- xpm/bitcoin16.xpm | 203 +++++++++++++++++++++++++++++++ xpm/bitcoin20.xpm | 226 ++++++++++++++++++++++++++++++++++ xpm/bitcoin32.xpm | 223 ++++++++++++++++++++++++++++++++++ xpm/bitcoin48.xpm | 278 ++++++++++++++++++++++++++++++++++++++++++ 8 files changed, 948 insertions(+), 313 deletions(-) delete mode 100644 xpm/bitcoin.xpm create mode 100644 xpm/bitcoin16.xpm create mode 100644 xpm/bitcoin20.xpm create mode 100644 xpm/bitcoin32.xpm create mode 100644 xpm/bitcoin48.xpm diff --git a/headers.h b/headers.h index 6cd2da07..34eafb26 100644 --- a/headers.h +++ b/headers.h @@ -101,7 +101,10 @@ using namespace boost; #include "xpm/addressbook16.xpm" #include "xpm/addressbook20.xpm" -#include "xpm/bitcoin.xpm" +#include "xpm/bitcoin16.xpm" +#include "xpm/bitcoin20.xpm" +#include "xpm/bitcoin32.xpm" +#include "xpm/bitcoin48.xpm" #include "xpm/check.xpm" #include "xpm/send16.xpm" #include "xpm/send16noshadow.xpm" diff --git a/net.cpp b/net.cpp index b4df35cb..d70a62a2 100644 --- a/net.cpp +++ b/net.cpp @@ -1044,13 +1044,13 @@ bool BindListenPort(string& strError) { int nErr = WSAGetLastError(); if (nErr == WSAEADDRINUSE) - strError = strprintf("Unable to bind to port %d on this computer. Bitcoin may be running already.", ntohs(sockaddr.sin_port)); + strError = strprintf("Unable to bind to port %d on this computer. Bitcoin is probably already running.", ntohs(sockaddr.sin_port)); else strError = strprintf("Error: Unable to bind to port %d on this computer (bind returned error %d)", ntohs(sockaddr.sin_port), nErr); printf("%s\n", strError.c_str()); return false; } - printf("bound to port %d\n", ntohs(sockaddr.sin_port)); + printf("Bound to port %d\n", ntohs(sockaddr.sin_port)); // Listen for incoming connections if (listen(hListenSocket, SOMAXCONN) == SOCKET_ERROR) diff --git a/ui.cpp b/ui.cpp index bfb0ad29..a1873d5c 100644 --- a/ui.cpp +++ b/ui.cpp @@ -316,13 +316,15 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) fOnSetFocusAddress = false; fRefresh = false; m_choiceFilter->SetSelection(0); -#ifndef __WXMSW__ - wxFont fontTmp = m_staticTextBalance->GetFont(); - fontTmp.SetPointSize(10); +#ifdef __WXMSW__ + SetIcon(wxICON(bitcoin)); +#else + SetIcon(bitcoin16_xpm); + wxFont fontTmp = m_staticText41->GetFont(); fontTmp.SetFamily(wxFONTFAMILY_TELETYPE); m_staticTextBalance->SetFont(fontTmp); m_staticTextBalance->SetSize(140, 17); - // ampersand underlines aren't working on gtk + // & underlines don't work on the toolbar buttons on gtk m_toolBar->ClearTools(); m_toolBar->AddTool(wxID_BUTTONSEND, "Send Coins", wxBitmap(send20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); m_toolBar->AddTool(wxID_BUTTONRECEIVE, "Address Book", wxBitmap(addressbook20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); @@ -330,7 +332,6 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) #endif m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); - SetIcon(wxICON(bitcoin)); ptaskbaricon = new CMyTaskBarIcon(); // Init column headers @@ -1659,7 +1660,8 @@ CSendDialog::CSendDialog(wxWindow* parent, const wxString& strAddress) : CSendDi //// todo: should add a display of your balance for convenience #ifndef __WXMSW__ wxFont fontTmp = m_staticTextInstructions->GetFont(); - fontTmp.SetPointSize(fontTmp.GetPointSize()-1); + if (fontTmp.GetPointSize() > 9); + fontTmp.SetPointSize(9); m_staticTextInstructions->SetFont(fontTmp); SetSize(725, wxDefaultCoord); #endif @@ -3270,7 +3272,11 @@ void CMyTaskBarIcon::Show(bool fShow) if (strncmp(pszPrevTip, strTooltip.c_str(), sizeof(pszPrevTip)-1) != 0) { strlcpy(pszPrevTip, strTooltip.c_str(), sizeof(pszPrevTip)); +#ifdef __WXMSW__ SetIcon(wxICON(bitcoin), strTooltip); +#else + SetIcon(bitcoin20_xpm, strTooltip); +#endif } } else diff --git a/xpm/bitcoin.xpm b/xpm/bitcoin.xpm deleted file mode 100644 index 166d5aa6..00000000 --- a/xpm/bitcoin.xpm +++ /dev/null @@ -1,304 +0,0 @@ -/* XPM */ -static char * bitcoin_xpm[] = { -/* columns rows colors chars-per-pixel */ -"48 48 250 2", -" c #725203", -". c #795603", -"X c #7D5903", -"o c #76560B", -"O c #77590E", -"+ c #795A0D", -"@ c #7B5D14", -"# c #7C5F18", -"$ c #7D6019", -"% c #825D05", -"& c #856007", -"* c #86620B", -"= c #8B660B", -"- c #8E690E", -"; c #906A0F", -": c #8F6B17", -"> c #83641C", -", c #8D6C1E", -"< c #926C11", -"1 c #967014", -"2 c #997215", -"3 c #9C761B", -"4 c #9E791D", -"5 c #A37C1E", -"6 c #816520", -"7 c #876A25", -"8 c #8E6E22", -"9 c #866A29", -"0 c #896E2C", -"q c #8E7020", -"w c #937324", -"e c #997722", -"r c #9E7B25", -"t c #94762B", -"y c #967828", -"u c #9A7B2D", -"i c #8B7131", -"p c #9E7E31", -"a c #947839", -"s c #A37D22", -"d c #A68125", -"f c #AA8325", -"g c #AE8827", -"h c #A6832D", -"j c #AA852B", -"k c #AD892B", -"l c #B08727", -"z c #B28827", -"x c #B08729", -"c c #B38B2C", -"v c #B88E2F", -"b c #B8902D", -"n c #A38334", -"m c #A98632", -"M c #AB8A34", -"N c #A4873C", -"B c #A78A3D", -"V c #AC8B3C", -"C c #B38D32", -"Z c #BA8F30", -"A c #B28E3C", -"S c #B69332", -"D c #BC9433", -"F c #BF9832", -"G c #B4923C", -"H c #BA963D", -"J c #B7993E", -"K c #BE9A3B", -"L c #C1932F", -"P c #C39732", -"I c #C49935", -"U c #C59C3A", -"Y c #C99E3D", -"T c #C2A13F", -"R c #CDA23F", -"E c #9D8342", -"W c #AB8C43", -"Q c #B28E40", -"! c #AE9144", -"~ c #AE914A", -"^ c #B49445", -"/ c #BC9B44", -"( c #B3964D", -") c #B5994C", -"_ c #BD9B4A", -"` c #A98F50", -"' c #B19553", -"] c #B59A54", -"[ c #BD9F51", -"{ c #B59B5C", -"} c #B89D5C", -"| c #BEA155", -" . c #BDA35D", -".. c #B59C61", -"X. c #B99F66", -"o. c #BCA363", -"O. c #BDA56C", -"+. c #BCA571", -"@. c #BDA873", -"#. c #BFAA78", -"$. c #C49D43", -"%. c #C99F45", -"&. c #C29E4B", -"*. c #C5A144", -"=. c #CCA244", -"-. c #C5A44B", -";. c #CAA54B", -":. c #C8A84C", -">. c #D0A644", -",. c #D3AA44", -"<. c #D3AC4C", -"1. c #D8AD4D", -"2. c #DAB046", -"3. c #DCB24E", -"4. c #C3A454", -"5. c #CBA751", -"6. c #CCAA53", -"7. c #C1A65B", -"8. c #C8A75A", -"9. c #CBAC5B", -"0. c #D0A650", -"q. c #D2AC53", -"w. c #DAAD54", -"e. c #D3AD5C", -"r. c #CFB259", -"t. c #D4B156", -"y. c #DDB454", -"u. c #D4B25C", -"i. c #DAB65A", -"p. c #D7B95F", -"a. c #DEBA5E", -"s. c #E2B555", -"d. c #E5BA53", -"f. c #E1B55A", -"g. c #E5BC5C", -"h. c #EABF5D", -"j. c #C1A761", -"k. c #C4AA63", -"l. c #CBAE63", -"z. c #CBB166", -"x. c #CBB26C", -"c. c #D4B263", -"v. c #DAB462", -"b. c #D6B864", -"n. c #DCB965", -"m. c #D3B669", -"M. c #DCB768", -"N. c #D4BA6E", -"B. c #DCBB6C", -"V. c #CDB672", -"C. c #D2B972", -"Z. c #DBBE72", -"A. c #E4BC62", -"S. c #E9BE62", -"D. c #E2BD6C", -"F. c #E0BF72", -"G. c #E6C05E", -"H. c #EFC05D", -"J. c #F0C15B", -"K. c #DFC167", -"L. c #D7C069", -"P. c #DDC36D", -"I. c #DBC376", -"U. c #D4C17B", -"Y. c #DAC17B", -"T. c #D8C878", -"R. c #E4C362", -"E. c #EBC364", -"W. c #E3C865", -"Q. c #EDC866", -"!. c #E4C36A", -"~. c #E9C66B", -"^. c #ECCA6B", -"/. c #F1C564", -"(. c #F8C765", -"). c #F5CB66", -"_. c #F8CC67", -"`. c #F6CC6A", -"'. c #F9CD6B", -"]. c #EED26A", -"[. c #F2D06F", -"{. c #FBD26D", -"}. c #E4C374", -"|. c #EBC474", -" X c #E1C972", -".X c #EDCD72", -"XX c #E4C57A", -"oX c #E9C67C", -"OX c #E5C87C", -"+X c #EACA7D", -"@X c #F2CC74", -"#X c #FBCF71", -"$X c #EED174", -"%X c #ECD37B", -"&X c #F4D274", -"*X c #FDD473", -"=X c #FFD975", -"-X c #F4D57C", -";X c #FCD57A", -":X c #F3DA7C", -">X c #FEDB7C", -",X c #FFE37D", -"X=XQ.s.=.v 5 1 < E HXHXHXHXHXHXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHXHXHX' g f F d.).).{.{.=X=X=X{.{.{.`.`.`.).g.U f 2 * a HXHXHXHXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHXHXV f b G.J.{.{.{.*X=X,X=X*X{.`.).`.).).{.`.{./.U 5 ; + HXHXHXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHX} h g 1.)._._.{.,X*X=X,X{.{.)._.).).`.`.`.{.*X*X*X`.y.g 2 & $ HXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHX{ j c G.).{.{.{.=X,X{.{.J.d.2.R 2.,.3.g.`.&X;X;X;X&X[.{.`.I 3 & + HXHXHXHXHXHXHXHX", -"HXHXHXHXHXHX{ d D /.{.{.*X=X=X*X).3.R I I I P F F U $.;.n.-XrXrX;X&X;X>XdX:.4 X o HXHXHXHXHXHXHX", -"HXHXHXHXHX..j v /.*X=X=X=X=X`.1.R R R R I I I P K U *.e.D.|.}.+XrXqXhXhXdXfX:.4 X o HXHXHXHXHXHX", -"HXHXHXHXHXh v `.{.>X,X*X{.g.>.Y R I I I I I I U U ;.t.D.|.oXB.z.F.kXvXcXjXjXjX/ 1 . + HXHXHXHXHX", -"HXHXHXHXV g Q.=X=X>X>X'.>.Y Y U R I I I P P I U U <.n.~.}.F.XXXX}.%XbXbXcXcXcXsXc = # HXHXHXHX", -"HXHXHX} j t.>X>X>X*X'.>.U U I U P U U I P P I U T 6.M.D.oX5XwXeXeXqX0XvXbXbXcXjXW.5 % HXHXHXHX", -"HXHXHXM G hXhXqX>X*X<.U U U I I I I I I D D U T T -.9.B.3XpXpXzXgXqX:X:XbXbXcXjXfXT < o HXHXHX", -"HXHX} k XlXkXkXrXA.$.D Z Z Z v v v b b v D U U *.-.9.B.OX2XOXI.P.L.K.W.$XbXcXjX,X].d % 9 HXHX", -"HXHXV J xXxXxXxXrX5.&.A M m m m h h n s 5 g S K *.:.8.4.| k.x.C.N.z.7.) :.$XjXfX,X,XT ; o HXHX", -"HXHXM L.vXxXxXxXF._ MXCXCXCXCXCXCXCXCXmXY.h g K *./ ^ Y.mXCXCXCXCXCXCXVXZ.4.hXfX,X,XW.4 X HXHX", -"HX] k gXxXxXxXgXe.V MXCXCXCXCXVXCXCXCXCXCXyXh D G [ mXCXCXCXCXCXCXCXCXCX4XG ~.fX,X,X,Xg & $ HX", -"HXV J vXxXxXxX6Xe.V MXCXCXCXCXk.N VXCXCXCXVX| h ^ MXCXCXCXCXuX( n V l.mX4XA y.fX,X,X=XT ; HX", -"HXk r.xXxXxXxX|.v.V MXCXCXCXCXo.> 4XCXCXCXCXx.w tXCXCXCXCXnXn V / / M V m.&.t.=X,X,X=Xy.2 o HX", -"HXk P.xXvXxXxX|.M.Q nXCXCXCXCXj.w X>X=XH.5 X $ ", -"o.k %XvXbXBXkX|.D.Q nXCXCXCXCXj., 7XCXCXCXCX~ k.CXCXCXCXCXV &.n.R.g.G.g.S.S.S.(.qX*X=X`.5 X $ ", -" .k 0XvXvXvXrX@XD.^ VXCXCXCXCXx.~ VXCXCXCXX=X*X*Xd X $ ", -"{ k sXcXvXBXeX@XD.( nXCXCXCXCXCXCXCXCXVXo., u T.CXCXCXVXmXn t.E.g.g.h.S.g.S.f.A.>X;X*X*Xf & o ", -"{ k sXjXlXcX0X~.n.^ MXCXCXCXCXCXCXCXCXCXCXU.t U.CXCXCXCXmXM p.~.W.g.s.s.s.s.f.A.>X*X*X*Xj % ", -"o.k :XjXlXlX-XD.v.A MXCXCXCXCXj.t mXCXCXCXCX .x.CXCXCXCXVXV p.$X^.E.g.s.w.w.w.A.9X;X*X*Xf X ", -"o.g ].dXjXhX-Xn.e.V MXCXCXCXCXj.8 X*X#X5 X O ", -"HXj K.dXdXhX9Xv.9.M MXCXCXCXCXk.a Z.CXCXCXCX7Xu CXCXCXCXCXV./ !.$X~.f.5.%.0.q.S.>X*X*XE.5 X # ", -"HXj t.dX,XdXdXi.6.N MXCXCXCXCXo.q XqXqXqX!.6.m MXCXCXCXCXV.' VXCXCXZXCXk.! ] VXCXCXCXCXVXC.[ 7.Z.VX2Xx %.#X#X'.'.%.- o 6 ", -"HX( g &X>X>XdX-X5.M MXCXCXCXCXCXCXCXCXCXCX7X) m.9. .MXCXCXCXCXCXCXCXCXCX2Xs 1.'.`.'.'.x % HX", -"HXO.j y.>X>X>X>X6.! zXMXMXMXMXMXMXMXMXuXx.( N.8X6Xz.) C.uXCXCXCXCXCXVX7X4.c h.'.(.(.s.5 X HX", -"HXHXj H &X=X:X>X~./ V h y u n n n N W ( 7.Z.8XpXpX+Xm.| V V ^ ) ( m e 3 s R (.(.'.'.Y ; . > HX", -"HXHX} f G.&X&X&X:Xt._ / ) _ 4.8.l.m.B.Z.2XwXpXeXwX6X+XP.c.8.-./ C x x z P J.(.'.(./.5 % HXHX", -"HXHXHXh D &X&X&X&X@X:.4.5.9.c.m.F.OX+XwXwXwXwX6X3XOX}.D.D.v.w.%.Y I P P d.(.(.'.'.=.< . + HXHX", -"HXHXHX] d y.[.&X&X&X~.:.4.9.e.M.B.}.oX3X5X3X3X+X}.F.M.e.0.0.0.%.%.Y Y s.#X#X#X#XS.s % HXHXHX", -"HXHXHXHXm g `.@X&X&X&X~.5.6.e.b.B.}.XX+X3XOX}.I.F.F.D.e.e.e.e.e.q.0.A.;X;X#X-X@XZ = o + HXHXHX", -"HXHXHXHX..j D @X&X&X&X&X@Xp.u.M.D.}.XXOXOXZ.Z.XX1XOXoXoXF.F.F.M.D.6XrXqX9X9X9X%.1 . HXHXHXHX", -"HXHXHXHXHX' f $.&X&X&X>XqXqX XB.D.!.XXXXZ.XXOX5XwXwXwXwXiXwXwXnXVXZXBXxXzXxXb.r X $ HXHXHXHX", -"HXHXHXHXHXHX~ j ;.qXqXqXqXrXkXrX+XD.Z.Z.XX1X2X5X5X5XwXwXiXnXCXGXGXFXFXSXAXT.s % @ HXHXHXHXHX", -"HXHXHXHXHXHXHX~ j -.0XrXzXxXzXzXzXzXwX3XXXXXOX1X2X5XpXmXAXFXGXGXGXFXFXSXL.r % O HXHXHXHXHXHX", -"HXHXHXHXHXHXHXHX! j / gXSXSXZXxXzXzXkXxXzXBXBXBXZXCXAXAXAXAXSXSXSXSXNX| 3 & O HXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHX} n V U.DXSXBXzXkXkXkXxXxXxXBXxXBXBXCXZXZXZXAXAXU.M < . @ HXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHXHXHXV d G Z.pXzXkXzXjXkXkXxXzXxXxXxXBXBXZXNXT.G 3 * . 9 HXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHXHXHXHX@.u s k -.K.6XhXjXhXkXlXzXzXzXeXOX9.k 3 = X O HXHXHXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHXHXHXHXHXHX..w < s j k K -.;.:.-./ C j 4 < & . O HXHXHXHXHXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXE * = - - < < - = & X . 0 HXHXHXHXHXHXHXHXHXHXHXHXHXHXHX", -"HXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXi 7 7 @ o o O > 0 i HXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHXHX" -}; diff --git a/xpm/bitcoin16.xpm b/xpm/bitcoin16.xpm new file mode 100644 index 00000000..8bec142c --- /dev/null +++ b/xpm/bitcoin16.xpm @@ -0,0 +1,203 @@ +/* XPM */ +static char * bitcoin16_xpm[] = { +/* columns rows colors chars-per-pixel */ +"16 16 181 2", +" c #775605", +". c #745507", +"X c #785806", +"o c #7D5A06", +"O c #745508", +"+ c #755508", +"@ c #755608", +"# c #775708", +"$ c #77580B", +"% c #7A5A0B", +"& c #7D5E14", +"* c #86630D", +"= c #8D6B16", +"- c #886818", +"; c #927019", +": c #91701E", +"> c #9A751E", +", c #957627", +"< c #9A7925", +"1 c #987827", +"2 c #98782B", +"3 c #9D7C2F", +"4 c #9E7E31", +"5 c #A57D22", +"6 c #A88227", +"7 c #B78D2E", +"8 c #B78F2E", +"9 c #A78430", +"0 c #A48733", +"q c #A68536", +"w c #A98937", +"e c #B98F31", +"r c #B49233", +"t c #B39337", +"y c #C09837", +"u c #C39936", +"i c #C49936", +"p c #C69C39", +"a c #C89C3A", +"s c #CCA23A", +"d c #AF9240", +"f c #B99644", +"g c #BE9C46", +"h c #BF9D4A", +"j c #BA9F58", +"k c #BEA04B", +"l c #BBA253", +"z c #BAA057", +"x c #C29E44", +"c c #C2A144", +"v c #CAA246", +"b c #CCA344", +"n c #CEA645", +"m c #C2A04A", +"M c #C7A349", +"N c #C5A34A", +"B c #C7A64D", +"V c #CFA649", +"C c #CEAA4C", +"Z c #D1A84B", +"A c #D4AC49", +"S c #D5AC49", +"D c #D7AD48", +"F c #D5AF4D", +"G c #C0A451", +"H c #CAA550", +"J c #CBAA58", +"K c #CAAD5D", +"L c #D9AD53", +"P c #DFB757", +"I c #D5B058", +"U c #D7B75D", +"Y c #DCB75A", +"T c #DFBB5E", +"R c #E1B957", +"E c #E3B759", +"W c #E6BC5B", +"Q c #E7BE5D", +"! c #E5BC5E", +"~ c #CAAE62", +"^ c #C8AF6A", +"/ c #CDB565", +"( c #CDB46D", +") c #D7B360", +"_ c #D5B862", +"` c #D1B66C", +"' c #D8BB68", +"] c #DBBA6E", +"[ c #DFBE6D", +"{ c #D0B872", +"} c #D2B974", +"| c #DABF74", +" . c #D8BF77", +".. c #D6BD79", +"X. c #D5BE7B", +"o. c #D7BF7A", +"O. c #E8BC61", +"+. c #DDC075", +"@. c #DCC279", +"#. c #DCC47E", +"$. c #EDC661", +"%. c #EEC562", +"&. c #E0C16B", +"*. c #E0C36B", +"=. c #E3C26A", +"-. c #E7C26A", +";. c #E3C569", +":. c #E3C26C", +">. c #E4C16C", +",. c #EEC969", +"<. c #F4C664", +"1. c #F0CA68", +"2. c #F7CA68", +"3. c #F6CD69", +"4. c #F7CD69", +"5. c #F7CF68", +"6. c #E6C374", +"7. c #E1C47A", +"8. c #E1C77B", +"9. c #E5C578", +"0. c #E4C579", +"q. c #E4C67A", +"w. c #E5C67C", +"e. c #E8C57B", +"r. c #E8CB7B", +"t. c #EDCC78", +"y. c #EBCB7C", +"u. c #F0CF73", +"i. c #F6CF74", +"p. c #F4D173", +"a. c #F7D072", +"s. c #F5D376", +"d. c #FAD071", +"f. c #FBD470", +"g. c #FAD572", +"h. c #FDD671", +"j. c #FDD773", +"k. c #F3DB76", +"l. c #F8D578", +"z. c #FBDB79", +"x. c #FFE57E", +"c. c #DEC681", +"v. c #DFC782", +"b. c #E0C682", +"n. c #E1C984", +"m. c #E2C985", +"M. c #E3CB87", +"N. c #E9C980", +"B. c #EBCC82", +"V. c #E3CC88", +"C. c #E4CF8D", +"Z. c #EFD187", +"A. c #EFD488", +"S. c #EFD58D", +"D. c #F6D581", +"F. c #F1D687", +"G. c #F9D680", +"H. c #F7DA8B", +"J. c #F2DE93", +"K. c #FADF93", +"L. c #F3DB98", +"P. c #F0DB9B", +"I. c #FEE081", +"U. c #FEE18C", +"Y. c #FCE38F", +"T. c #F7E98E", +"R. c #FFE88C", +"E. c #F6E491", +"W. c #FBEA91", +"Q. c #FFE897", +"!. c #FFEE9A", +"~. c #FEE99D", +"^. c #FEEC9F", +"/. c #FEF092", +"(. c #FFF29A", +"). c #FBE7A9", +"_. c #F4EDA8", +"`. c #FAEBAA", +"'. c #FEEBAD", +"]. c #FEFABD", +"[. c None", +/* pixels */ +"[.[.[.[.[.3 f M m q [.[.[.[.[.[.", +"[.[.[.4 A 5.j.f.3.3.! 6 [.[.[.[.", +"[.[.9 %.h.%.D s n -.z.l.c % [.[.", +"[.1 1.g.S p i i C 6.7.W./.t [.[.", +"[._ U.Z e 7 7 y B | #.*.T.k.* [.", +", E.F.c.).).C.g K `.`.C.' x.r + ", +"d !.y.M.).j '.( ).L.h M I z.F + ", +"G (.t.M.'.'.C.j ).b.T Q O.d.R X ", +"k T.O.M.`.^ `.X.).V.;.! L i.E ", +"w I.6.c.'.).C.{ c.).).C.H d.v + ", +"[.,.u.~ } X.@.@.| .#.` V 2.5 ", +"[.x p.U J ] B.Z.9.) v a <.E o [.", +"[.[.Y s.-.[ 0.0.N.e.w.H.D.> + [.", +"[.[.2 &.).Y.A.S.L.`.]._.0 . [.[.", +"[.[.[.[.l r.Y.Q.~.J./ = $ [.[.[.", +"[.[.[.[.[.[.: 1 ; - & [.[.[.[.[." +}; diff --git a/xpm/bitcoin20.xpm b/xpm/bitcoin20.xpm new file mode 100644 index 00000000..2dd61a59 --- /dev/null +++ b/xpm/bitcoin20.xpm @@ -0,0 +1,226 @@ +/* XPM */ +static char * bitcoin20_xpm[] = { +/* columns rows colors chars-per-pixel */ +"20 20 200 2", +" c #7B5500", +". c #7B5900", +"X c #735508", +"o c #7B5908", +"O c #7B5D08", +"+ c #7B5910", +"@ c #7B6118", +"# c #845D08", +"$ c #846108", +"% c #8C6510", +"& c #8C6910", +"* c #8C6918", +"= c #946D10", +"- c #947118", +"; c #9C7518", +": c #A57918", +"> c #846929", +", c #846D29", +"< c #947121", +"1 c #8C7539", +"2 c #947939", +"3 c #8C7542", +"4 c #AD8221", +"5 c #B58E29", +"6 c #B58E31", +"7 c #B59231", +"8 c #BD9231", +"9 c #BD9631", +"0 c #C69A31", +"q c #C69A39", +"w c #C69E39", +"e c #CE9E39", +"r c #CEA239", +"t c #948652", +"y c #A58A4A", +"u c #BD9642", +"i c #BD9A42", +"p c #B5964A", +"a c #B59A4A", +"s c #BD9E4A", +"d c #A58E5A", +"f c #BD9A52", +"g c #BD9E52", +"h c #BDA252", +"j c #BDA25A", +"k c #BD9E63", +"l c #A59673", +"z c #AD9A73", +"x c #AD9E7B", +"c c #BDA263", +"v c #BDA26B", +"b c #BDA273", +"n c #BDA673", +"m c #B5A27B", +"M c #BDAA7B", +"N c #C69E42", +"B c #CE9E42", +"V c #C6A242", +"C c #CEA242", +"Z c #CEA642", +"A c #C6A24A", +"S c #C6A64A", +"D c #CEA64A", +"F c #CEAA4A", +"G c #D6A642", +"H c #DEAE4A", +"J c #DEB24A", +"K c #C6A252", +"L c #C6A652", +"P c #CEAA52", +"I c #CEAE52", +"U c #C6A65A", +"Y c #C6AA5A", +"T c #CEAA5A", +"R c #CEAE5A", +"E c #D6AE52", +"W c #DEAE52", +"Q c #D6AE5A", +"! c #D6B252", +"~ c #DEB252", +"^ c #DEB652", +"/ c #D6B65A", +"( c #DEB65A", +") c #DEBA5A", +"_ c #EFBE52", +"` c #E7BA5A", +"' c #E7BE5A", +"] c #EFBE5A", +"[ c #C6A663", +"{ c #C6AE63", +"} c #CEAE63", +"| c #D6AE63", +" . c #CEB26B", +".. c #CEB66B", +"X. c #DEB663", +"o. c #D6BE63", +"O. c #DEBA63", +"+. c #DEBE63", +"@. c #D6B66B", +"#. c #DEB66B", +"$. c #D6BA6B", +"%. c #D6BE6B", +"&. c #DEBA6B", +"*. c #DEBE6B", +"=. c #D6BA73", +"-. c #DEBE73", +";. c #EFBE63", +":. c #E7BE73", +">. c #DEC37B", +",. c #E7C363", +"<. c #EFC763", +"1. c #EFCF63", +"2. c #E7C36B", +"3. c #E7C76B", +"4. c #EFC36B", +"5. c #EFC76B", +"6. c #E7CB6B", +"7. c #EFCB6B", +"8. c #F7CB63", +"9. c #F7CB6B", +"0. c #F7CF6B", +"q. c #FFCB6B", +"w. c #F7D36B", +"e. c #FFD36B", +"r. c #E7C373", +"t. c #E7CB73", +"y. c #EFCF73", +"u. c #E7C37B", +"i. c #E7C77B", +"p. c #E7CB7B", +"a. c #EFCB7B", +"s. c #F7CF73", +"d. c #EFD373", +"f. c #EFD37B", +"g. c #F7D373", +"h. c #FFD373", +"j. c #FFD773", +"k. c #FFDB73", +"l. c #F7DB7B", +"z. c #FFDF7B", +"x. c #ADA284", +"c. c #BDAA84", +"v. c #BDAE84", +"b. c #B5A68C", +"n. c #B5AE9C", +"m. c #BDB6A5", +"M. c #C6BA9C", +"N. c #C6BAA5", +"B. c #C6BEA5", +"V. c #DEC784", +"C. c #E7CB84", +"Z. c #E7CF84", +"A. c #EFCF84", +"S. c #E7CF8C", +"D. c #EFCF8C", +"F. c #EFD384", +"G. c #E7D38C", +"H. c #EFD38C", +"J. c #EFD78C", +"K. c #F7D784", +"L. c #FFD784", +"P. c #F7DB84", +"I. c #F7DF84", +"U. c #FFDB84", +"Y. c #FFDF84", +"T. c #F7DB8C", +"R. c #EFD394", +"E. c #EFD794", +"W. c #EFDB94", +"Q. c #EFDB9C", +"!. c #F7DB9C", +"~. c #F7DF9C", +"^. c #FFE384", +"/. c #FFE784", +"(. c #FFE38C", +"). c #FFEB8C", +"_. c #EFE79C", +"`. c #FFE794", +"'. c #FFEB94", +"]. c #FFEF94", +"[. c #FFEB9C", +"{. c #FFEF9C", +"}. c #FFF394", +"|. c #FFF794", +" X c #C6C3B5", +".X c #CEC7BD", +"XX c #F7E3A5", +"oX c #FFE7A5", +"OX c #F7EBA5", +"+X c #FFEBA5", +"@X c #FFEFA5", +"#X c #FFE7AD", +"$X c #FFEBAD", +"%X c #FFEFAD", +"&X c #FFF3AD", +"*X c #FFF7B5", +"=X c #FFFBB5", +"-X c #FFFFBD", +";X c #CEC7C6", +":X c None", +/* pixels */ +":X:X:X:X:X:XM.v f i g k c..X:X:X:X:X:X:X", +":X:X:X:XM.u H 8.j.j.e.0.^ 7 d X:X:X:X:X", +":X:X:Xn Z 0.k.j.8._ ] 9.h.h.~ ; b.:X:X:X", +":X:Xn J j.j.' C 0 0 w E a.K.^.d.- x.:X:X", +":XN.F k.w.G w q 0 0 D 2.i.a.].|.6.$ m.:X", +":Xg (.U.C 9 8 8 8 q S *.H.f.y.].).9 , :X", +"B.o.{.p.-.>.>.>.R 7 N =.G.E.Z.&./.1.# n.", +"M I.[.| R.$X..~.#Xs V.$XG.h @.T l.z.; t ", +"c {.'.X.E.$Xj G.$XU #X$Xg ) ! ( 0.k.5 > ", +"j }.`.O.E.$XE.oXC.p $X$XA ,.' ;.5.j.9 o ", +"j ).I.Q E.$X .Q.#X .$X$XP 7.` W 4.j.8 X ", +"c z.Y.P R.$X[ S.$X{ XX$X$.) P D 5.h.4 @ ", +"v.<.Y.I R.$XW.oXXX} @.XX#XE.XXK 8.8.& 3 ", +".XC j.3.s a h Y .J.A.T Y h 6 e 8.H . x ", +":Xk <.g./ P #.i.F.A.r.X.E B r 9.q.: + :X", +":X.Xi s.g.+.O.r.i.u.i.u.:.r.L.L.N l :X", +":X:XN.V U.(.T.a.i.C.D.!.%X-X=X%.# 1 :X:X", +":X:X:XN.g _.+X`.[.+X@X&X*XOXh O 1 :X:X:X", +":X:X:X:X;Xb i +.f.P.K.t.L = o z :X:X:X:X", +":X:X:X:X:X:X.Xm y < % * 2 x :X:X:X:X:X:X" +}; diff --git a/xpm/bitcoin32.xpm b/xpm/bitcoin32.xpm new file mode 100644 index 00000000..25da102f --- /dev/null +++ b/xpm/bitcoin32.xpm @@ -0,0 +1,223 @@ +/* XPM */ +static char * bitcoin32_xpm[] = { +/* columns rows colors chars-per-pixel */ +"32 32 185 2", +" c #715103", +". c #725203", +"X c #725204", +"o c #735304", +"O c #745404", +"+ c #765504", +"@ c #775504", +"# c #785604", +"$ c #795704", +"% c #795705", +"& c #77580A", +"* c #77580B", +"= c #77580C", +"- c #785808", +"; c #785809", +": c #78590D", +"> c #795A0D", +", c #7F5E0D", +"< c #7C5C0E", +"1 c #815F0E", +"2 c #89650F", +"3 c #8C670F", +"4 c #8D680F", +"5 c #836110", +"6 c #866410", +"7 c #8F6A11", +"8 c #926C12", +"9 c #946F14", +"0 c #967015", +"q c #987217", +"w c #997318", +"e c #9A751C", +"r c #9B761E", +"t c #9C7720", +"y c #9D7924", +"u c #9D7B28", +"i c #9E7C2C", +"p c #A07F31", +"a c #AA852D", +"s c #A9852E", +"d c #AC872D", +"f c #AE892E", +"g c #AF8A2E", +"h c #B08B2F", +"j c #A18133", +"k c #A78533", +"l c #A28235", +"z c #A48435", +"x c #A68535", +"c c #A58536", +"v c #A68536", +"b c #A88530", +"n c #B18C30", +"m c #B18D33", +"M c #B08D37", +"N c #B08F3B", +"B c #B08F3D", +"V c #BC9537", +"C c #BE9736", +"Z c #BF9737", +"A c #B6933B", +"S c #B0903F", +"D c #B2913E", +"F c #B4923D", +"G c #B99438", +"H c #C09736", +"J c #C19835", +"K c #C29836", +"L c #C39937", +"P c #C39A37", +"I c #C49B38", +"U c #C59C3A", +"Y c #C69D3B", +"T c #C79E3C", +"R c #C89F3D", +"E c #CAA03F", +"W c #C6A247", +"Q c #CAA140", +"! c #C9A242", +"~ c #C8A245", +"^ c #C2A14B", +"/ c #C3A24D", +"( c #C3A24F", +") c #C2A351", +"_ c #C1A352", +"` c #C0A355", +"' c #C1A457", +"] c #C1A458", +"[ c #C1A55A", +"{ c #C2A65C", +"} c #C3A75D", +"| c #C3A85E", +" . c #C6AA5D", +".. c #C7AA5D", +"X. c #C5A95E", +"o. c #CFAE5A", +"O. c #C8AB5E", +"+. c #D4AF56", +"@. c #D1AE58", +"#. c #D6B056", +"$. c #D8B155", +"%. c #D9B256", +"&. c #DAB357", +"*. c #DBB457", +"=. c #DDB558", +"-. c #DFB759", +";. c #E2BB5B", +":. c #E3BC5D", +">. c #E4BD5E", +",. c #C9AC61", +"<. c #C9AD62", +"1. c #CAAE62", +"2. c #CCAF62", +"3. c #DDBC69", +"4. c #DFBD68", +"5. c #DCBB6E", +"6. c #DCBC6C", +"7. c #DBBC70", +"8. c #DABD72", +"9. c #DBBE74", +"0. c #E4BD61", +"q. c #E4BE60", +"w. c #E3BE65", +"e. c #E2BF66", +"r. c #DDC177", +"t. c #DFC279", +"y. c #EFCB6F", +"u. c #F1CA6B", +"i. c #F4CB6A", +"p. c #F5CC6A", +"a. c #F7CD6B", +"s. c #F0CA6D", +"d. c #F7CF6E", +"f. c #EFCD72", +"g. c #EFCE77", +"h. c #E0C47B", +"j. c #E3C77E", +"k. c #E4C87F", +"l. c #F7D070", +"z. c #F8D171", +"x. c #F8D272", +"c. c #FAD473", +"v. c #FBD676", +"b. c #FCD574", +"n. c #FCD674", +"m. c #FCD777", +"M. c #F0D17E", +"N. c #FCD778", +"B. c #FCDA7A", +"V. c #FDDC7C", +"C. c #FDDE7E", +"Z. c #E6CA80", +"A. c #E8CC83", +"S. c #EACD84", +"D. c #ECD086", +"F. c #EFD286", +"G. c #EED287", +"H. c #F0D283", +"J. c #FDDF80", +"K. c #F6DF91", +"L. c #F5DE92", +"P. c #F4DE95", +"I. c #F4DF98", +"U. c #FDE081", +"Y. c #FCE184", +"T. c #FBE188", +"R. c #FAE18B", +"E. c #F8E08D", +"W. c #F5E19B", +"Q. c #F5E29C", +"!. c #F7E49D", +"~. c #F9E69B", +"^. c #FBE89B", +"/. c #FDEB9B", +"(. c #FDEC9B", +"). c #FEEE9B", +"_. c #FEEF9C", +"`. c #FEEEA3", +"'. c #FDEBA9", +"]. c #FDEBAC", +"[. c #FDEBAD", +"{. c #FDECAD", +"}. c #FDF0B0", +"|. c #FDF2B1", +" X c None", +/* pixels */ +" X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X", +" X X X X X X X X X X X Xc F z z F z j X X X X X X X X X X X X X", +" X X X X X X X X Xz b V ~ %.;.u.e.-.! f e i X X X X X X X X X X", +" X X X X X X Xz d ! >.a.l.B.v.c.a.a.a.a.>.R w 6 X X X X X X X X", +" X X X X X Xb L q.a.n.c.n.n.c.a.i.i.a.a.z.z.>.m 3 5 X X X X X X", +" X X X X Xs +.a.c.n.B.c.>.#.E E ! +.>.n.J.v.z.z.~ 8 & X X X X X", +" X X X Xs %.c.n.B.l.*.E Y L L L L W +.r.Z.H.Y.Y.R.+.7 ; X X X X", +" X X Xk %.n.J.n.>.! Y Y I L L L T *.w.h.7.5.K.).).).W 6 = X X X", +" X Xj T V.B.n.;.L L L I L L L Y ~ *.s.Z.G.D.B._._._.Y.h % > X X", +" X Xm H.R.Y.0.L H H L L C V C Y ! ..8.G.E.H.M.g._._.Y.q.8 o X X", +" Xj o.)./.K.@.^ A F M N M s n C W ) { <.9.9.2.X.#./.Y.C.m % > X", +" XM D.)._.t.,.[.[.[.[.{.[.W.A G C O.I.[.[.[.[.{.5.y.Y.B.*.2 o X", +" XA /.).~.%.{ [.[.[.| D.[.[.I.b ,.[.[.[.j.) ' D.5.@.Y.V.a.w . X", +"i O.).).E.3.X.[.[.[.S X.[.[.'.i Q.[.{.D.z X.~ A ) %.B.V.n.s + = ", +"z 6._._.R.w.<.[.[.[.S <.[.[.I.F [.[.[.X.) 0.>.;.>.>.l.B.n.C % > ", +"z e._._.H.e.,.[.[.[.j.W.[.D.S { [.[.].' -.>.>.>.e.0.s.N.n.Y % & ", +"v s.)._.M.3.O.[.].].W.'.[.].{ { [.[.[.' 3.e.;.;.%.-.s.N.n.R # . ", +"z ;.T.E.g.-.{ [.].].S <.[.{.].` [.[.[.<.e.f.e.*.$.$.s.V.n.L $ O ", +"j o.Y.Y.g.+.] {.].]._ | [.[.[.F ].[.[.A.) y.e.@.W @.s.N.n.h $ = ", +" XZ J.Y.N.@.' [.[.].F 7.[.[.].z h.[.[.].7.^ ^ 5.2.U z.z.a.r X = ", +" Xs n.V.Y.+.` [.[.].].].{.{.9.2...A.{.[.[.[.].[.2.! a.u.;.3 = ", +" XF *.m.B.s.^ X.,.{ ,.<.1.] 2.G.D.O.` 2.9.7.2.^ d ;.u.a.K $ X X", +" X Xm z.c.v.o.^ ^ ^ { <.7.Z.K.K.H.Z.6.o.^ A f h E a.i.0.w X = X", +" X Xz #.l.z.f.X.O.<.5.t.Z.D.D.A.j.7.*.@.^ ! Y ! i.a.a.Y , X X X", +" X X Xd u.l.z.y.o.o.3.r.j.Z.h.r.9.5.%.%.#.+.#.c.B.z.-.8 . & X X", +" X X Xj m f.c.v.l.s.3.4.h.t.r.k.D.G.H.D.A.D./.!.E.M.y + X X X X", +" X X X Xj Z v.J.T.R.E.Z.7.t.Z.A.S.H.D.Q.|.|.|.|.!.b % . X X X X", +" X X X X Xj A D.|.`.~.~.!.E.I.I.Q.{.|.|.|.|.|.D.u % X < X X X X", +" X X X X X Xj z 9.{.`.~.~.^././.`.`.`.}.|.Q.] 9 $ X X X X X X X", +" X X X X X X X Xj s X.k.Y.R.~.~./.~.K.h.) e , . = X X X X X X X", +" X X X X X X X X X X9 w t n A C A s r 3 $ X > X X X X X X X X X", +" X X X X X X X X X X X X X5 2 1 $ ; 5 5 5 X X X X X X X X X X X" +}; diff --git a/xpm/bitcoin48.xpm b/xpm/bitcoin48.xpm new file mode 100644 index 00000000..788e855d --- /dev/null +++ b/xpm/bitcoin48.xpm @@ -0,0 +1,278 @@ +/* XPM */ +static char * bitcoin48_xpm[] = { +/* columns rows colors chars-per-pixel */ +"48 48 224 2", +" c #715103", +". c #735203", +"X c #735204", +"o c #745405", +"O c #755506", +"+ c #775606", +"@ c #785707", +"# c #7A5806", +"$ c #7C5905", +"% c #7D5A05", +"& c #7E5B05", +"* c #7F5C07", +"= c #7E5C0A", +"- c #7B5C11", +"; c #7C5D13", +": c #7D5E15", +"> c #805C05", +", c #805C06", +"< c #8B6813", +"1 c #8D6912", +"2 c #8F6A12", +"3 c #896816", +"4 c #806219", +"5 c #82631A", +"6 c #876719", +"7 c #84651B", +"8 c #906C13", +"9 c #916D15", +"0 c #936F18", +"q c #94701B", +"w c #95721B", +"e c #98731A", +"r c #99741B", +"t c #99751E", +"y c #9A761F", +"u c #9B7822", +"i c #9F7A21", +"p c #9D7922", +"a c #A27C20", +"s c #A47E21", +"d c #A67F22", +"f c #9E8038", +"g c #9E803B", +"h c #9E813C", +"j c #9F833D", +"k c #A68023", +"l c #A78124", +"z c #A98326", +"x c #AA8427", +"c c #A7832B", +"v c #A7832C", +"b c #A6832D", +"n c #A98429", +"m c #A8842B", +"M c #AD892F", +"N c #AE8A2F", +"B c #AF8A2F", +"V c #B28B2E", +"C c #B48C2E", +"Z c #B68E2F", +"A c #B78E2F", +"S c #A38232", +"D c #A08136", +"F c #AC8931", +"G c #AB8934", +"H c #AA8836", +"J c #AA8937", +"K c #A1843F", +"L c #A2853F", +"P c #AB8A39", +"I c #AB8B3A", +"U c #AD8C3C", +"Y c #AE8E3E", +"T c #B89030", +"R c #BA9131", +"E c #BC9332", +"W c #BE9533", +"Q c #C09733", +"! c #C29834", +"~ c #C39934", +"^ c #C39935", +"/ c #C39A37", +"( c #C39A38", +") c #C49A38", +"_ c #C49B38", +"` c #C59C3A", +"' c #C69E3C", +"] c #C69E3E", +"[ c #C79F3F", +"{ c #A48640", +"} c #A58741", +"| c #A68842", +" . c #A78A43", +".. c #AA8D45", +"X. c #AD9046", +"o. c #B19040", +"O. c #B59443", +"+. c #B29447", +"@. c #B79745", +"#. c #B79847", +"$. c #B89846", +"%. c #B69748", +"&. c #C8A040", +"*. c #CAA241", +"=. c #CBA343", +"-. c #C9A346", +";. c #CBA445", +":. c #C7A44D", +">. c #C5A44F", +",. c #C9A448", +"<. c #C9A44A", +"1. c #C5A551", +"2. c #C4A554", +"3. c #CBAA57", +"4. c #CDAA57", +"5. c #C3A559", +"6. c #C4A65A", +"7. c #C7A85A", +"8. c #D0AB55", +"9. c #D2AD56", +"0. c #D4AE57", +"q. c #D4AF58", +"w. c #D5B05A", +"e. c #D5B15C", +"r. c #D5B25E", +"t. c #DFB65A", +"y. c #DEB75B", +"u. c #E1B759", +"i. c #E2B95B", +"p. c #E4BA5C", +"a. c #E4BC5F", +"s. c #D4B567", +"d. c #D8B764", +"f. c #D5B769", +"g. c #D4B76A", +"h. c #D4B86B", +"j. c #E5BD61", +"k. c #E6BE62", +"l. c #E6BF63", +"z. c #E0BF6F", +"x. c #E7C063", +"c. c #EAC263", +"v. c #EDC563", +"b. c #EBC364", +"n. c #EEC565", +"m. c #EEC767", +"M. c #E1C06F", +"N. c #EBC76D", +"B. c #EEC869", +"V. c #F7CD6A", +"C. c #F6CD6B", +"Z. c #F4CE6F", +"A. c #F8CD6A", +"S. c #F9CF6E", +"D. c #FAD16F", +"F. c #E1C071", +"G. c #E4C370", +"H. c #E1C174", +"J. c #E0C276", +"K. c #E1C377", +"L. c #E8C670", +"P. c #E2C479", +"I. c #E0C47A", +"U. c #E2C57C", +"Y. c #E3C77E", +"T. c #F3D073", +"R. c #FBD270", +"E. c #FCD572", +"W. c #FCD674", +"Q. c #FDD774", +"!. c #FED876", +"~. c #FED977", +"^. c #F2D278", +"/. c #F1D37B", +"(. c #FDDA78", +"). c #FDDB7A", +"_. c #FDDB7C", +"`. c #FDDD7D", +"'. c #FDDF7F", +"]. c #E4C880", +"[. c #E6CA83", +"{. c #E7CD86", +"}. c #E9CF89", +"|. c #EBD089", +" X c #EFD289", +".X c #F2D382", +"XX c #F0D387", +"oX c #FDDF80", +"OX c #FCDF84", +"+X c #FBDF89", +"@X c #F9DE8B", +"#X c #FBE08B", +"$X c #FBE28C", +"%X c #FCE48F", +"&X c #FDE592", +"*X c #FEE692", +"=X c #FEE693", +"-X c #FEE895", +";X c #FEEA96", +":X c #FEEC97", +">X c #FEEE98", +",X c #FEEE99", +"X>X:X:X%X' 0 O # sXsXsXsXsX", +"sXsXsXsXG n m.(.!.(.(.S.;.] [ _ ` _ ^ ^ ! Q ^ _ ` 9.k.M.U.K.U.H.U./.>X>X>X>X>X#XV < . - sXsXsXsX", +"sXsXsXh n w._._.(.W.S.;._ _ _ / _ _ _ _ Q Q / ` [ 9.d.G.H.|.XX@X#X#X).:X>X>X>X>Xm.a > sXsXsXsX", +"sXsXsXG @.#X+XOX_.R.8._ _ _ _ ! ^ _ ) ! W W _ ] [ <.r.h.{.@X6X&X#X#X).).>X>X>X;X#X' 8 + + sXsXsX", +"sXsXL M H.;X=X&X#Xk.] E R T Z T A A A A R E _ ` &.<.4.F.Y.[.U.H.z.h.z.L.^.>X>X;X'.m.k > : sXsX", +"sXsXP @.,X,X1X;X+X3.%.o.G H F m m b b a s z E _ [ 1.2.2.2.7.h.h.h.h.2.$.1.^.;X#X'.).` 2 . + sXsX", +"sXsXF f.X>X/.L.d.o.6X0X0XwXwX0X0XwXwX0X9XI.u I.0XwXqXqX3XG r.B.c.i.u.u.u.u.y.p.`.!.E.E.x * X ", +" .n ^.&X;X;X.Xx.d.Y 6X0XwX0X0X5.u 3X0XwXwX9X5.h.wX0XqXqX6XP d.T.N.c.i.u.y.0.0.k.(.!.W.E.k * X ", +"L x C.'.*X&X^.j.r.P 7XqX0XwX0X6.0 I.wX0XwX0XI.@.0XqXqXqX0X+.q.T.T.N.a.y.9.8.9.l._.!.!.D.s $ - ", +"sXx k.#X#X$X.Xy.8.G 3XqXwX0XwX7.u I.0XwX0XwX}.u qXqXqXqX0Xh.$.G.T.N.p.9.-.-.8.N._.!.E.b.i # X : ", +"sXc 9.'.'.#X'.y.4.H 7XqX0XwX0X6.q ].9XwX0XwXU.u {.0XwX0XwX8X%.#.w.4.$.#.f.W -.Z.W.R.R.y.r O = ", +"sXN _ '.'.'.#Xx.3.P 7XqXwX0XwXh.+.8XwX0X0X0X7.X.#.8X0XwX0XwX6Xh.2.5.U.6X{.F *.R.R.R.S.*.1 ; ", +"sXX.x E.`.).`.T.1.G 8XwX0XwXwX0XwX0X0XwX0X}.%.h.7.5.3X0X0XwX0XwXwX0X0X0X[.z 9.A.V.V.V.n * X sX", +"sXg x y.~.).).#X4.o.3X3X3X3X3X3X3X7X3X Xh.%.h. X|.3.#.h.3X0X0X0X0X0X8X}.2.V c.A.C.V.i.e # X X sX", +"sXsXn W E.E.).`.L.$.P g u p f D d . .+.6.U. X@X@X}.f.1.U Y %.%.+.H y e l &.A.A.V.V.' 8 X ; sX", +"sXsX..x k.T.T.T.W.9.#.#.#.#.1.7.4.s.h.z.}.}.@X@X X X].z.d.3.:.@.V V V C E c.A.A.A.c.s > sXsX", +"sXsXsXb W T.T.T./.T.:.1.3.4.r.h.z.].[. X X X X X[.Y.H.z.z.d.0.;.` / Q Q u.A.A.A.D.*.9 O X # sXsX", +"sXsXsX| l y.T.T.T./.N.<.1.4.r.f.z.z.].[.{.{.[.].H.z.s.r.8.8.-.-.*.] ] u.D.E.D.D.k.a > sXsXsX", +"sXsXsXsXn F B.T.T.T.^.N.1.4.w.s.z.z.U.[.{.].J.J.H.F.h.r.w.r.w.0.8.8.k.W.W.R.D.T.Z 1 O @ sXsXsX", +"sXsXsXsXf x W T.T.T.^./.T.r.r.d.z.L.U.].U.z.z.P.Y.[.[.U.P.F.G.h.h..X@XOXOX.X.X<.9 # sXsXsXsX", +"sXsXsXsXsXL x [ W.W.W.)._.OXN.d.z.z.I.U.I.z.U. X.X X XXXXX X XXXwXyX,X1X1X=Xs.p > X ; sXsXsXsX", +"sXsXsXsXsXsX| x :._.oX_.OX$X=X+X/.G.h.z.I.].}.}.|.|. X X X3XtXpXpXpXpXpXaXP.a > X - sXsXsXsXsX", +"sXsXsXsXsXsXsX{ c <.+X@X=X;X;X&X=X&XXX[.U.P.U.].[.}.XX3XyXpXpXpXpXpXpXpXh.a > O sXsXsXsXsXsX", +"sXsXsXsXsXsXsXsXL c @.+XpXpXuX1X=X=X;X1X1X1X1XrXwXwXyXyXyXuXpXaXpXuX3X2.u > . @ sXsXsXsXsXsXsX", +"sXsXsXsXsXsXsXsXsXj n Y I.aXaX0X=X=X&X;X=X1X1X1X Date: Sun, 8 Nov 2009 04:24:52 +0000 Subject: [PATCH 026/133] Get local host IP on Linux, Shutdown() a little cleaner -- linux-0.1.6-test1 waypoint git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@37 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 2 ++ main.cpp | 4 ++-- makefile.unix | 23 +++++++++++--------- net.cpp | 58 +++++++++++++++++++++++++++++++++++++++++++-------- ui.cpp | 13 +++++++++++- 5 files changed, 78 insertions(+), 22 deletions(-) diff --git a/headers.h b/headers.h index 34eafb26..76881bb9 100644 --- a/headers.h +++ b/headers.h @@ -72,6 +72,8 @@ #include #include #include +#include +#include #include #include #include diff --git a/main.cpp b/main.cpp index fe213c09..b957027f 100644 --- a/main.cpp +++ b/main.cpp @@ -1215,7 +1215,7 @@ bool CBlock::AcceptBlock() return error("AcceptBlock() : block's timestamp is too early"); // Check that all transactions are finalized (starting around Dec 2009) - if (nBestHeight > 31000) // 25620 + 5320 + if (nBestHeight > 31000) foreach(const CTransaction& tx, vtx) if (!tx.IsFinal(nTime)) return error("AcceptBlock() : contains a non-final transaction"); @@ -1802,7 +1802,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) { if (fShutdown) return true; - addr.nTime = GetAdjustedTime(); + addr.nTime = GetAdjustedTime() - 2 * 60 * 60; if (pfrom->fGetAddr) addr.nTime -= 5 * 24 * 60 * 60; AddAddress(addrdb, addr, false); diff --git a/makefile.unix b/makefile.unix index c62efc00..24aa8bfe 100644 --- a/makefile.unix +++ b/makefile.unix @@ -15,21 +15,24 @@ endif -INCLUDEPATHS=-I"/usr/include" \ - -I"/usr/local/boost_1_40_0" \ - -I"/usr/local/db-4.7.25.NC/build_unix" \ - -I"/usr/local/include/wx-2.8" \ - -I"/usr/local/lib/wx/include/gtk2-ansi-debug-static-2.8" - -LIBPATHS=-L"/usr/lib" \ - -L"/usr/local/lib" \ - -L"/usr/local/db-4.7.25.NC/build_unix" +INCLUDEPATHS= \ + -I"/usr/include" \ + -I"/usr/local/boost_1_40_0" \ + -I"/usr/local/db-4.7.25.NC/build_unix" \ + -I"/usr/local/include/wx-2.8" \ + -I"/usr/local/lib/wx/include/gtk2-ansi-debug-static-2.8" + +LIBPATHS= \ + -L"/usr/lib" \ + -L"/usr/local/lib" \ + -L"/usr/local/db-4.7.25.NC/build_unix" LIBS= \ -Wl,-Bstatic -l boost_thread -l boost_system -l boost_filesystem -Wl,-Bdynamic \ -Wl,-Bstatic -l db_cxx -l wx_gtk2$(D)-2.8 -Wl,-Bdynamic \ -l crypto \ -l gtk-x11-2.0 -l gthread-2.0 -l SM + WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h @@ -76,7 +79,7 @@ obj/irc.o: irc.cpp $(HEADERS) OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/net.cpp b/net.cpp index d70a62a2..0569604b 100644 --- a/net.cpp +++ b/net.cpp @@ -842,20 +842,20 @@ void ThreadOpenConnections2(void* parg) if (!addr.IsIPv4() || !addr.IsValid() || setConnected.count(addr.ip)) continue; + // Randomize the order in a deterministic way, putting the standard port first + int64 nRandomizer = (uint64)(addr.nLastFailed * 9567851 + addr.ip * 7789) % (1 * 60 * 60); + if (addr.port != DEFAULT_PORT) + nRandomizer += 1 * 60 * 60; + // Limit retry frequency - if (GetAdjustedTime() < addr.nLastFailed + nDelay) + if (GetAdjustedTime() < addr.nLastFailed + nDelay + nRandomizer) continue; // Try again only after all addresses had a first attempt - int64 nTime = addr.nTime; + int64 nTime = addr.nTime - nRandomizer; if (addr.nLastFailed > addr.nTime) nTime -= 365 * 24 * 60 * 60; - // Randomize the order a little, putting the standard port first - nTime += GetRand(1 * 60 * 60); - if (addr.port != DEFAULT_PORT) - nTime -= 1 * 60 * 60; - if (nTime > nBestTime) { nBestTime = nTime; @@ -1069,6 +1069,7 @@ bool StartNode(string& strError) if (pnodeLocalHost == NULL) pnodeLocalHost = new CNode(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); +#ifdef __WXMSW__ // Get local host ip char pszHostName[255]; if (gethostname(pszHostName, sizeof(pszHostName)) == SOCKET_ERROR) @@ -1090,10 +1091,49 @@ bool StartNode(string& strError) printf("host ip %d: %s\n", i, CAddress(*(unsigned int*)phostent->h_addr_list[i]).ToStringIP().c_str()); for (int i = 0; phostent->h_addr_list[i] != NULL; i++) { - addrLocalHost = CAddress(*(unsigned int*)phostent->h_addr_list[i], DEFAULT_PORT, nLocalServices); - if (addrLocalHost.IsValid() && addrLocalHost.GetByte(3) != 127) + CAddress addr(*(unsigned int*)phostent->h_addr_list[i], DEFAULT_PORT, nLocalServices); + if (addr.IsValid() && addr.GetByte(3) != 127) + { + addrLocalHost = addr; break; + } + } +#else + // Get local host ip + struct ifaddrs* myaddrs; + if (getifaddrs(&myaddrs) == 0) + { + for (struct ifaddrs* ifa = myaddrs; ifa != NULL; ifa = ifa->ifa_next) + { + if (ifa->ifa_addr == NULL) continue; + if ((ifa->ifa_flags & IFF_UP) == 0) continue; + if (strcmp(ifa->ifa_name, "lo") == 0) continue; + if (strcmp(ifa->ifa_name, "lo0") == 0) continue; + char pszIP[100]; + if (ifa->ifa_addr->sa_family == AF_INET) + { + struct sockaddr_in* s4 = (struct sockaddr_in*)(ifa->ifa_addr); + if (inet_ntop(ifa->ifa_addr->sa_family, (void*)&(s4->sin_addr), pszIP, sizeof(pszIP)) != NULL) + printf("ipv4 %s: %s\n", ifa->ifa_name, pszIP); + + // Take the first IP that isn't loopback 127.x.x.x + CAddress addr(*(unsigned int*)&s4->sin_addr, DEFAULT_PORT, nLocalServices); + if (addr.IsValid() && addr.GetByte(3) != 127) + { + addrLocalHost = addr; + break; + } + } + else if (ifa->ifa_addr->sa_family == AF_INET6) + { + struct sockaddr_in6* s6 = (struct sockaddr_in6*)(ifa->ifa_addr); + if (inet_ntop(ifa->ifa_addr->sa_family, (void*)&(s6->sin6_addr), pszIP, sizeof(pszIP)) != NULL) + printf("ipv6 %s: %s\n", ifa->ifa_name, pszIP); + } + } + freeifaddrs(myaddrs); } +#endif printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); // Get our external IP address for incoming connections diff --git a/ui.cpp b/ui.cpp index a1873d5c..e12970b8 100644 --- a/ui.cpp +++ b/ui.cpp @@ -387,17 +387,28 @@ CMainFrame::~CMainFrame() void Shutdown(void* parg) { static CCriticalSection cs_Shutdown; + static bool fTaken; + bool fFirstThread; CRITICAL_BLOCK(cs_Shutdown) + { + fFirstThread = !fTaken; + fTaken = true; + } + if (fFirstThread) { fShutdown = true; nTransactionsUpdated++; DBFlush(false); StopNode(); DBFlush(true); - printf("Bitcoin exiting\n\n"); exit(0); } + else + { + loop + Sleep(100000); + } } void CMainFrame::OnClose(wxCloseEvent& event) From 70e79525c97c33a5fca8674d47d86cc2b7ae197a Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 13 Nov 2009 01:23:08 +0000 Subject: [PATCH 027/133] monitor ThreadSocketHandler and terminate and restart if hung, convert _beginthread to CreateThread wrapper, disconnect inactive connections, ping, break up long messages to speed up initial download, better priorities for initiating connections, track how many nodes have requested our blocks and transactions, status #/offline and warning message on unsent blocks, minimize on close as separate option -- linux-test5 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@38 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-msw.txt | 2 +- build-unix.txt | 1 + db.cpp | 9 +- headers.h | 1 - irc.cpp | 5 +- main.cpp | 144 +++++++++++++++++---- main.h | 25 ++-- net.cpp | 342 +++++++++++++++++++++++++++++++------------------ net.h | 25 ++-- ui.cpp | 137 +++++++++++++------- ui.h | 2 +- uibase.cpp | 16 +-- uibase.h | 92 ++++++------- uiproject.fbp | 121 ++++++++--------- util.cpp | 48 ++++--- util.h | 105 ++++++++++++--- 16 files changed, 684 insertions(+), 391 deletions(-) diff --git a/build-msw.txt b/build-msw.txt index eb6348ba..a159a71a 100644 --- a/build-msw.txt +++ b/build-msw.txt @@ -8,7 +8,7 @@ the OpenSSL Toolkit (http://www.openssl.org/). This product includes cryptographic software written by Eric Young (eay@cryptsoft.com). - WINDOWS BUILD NOTES +WINDOWS BUILD NOTES Compilers Supported diff --git a/build-unix.txt b/build-unix.txt index 6b29e1e5..ceb61ad5 100644 --- a/build-unix.txt +++ b/build-unix.txt @@ -13,6 +13,7 @@ UNIX BUILD NOTES Dependencies ------------ +Install the dev files for the shared libraries: apt-get install build-essential apt-get install libgtk2.0-dev apt-get install libssl-dev diff --git a/db.cpp b/db.cpp index 61d60253..a9c42880 100644 --- a/db.cpp +++ b/db.cpp @@ -505,6 +505,13 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) { vchDefaultKeyRet.clear(); + // Modify defaults +#ifndef __WXMSW__ + // Reports that tray icon can disappear on gnome, leaving no way to access the program + fMinimizeToTray = false; + fMinimizeOnClose = false; +#endif + //// todo: shouldn't we catch exceptions and try to recover and continue? CRITICAL_BLOCK(cs_mapKeys) CRITICAL_BLOCK(cs_mapWallet) @@ -638,7 +645,7 @@ bool LoadWallet(bool& fFirstRunRet) CWalletDB().WriteDefaultKey(keyUser.GetPubKey()); } - _beginthread(ThreadFlushWalletDB, 0, NULL); + CreateThread(ThreadFlushWalletDB, NULL); return true; } diff --git a/headers.h b/headers.h index 76881bb9..22bb830f 100644 --- a/headers.h +++ b/headers.h @@ -75,7 +75,6 @@ #include #include #include -#include #include #endif diff --git a/irc.cpp b/irc.cpp index 3b232cae..8432c6d1 100644 --- a/irc.cpp +++ b/irc.cpp @@ -54,7 +54,7 @@ static bool Send(SOCKET hSocket, const char* pszSend) const char* pszEnd = psz + strlen(psz); while (psz < pszEnd) { - int ret = send(hSocket, psz, pszEnd - psz, 0); + int ret = send(hSocket, psz, pszEnd - psz, MSG_NOSIGNAL); if (ret < 0) return false; psz += ret; @@ -156,7 +156,7 @@ bool Wait(int nSeconds) void ThreadIRCSeed(void* parg) { - SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL); + SetThreadPriority(THREAD_PRIORITY_NORMAL); int nErrorWait = 10; int nRetryWait = 10; @@ -256,6 +256,7 @@ void ThreadIRCSeed(void* parg) CAddress addr; if (DecodeAddress(pszName, addr)) { + addr.nTime = GetAdjustedTime() - 51 * 60; CAddrDB addrdb; if (AddAddress(addrdb, addr)) printf("IRC got new address\n"); diff --git a/main.cpp b/main.cpp index b957027f..13a9f9b7 100644 --- a/main.cpp +++ b/main.cpp @@ -42,6 +42,9 @@ map > mapPubKeys; CCriticalSection cs_mapKeys; CKey keyUser; +map mapRequestCount; +CCriticalSection cs_mapRequestCount; + // Settings int fGenerateBitcoins = false; int64 nTransactionFee = 0; @@ -274,7 +277,44 @@ int64 CWalletTx::GetTxTime() const return nTimeReceived; } +int CWalletTx::GetRequestCount() const +{ + // Returns -1 if it wasn't being tracked + int nRequests = -1; + CRITICAL_BLOCK(cs_mapRequestCount) + { + if (IsCoinBase()) + { + // Generated block + if (hashBlock != 0) + { + map::iterator mi = mapRequestCount.find(hashBlock); + if (mi != mapRequestCount.end()) + nRequests = (*mi).second; + } + } + else + { + // Did anyone request this transaction? + map::iterator mi = mapRequestCount.find(GetHash()); + if (mi != mapRequestCount.end()) + { + nRequests = (*mi).second; + // How about the block it's in? + if (nRequests == 0 && hashBlock != 0) + { + map::iterator mi = mapRequestCount.find(hashBlock); + if (mi != mapRequestCount.end()) + nRequests = (*mi).second; + else + nRequests = 1; // If it's in someone else's block it must have got out + } + } + } + } + return nRequests; +} @@ -295,7 +335,7 @@ int CMerkleTx::SetMerkleBranch(const CBlock* pblock) CTxIndex txindex; if (!CTxDB("r").ReadTxIndex(GetHash(), txindex)) return 0; - if (!blockTmp.ReadFromDisk(txindex.pos.nFile, txindex.pos.nBlockPos, true)) + if (!blockTmp.ReadFromDisk(txindex.pos.nFile, txindex.pos.nBlockPos)) return 0; pblock = &blockTmp; } @@ -1003,7 +1043,7 @@ bool Reorganize(CTxDB& txdb, CBlockIndex* pindexNew) foreach(CBlockIndex* pindex, vDisconnect) { CBlock block; - if (!block.ReadFromDisk(pindex->nFile, pindex->nBlockPos, true)) + if (!block.ReadFromDisk(pindex->nFile, pindex->nBlockPos)) return error("Reorganize() : ReadFromDisk for disconnect failed"); if (!block.DisconnectBlock(txdb, pindex)) return error("Reorganize() : DisconnectBlock failed"); @@ -1020,7 +1060,7 @@ bool Reorganize(CTxDB& txdb, CBlockIndex* pindexNew) { CBlockIndex* pindex = vConnect[i]; CBlock block; - if (!block.ReadFromDisk(pindex->nFile, pindex->nBlockPos, true)) + if (!block.ReadFromDisk(pindex->nFile, pindex->nBlockPos)) return error("Reorganize() : ReadFromDisk for connect failed"); if (!block.ConnectBlock(txdb, pindex)) { @@ -1380,7 +1420,7 @@ bool CheckDiskSpace(int64 nAdditionalBytes) { fShutdown = true; ThreadSafeMessageBox("Warning: Your disk space is low ", "Bitcoin", wxOK | wxICON_EXCLAMATION); - _beginthread(Shutdown, 0, NULL); + CreateThread(Shutdown, NULL); return false; } return true; @@ -1547,7 +1587,7 @@ void PrintBlockTree() // print item CBlock block; - block.ReadFromDisk(pindex, true); + block.ReadFromDisk(pindex); printf("%d (%u,%u) %s %s tx %d", pindex->nHeight, pindex->nFile, @@ -1623,7 +1663,8 @@ bool ProcessMessages(CNode* pfrom) CDataStream& vRecv = pfrom->vRecv; if (vRecv.empty()) return true; - //printf("ProcessMessages(%d bytes)\n", vRecv.size()); + //if (fDebug) + // printf("ProcessMessages(%d bytes)\n", vRecv.size()); // // Message format @@ -1666,7 +1707,8 @@ bool ProcessMessages(CNode* pfrom) { // Rewind and wait for rest of message ///// need a mechanism to give up waiting for overlong message size error - //printf("message-break\n"); + //if (fDebug) + // printf("message-break\n"); vRecv.insert(vRecv.begin(), BEGIN(hdr), END(hdr)); Sleep(100); break; @@ -1718,6 +1760,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) { static map > mapReuseKey; RandAddSeedPerfmon(); + if (fDebug) + printf("%s ", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); printf("received: %s (%d bytes)\n", strCommand.c_str(), vRecv.size()); if (mapArgs.count("-dropmessagestest") && GetRand(atoi(mapArgs["-dropmessagestest"])) == 0) { @@ -1739,18 +1783,19 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) CAddress addrMe; CAddress addrFrom; uint64 nNonce = 1; + string strSubVer; vRecv >> pfrom->nVersion >> pfrom->nServices >> nTime >> addrMe; if (pfrom->nVersion >= 106 && !vRecv.empty()) vRecv >> addrFrom >> nNonce; + if (pfrom->nVersion >= 106 && !vRecv.empty()) + vRecv >> strSubVer; if (pfrom->nVersion == 0) return false; // Disconnect if we connected to ourself - if (nNonce == nLocalHostNonce) + if (nNonce == nLocalHostNonce && nNonce > 1) { pfrom->fDisconnect = true; - pfrom->vRecv.clear(); - pfrom->vSend.clear(); return true; } @@ -1776,10 +1821,6 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) pfrom->fSuccessfullyConnected = true; - // Update the last seen time - if (pfrom->fNetworkNode) - AddressCurrentlyConnected(pfrom->addr); - printf("version message: version %d\n", pfrom->nVersion); } @@ -1824,10 +1865,6 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vector vInv; vRecv >> vInv; - // Update the last seen time for this node's address - if (pfrom->fNetworkNode) - AddressCurrentlyConnected(pfrom->addr); - CTxDB txdb("r"); foreach(const CInv& inv, vInv) { @@ -1842,6 +1879,14 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) pfrom->AskFor(inv); else if (inv.type == MSG_BLOCK && mapOrphanBlocks.count(inv.hash)) pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), GetOrphanRoot(mapOrphanBlocks[inv.hash])); + + // Track requests for our stuff + CRITICAL_BLOCK(cs_mapRequestCount) + { + map::iterator mi = mapRequestCount.find(inv.hash); + if (mi != mapRequestCount.end()) + (*mi).second++; + } } } @@ -1879,6 +1924,14 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) pfrom->PushMessage(inv.GetCommand(), (*mi).second); } } + + // Track requests for our stuff + CRITICAL_BLOCK(cs_mapRequestCount) + { + map::iterator mi = mapRequestCount.find(inv.hash); + if (mi != mapRequestCount.end()) + (*mi).second++; + } } } @@ -2086,11 +2139,23 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) } + else if (strCommand == "ping") + { + } + + else { // Ignore unknown commands for extensibility } + + // Update the last seen time for this node's address + if (pfrom->fNetworkNode) + if (strCommand == "version" || strCommand == "addr" || strCommand == "inv" || strCommand == "getdata" || strCommand == "ping") + AddressCurrentlyConnected(pfrom->addr); + + return true; } @@ -2129,6 +2194,10 @@ bool SendMessages(CNode* pto) } } + // Keep-alive ping + if (pto->nLastSend && GetTime() - pto->nLastSend > 12 * 60 && pto->vSend.empty()) + pto->PushMessage("ping"); + // // Message: addr @@ -2139,7 +2208,14 @@ bool SendMessages(CNode* pto) { // returns true if wasn't already contained in the set if (pto->setAddrKnown.insert(addr).second) + { vAddrToSend.push_back(addr); + if (vAddrToSend.size() >= 1000) + { + pto->PushMessage("addr", vAddrToSend); + vAddrToSend.clear(); + } + } } pto->vAddrToSend.clear(); if (!vAddrToSend.empty()) @@ -2157,7 +2233,14 @@ bool SendMessages(CNode* pto) { // returns true if wasn't already contained in the set if (pto->setInventoryKnown.insert(inv).second) + { vInventoryToSend.push_back(inv); + if (vInventoryToSend.size() >= 1000) + { + pto->PushMessage("inv", vInventoryToSend); + vInventoryToSend.clear(); + } + } } pto->vInventoryToSend.clear(); pto->setInventoryKnown2.clear(); @@ -2179,6 +2262,11 @@ bool SendMessages(CNode* pto) { printf("sending getdata: %s\n", inv.ToString().c_str()); vAskFor.push_back(inv); + if (vAskFor.size() >= 1000) + { + pto->PushMessage("getdata", vAskFor); + vAskFor.clear(); + } } pto->mapAskFor.erase(pto->mapAskFor.begin()); } @@ -2226,8 +2314,8 @@ void GenerateBitcoins(bool fGenerate) int nAddThreads = nProcessors - vnThreadsRunning[3]; printf("Starting %d BitcoinMiner threads\n", nAddThreads); for (int i = 0; i < nAddThreads; i++) - if (_beginthread(ThreadBitcoinMiner, 0, NULL) == -1) - printf("Error: _beginthread(ThreadBitcoinMiner) failed\n"); + if (!CreateThread(ThreadBitcoinMiner, NULL)) + printf("Error: CreateThread(ThreadBitcoinMiner) failed\n"); } } @@ -2304,7 +2392,7 @@ void BitcoinMiner() CBigNum bnExtraNonce = 0; while (fGenerateBitcoins) { - SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); + SetThreadPriority(THREAD_PRIORITY_LOWEST); Sleep(50); if (fShutdown) return; @@ -2440,7 +2528,7 @@ void BitcoinMiner() printf("proof-of-work found \n hash: %s \ntarget: %s\n", hash.GetHex().c_str(), hashTarget.GetHex().c_str()); pblock->print(); - SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL); + SetThreadPriority(THREAD_PRIORITY_NORMAL); CRITICAL_BLOCK(cs_main) { if (pindexPrev == pindexBest) @@ -2450,12 +2538,16 @@ void BitcoinMiner() return; key.MakeNewKey(); + // Track how many getdata requests this block gets + CRITICAL_BLOCK(cs_mapRequestCount) + mapRequestCount[pblock->GetHash()] = 0; + // Process this block the same as if we had received it from another node if (!ProcessBlock(NULL, pblock.release())) printf("ERROR in BitcoinMiner, ProcessBlock, block not accepted\n"); } } - SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST); + SetThreadPriority(THREAD_PRIORITY_LOWEST); Sleep(500); break; @@ -2534,7 +2626,7 @@ bool SelectCoins(int64 nTargetValue, set& setCoinsRet) setCoinsRet.clear(); // List of values less than target - int64 nLowestLarger = _I64_MAX; + int64 nLowestLarger = INT64_MAX; CWalletTx* pcoinLowestLarger = NULL; vector > vValue; int64 nTotalLower = 0; @@ -2777,6 +2869,10 @@ bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) return error("SendMoney() : Error finalizing transaction"); } + // Track how many getdata requests our transaction gets + CRITICAL_BLOCK(cs_mapRequestCount) + mapRequestCount[wtxNew.GetHash()] = 0; + printf("SendMoney: %s\n", wtxNew.GetHash().ToString().substr(0,6).c_str()); // Broadcast diff --git a/main.h b/main.h index 853fdfa5..8bb1e19e 100644 --- a/main.h +++ b/main.h @@ -34,6 +34,8 @@ extern int nBestHeight; extern uint256 hashBestChain; extern CBlockIndex* pindexBest; extern unsigned int nTransactionsUpdated; +extern map mapRequestCount; +extern CCriticalSection cs_mapRequestCount; // Settings extern int fGenerateBitcoins; @@ -647,6 +649,15 @@ public: nGetCreditCached = 0; } + IMPLEMENT_SERIALIZE + ( + nSerSize += SerReadWrite(s, *(CTransaction*)this, nType, nVersion, ser_action); + nVersion = this->nVersion; + READWRITE(hashBlock); + READWRITE(vMerkleBranch); + READWRITE(nIndex); + ) + int64 GetCredit(bool fUseCache=false) const { // Must wait until coinbase is safely deep enough in the chain before valuing it @@ -661,15 +672,6 @@ public: return nGetCreditCached; } - IMPLEMENT_SERIALIZE - ( - nSerSize += SerReadWrite(s, *(CTransaction*)this, nType, nVersion, ser_action); - nVersion = this->nVersion; - READWRITE(hashBlock); - READWRITE(vMerkleBranch); - READWRITE(nIndex); - ) - int SetMerkleBranch(const CBlock* pblock=NULL); int GetDepthInMainChain() const; @@ -749,6 +751,7 @@ public: int64 GetTxTime() const; + int GetRequestCount() const; void AddSupportingTransactions(CTxDB& txdb); @@ -978,7 +981,7 @@ public: return true; } - bool ReadFromDisk(unsigned int nFile, unsigned int nBlockPos, bool fReadTransactions) + bool ReadFromDisk(unsigned int nFile, unsigned int nBlockPos, bool fReadTransactions=true) { SetNull(); @@ -1027,7 +1030,7 @@ public: int64 GetBlockValue(int64 nFees) const; bool DisconnectBlock(CTxDB& txdb, CBlockIndex* pindex); bool ConnectBlock(CTxDB& txdb, CBlockIndex* pindex); - bool ReadFromDisk(const CBlockIndex* blockindex, bool fReadTransactions); + bool ReadFromDisk(const CBlockIndex* blockindex, bool fReadTransactions=true); bool AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos); bool CheckBlock() const; bool AcceptBlock(); diff --git a/net.cpp b/net.cpp index 0569604b..71295d5d 100644 --- a/net.cpp +++ b/net.cpp @@ -13,7 +13,6 @@ bool OpenNetworkConnection(const CAddress& addrConnect); - // // Global state variables // @@ -25,6 +24,7 @@ uint64 nLocalHostNonce = 0; bool fShutdown = false; array vnThreadsRunning; SOCKET hListenSocket = INVALID_SOCKET; +int64 nThreadSocketHandlerHeartbeat = INT64_MAX; vector vNodes; CCriticalSection cs_vNodes; @@ -65,7 +65,7 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) if (fProxy) { - printf("Proxy connecting %s\n", addrConnect.ToStringLog().c_str()); + printf("proxy connecting %s\n", addrConnect.ToStringLog().c_str()); char pszSocks4IP[] = "\4\1\0\0\0\0\0\0user"; memcpy(pszSocks4IP + 2, &addrConnect.port, 2); memcpy(pszSocks4IP + 4, &addrConnect.ip, 4); @@ -87,9 +87,11 @@ bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) if (pchRet[1] != 0x5a) { closesocket(hSocket); - return error("Proxy returned error %d", pchRet[1]); + if (pchRet[1] != 0x5b) + printf("ERROR: Proxy returned error %d\n", pchRet[1]); + return false; } - printf("Proxy connection established %s\n", addrConnect.ToStringLog().c_str()); + printf("proxy connected %s\n", addrConnect.ToStringLog().c_str()); } hSocketRet = hSocket; @@ -219,6 +221,7 @@ bool AddAddress(CAddrDB& addrdb, CAddress addr, bool fCurrentlyOnline) if (it == mapAddresses.end()) { // New address + printf("AddAddress(%s)\n", addr.ToStringLog().c_str()); mapAddresses.insert(make_pair(addr.GetKey(), addr)); addrdb.WriteAddress(addr); return true; @@ -256,7 +259,7 @@ void AddressCurrentlyConnected(const CAddress& addr) if (it != mapAddresses.end()) { CAddress& addrFound = (*it).second; - int64 nUpdateInterval = 60 * 60; + int64 nUpdateInterval = 20 * 60; if (addrFound.nTime < GetAdjustedTime() - nUpdateInterval) { // Periodically update most recently seen time @@ -417,7 +420,13 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) } /// debug print - printf("trying connection %s\n", addrConnect.ToStringLog().c_str()); + printf("trying connection %s lastseen=%.1fhrs lasttry=%.1fhrs\n", + addrConnect.ToStringLog().c_str(), + (double)(addrConnect.nTime - GetAdjustedTime())/3600.0, + (double)(addrConnect.nLastTry - GetAdjustedTime())/3600.0); + + CRITICAL_BLOCK(cs_mapAddresses) + mapAddresses[addrConnect.GetKey()].nLastTry = GetAdjustedTime(); // Connect SOCKET hSocket; @@ -428,7 +437,7 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) // Set to nonblocking #ifdef __WXMSW__ - u_long nOne = 1; + u_long nOne = 1; if (ioctlsocket(hSocket, FIONBIO, &nOne) == SOCKET_ERROR) printf("ConnectSocket() : ioctlsocket nonblocking setting failed, error %d\n", WSAGetLastError()); #else @@ -445,29 +454,23 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) CRITICAL_BLOCK(cs_vNodes) vNodes.push_back(pnode); - CRITICAL_BLOCK(cs_mapAddresses) - mapAddresses[addrConnect.GetKey()].nLastFailed = 0; + pnode->nTimeConnected = GetTime(); return pnode; } else { - CRITICAL_BLOCK(cs_mapAddresses) - mapAddresses[addrConnect.GetKey()].nLastFailed = GetAdjustedTime(); return NULL; } } void CNode::DoDisconnect() { + if (fDebug) + printf("%s ", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); printf("disconnecting node %s\n", addr.ToStringLog().c_str()); closesocket(hSocket); - // If outbound and never got version message, mark address as failed - if (!fInbound && !fSuccessfullyConnected) - CRITICAL_BLOCK(cs_mapAddresses) - mapAddresses[addr.GetKey()].nLastFailed = GetAdjustedTime(); - // All of a nodes broadcasts and subscriptions are automatically torn down // when it goes down, so a node has to stay up to keep its broadcast going. @@ -508,7 +511,7 @@ void ThreadSocketHandler(void* parg) PrintException(&e, "ThreadSocketHandler()"); } catch (...) { vnThreadsRunning[0]--; - PrintException(NULL, "ThreadSocketHandler()"); + throw; // support pthread_cancel() } printf("ThreadSocketHandler exiting\n"); @@ -531,15 +534,18 @@ void ThreadSocketHandler2(void* parg) vector vNodesCopy = vNodes; foreach(CNode* pnode, vNodesCopy) { - if (pnode->ReadyToDisconnect() && pnode->vRecv.empty() && pnode->vSend.empty()) + if (pnode->fDisconnect || + (pnode->GetRefCount() <= 0 && pnode->vRecv.empty() && pnode->vSend.empty())) { // remove from vNodes vNodes.erase(remove(vNodes.begin(), vNodes.end(), pnode), vNodes.end()); + + // close socket pnode->DoDisconnect(); // hold in disconnected pool until all refs are released pnode->nReleaseTime = max(pnode->nReleaseTime, GetTime() + 5 * 60); - if (pnode->fNetworkNode) + if (pnode->fNetworkNode || pnode->fInbound) pnode->Release(); vNodesDisconnected.push_back(pnode); } @@ -582,8 +588,10 @@ void ThreadSocketHandler2(void* parg) fd_set fdsetRecv; fd_set fdsetSend; + fd_set fdsetError; FD_ZERO(&fdsetRecv); FD_ZERO(&fdsetSend); + FD_ZERO(&fdsetError); SOCKET hSocketMax = 0; FD_SET(hListenSocket, &fdsetRecv); hSocketMax = max(hSocketMax, hListenSocket); @@ -592,6 +600,7 @@ void ThreadSocketHandler2(void* parg) foreach(CNode* pnode, vNodes) { FD_SET(pnode->hSocket, &fdsetRecv); + FD_SET(pnode->hSocket, &fdsetError); hSocketMax = max(hSocketMax, pnode->hSocket); TRY_CRITICAL_BLOCK(pnode->cs_vSend) if (!pnode->vSend.empty()) @@ -600,30 +609,21 @@ void ThreadSocketHandler2(void* parg) } vnThreadsRunning[0]--; - int nSelect = select(hSocketMax + 1, &fdsetRecv, &fdsetSend, NULL, &timeout); + int nSelect = select(hSocketMax + 1, &fdsetRecv, &fdsetSend, &fdsetError, &timeout); vnThreadsRunning[0]++; if (fShutdown) return; if (nSelect == SOCKET_ERROR) { int nErr = WSAGetLastError(); - printf("select failed: %d\n", nErr); + printf("socket select error %d\n", nErr); for (int i = 0; i <= hSocketMax; i++) - { FD_SET(i, &fdsetRecv); - FD_SET(i, &fdsetSend); - } + FD_ZERO(&fdsetSend); + FD_ZERO(&fdsetError); Sleep(timeout.tv_usec/1000); } - //// debug print - //foreach(CNode* pnode, vNodes) - //{ - // printf("vRecv = %-5d ", pnode->vRecv.size()); - // printf("vSend = %-5d ", pnode->vSend.size()); - //} - //printf("\n"); - // // Accept new connections @@ -641,7 +641,7 @@ void ThreadSocketHandler2(void* parg) if (hSocket == INVALID_SOCKET) { if (WSAGetLastError() != WSAEWOULDBLOCK) - printf("ERROR ThreadSocketHandler accept failed: %d\n", WSAGetLastError()); + printf("socket error accept failed: %d\n", WSAGetLastError()); } else { @@ -669,7 +669,7 @@ void ThreadSocketHandler2(void* parg) // // Receive // - if (FD_ISSET(hSocket, &fdsetRecv)) + if (FD_ISSET(hSocket, &fdsetRecv) || FD_ISSET(hSocket, &fdsetError)) { TRY_CRITICAL_BLOCK(pnode->cs_vRecv) { @@ -677,25 +677,29 @@ void ThreadSocketHandler2(void* parg) unsigned int nPos = vRecv.size(); // typical socket buffer is 8K-64K - const unsigned int nBufSize = 0x10000; - vRecv.resize(nPos + nBufSize); - int nBytes = recv(hSocket, &vRecv[nPos], nBufSize, 0); - vRecv.resize(nPos + max(nBytes, 0)); - if (nBytes == 0) + char pchBuf[0x10000]; + int nBytes = recv(hSocket, pchBuf, sizeof(pchBuf), MSG_DONTWAIT); + if (nBytes > 0) + { + vRecv.resize(nPos + nBytes); + memcpy(&vRecv[nPos], pchBuf, nBytes); + pnode->nLastRecv = GetTime(); + } + else if (nBytes == 0) { // socket closed gracefully if (!pnode->fDisconnect) - printf("recv: socket closed\n"); + printf("socket closed\n"); pnode->fDisconnect = true; } else if (nBytes < 0) { - // socket error + // error int nErr = WSAGetLastError(); if (nErr != WSAEWOULDBLOCK && nErr != WSAEMSGSIZE && nErr != WSAEINTR && nErr != WSAEINPROGRESS) { if (!pnode->fDisconnect) - printf("recv failed: %d\n", nErr); + printf("socket recv error %d\n", nErr); pnode->fDisconnect = true; } } @@ -712,28 +716,63 @@ void ThreadSocketHandler2(void* parg) CDataStream& vSend = pnode->vSend; if (!vSend.empty()) { - int nBytes = send(hSocket, &vSend[0], vSend.size(), MSG_NOSIGNAL); + int nBytes = send(hSocket, &vSend[0], vSend.size(), MSG_NOSIGNAL | MSG_DONTWAIT); if (nBytes > 0) { vSend.erase(vSend.begin(), vSend.begin() + nBytes); + pnode->nLastSend = GetTime(); } - else if (nBytes == 0) - { - if (pnode->ReadyToDisconnect()) - pnode->vSend.clear(); - } - else + else if (nBytes < 0) { - printf("send error %d\n", nBytes); - if (pnode->ReadyToDisconnect()) - pnode->vSend.clear(); + // error + int nErr = WSAGetLastError(); + if (nErr != WSAEWOULDBLOCK && nErr != WSAEMSGSIZE && nErr != WSAEINTR && nErr != WSAEINPROGRESS) + { + printf("socket send error %d\n", nErr); + pnode->fDisconnect = true; + } } } } } + + // + // Inactivity checking + // + if (pnode->vSend.empty()) + pnode->nLastSendEmpty = GetTime(); + if (GetTime() - pnode->nTimeConnected > 60) + { + if (pnode->nLastRecv == 0 || pnode->nLastSend == 0) + { + printf("socket no message in first 60 seconds, %d %d\n", pnode->nLastRecv != 0, pnode->nLastSend != 0); + pnode->fDisconnect = true; + } + else if (GetTime() - pnode->nLastSend > 10 * 60 && GetTime() - pnode->nLastSendEmpty > 10 * 60) + { + printf("socket not sending\n"); + pnode->fDisconnect = true; + } + else if (GetTime() - pnode->nLastRecv > (pnode->nVersion >= 107 ? 15*60 : 90*60)) + { + printf("socket inactivity timeout\n"); + pnode->fDisconnect = true; + } + } + } + + + //// debug heartbeat + static int64 nHeartbeat1; + if (GetTime() - nHeartbeat1 >= 5 * 60) + { + printf("%s sendrecv\n", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); + nHeartbeat1 = GetTime(); + fDebug = true; } + nThreadSocketHandlerHeartbeat = GetTime(); Sleep(10); } } @@ -772,15 +811,20 @@ void ThreadOpenConnections2(void* parg) { printf("ThreadOpenConnections started\n"); - // Connect to one specified address + // Connect to specific addresses while (mapArgs.count("-connect")) { - OpenNetworkConnection(CAddress(mapArgs["-connect"])); - for (int i = 0; i < 10; i++) + foreach(string strAddr, mapMultiArgs["-connect"]) { - Sleep(1000); - if (fShutdown) - return; + CAddress addr(strAddr, NODE_NETWORK); + if (addr.IsValid()) + OpenNetworkConnection(addr); + for (int i = 0; i < 10; i++) + { + Sleep(1000); + if (fShutdown) + return; + } } } @@ -821,12 +865,7 @@ void ThreadOpenConnections2(void* parg) // Choose an address to connect to based on most recently seen // CAddress addrConnect; - int64 nBestTime = 0; - int64 nDelay = ((60 * 60) << vNodes.size()); - if (vNodes.size() >= 3) - nDelay *= 4; - if (nGotIRCAddresses > 0) - nDelay *= 100; + int64 nBest = INT64_MIN; // Do this here so we don't have to critsect vNodes inside mapAddresses critsect set setConnected; @@ -841,24 +880,51 @@ void ThreadOpenConnections2(void* parg) const CAddress& addr = item.second; if (!addr.IsIPv4() || !addr.IsValid() || setConnected.count(addr.ip)) continue; + int64 nSinceLastSeen = GetAdjustedTime() - addr.nTime; + int64 nSinceLastTry = GetAdjustedTime() - addr.nLastTry; // Randomize the order in a deterministic way, putting the standard port first - int64 nRandomizer = (uint64)(addr.nLastFailed * 9567851 + addr.ip * 7789) % (1 * 60 * 60); + int64 nRandomizer = (uint64)(addr.nLastTry * 9567851 + addr.ip * 7789) % (30 * 60); if (addr.port != DEFAULT_PORT) - nRandomizer += 1 * 60 * 60; + nRandomizer += 30 * 60; + + // Last seen Base retry frequency + // <1 hour 10 min + // 1 hour 1 hour + // 4 hours 2 hours + // 24 hours 5 hours + // 48 hours 7 hours + // 7 days 13 hours + // 30 days 27 hours + // 90 days 46 hours + // 365 days 93 hours + int64 nDelay = 3600.0 * sqrt(fabs(nSinceLastSeen) / 3600.0) + nRandomizer; + + // Fast reconnect for one hour after last seen + if (nSinceLastSeen < 60 * 60) + nDelay = 10 * 60; // Limit retry frequency - if (GetAdjustedTime() < addr.nLastFailed + nDelay + nRandomizer) + if (nSinceLastTry < nDelay) continue; - // Try again only after all addresses had a first attempt - int64 nTime = addr.nTime - nRandomizer; - if (addr.nLastFailed > addr.nTime) - nTime -= 365 * 24 * 60 * 60; + // If we have IRC, we'll be notified when they first come online, + // and again every 24 hours by the refresh broadcast. + if (nGotIRCAddresses > 0 && vNodes.size() >= 2 && nSinceLastSeen > 24 * 60 * 60) + continue; - if (nTime > nBestTime) + // Only try the old stuff if we don't have enough connections + if (vNodes.size() >= 2 && nSinceLastSeen > 7 * 24 * 60 * 60) + continue; + if (vNodes.size() >= 4 && nSinceLastSeen > 24 * 60 * 60) + continue; + + // If multiple addresses are ready, prioritize by time since + // last seen and time since last tried. + int64 nScore = min(nSinceLastTry, (int64)24 * 60 * 60) - nSinceLastSeen - nRandomizer; + if (nScore > nBest) { - nBestTime = nTime; + nBest = nScore; addrConnect = addr; } } @@ -941,7 +1007,7 @@ void ThreadMessageHandler(void* parg) void ThreadMessageHandler2(void* parg) { printf("ThreadMessageHandler started\n"); - SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_BELOW_NORMAL); + SetThreadPriority(THREAD_PRIORITY_BELOW_NORMAL); loop { // Poll the connected nodes for messages @@ -1063,39 +1129,31 @@ bool BindListenPort(string& strError) return true; } -bool StartNode(string& strError) +void StartNode(void* parg) { - strError = ""; if (pnodeLocalHost == NULL) pnodeLocalHost = new CNode(INVALID_SOCKET, CAddress("127.0.0.1", nLocalServices)); #ifdef __WXMSW__ // Get local host ip - char pszHostName[255]; - if (gethostname(pszHostName, sizeof(pszHostName)) == SOCKET_ERROR) + char pszHostName[1000] = ""; + if (gethostname(pszHostName, sizeof(pszHostName)) != SOCKET_ERROR) { - strError = strprintf("Error: Unable to get IP address of this computer (gethostname returned error %d)", WSAGetLastError()); - printf("%s\n", strError.c_str()); - return false; - } - struct hostent* phostent = gethostbyname(pszHostName); - if (!phostent) - { - strError = strprintf("Error: Unable to get IP address of this computer (gethostbyname returned error %d)", WSAGetLastError()); - printf("%s\n", strError.c_str()); - return false; - } - - // Take the first IP that isn't loopback 127.x.x.x - for (int i = 0; phostent->h_addr_list[i] != NULL; i++) - printf("host ip %d: %s\n", i, CAddress(*(unsigned int*)phostent->h_addr_list[i]).ToStringIP().c_str()); - for (int i = 0; phostent->h_addr_list[i] != NULL; i++) - { - CAddress addr(*(unsigned int*)phostent->h_addr_list[i], DEFAULT_PORT, nLocalServices); - if (addr.IsValid() && addr.GetByte(3) != 127) + struct hostent* phostent = gethostbyname(pszHostName); + if (phostent) { - addrLocalHost = addr; - break; + // Take the first IP that isn't loopback 127.x.x.x + for (int i = 0; phostent->h_addr_list[i] != NULL; i++) + printf("host ip %d: %s\n", i, CAddress(*(unsigned int*)phostent->h_addr_list[i]).ToStringIP().c_str()); + for (int i = 0; phostent->h_addr_list[i] != NULL; i++) + { + CAddress addr(*(unsigned int*)phostent->h_addr_list[i], DEFAULT_PORT, nLocalServices); + if (addr.IsValid() && addr.GetByte(3) != 127) + { + addrLocalHost = addr; + break; + } + } } } #else @@ -1145,45 +1203,85 @@ bool StartNode(string& strError) } else { - if (addrIncoming.ip) + if (addrIncoming.IsValid()) addrLocalHost.ip = addrIncoming.ip; if (GetMyExternalIP(addrLocalHost.ip)) { addrIncoming = addrLocalHost; CWalletDB().WriteSetting("addrIncoming", addrIncoming); + printf("addrLocalHost = %s\n", addrLocalHost.ToString().c_str()); } } - // Get addresses from IRC and advertise ours - if (_beginthread(ThreadIRCSeed, 0, NULL) == -1) - printf("Error: _beginthread(ThreadIRCSeed) failed\n"); - // // Start threads // - if (_beginthread(ThreadSocketHandler, 0, NULL) == -1) - { - strError = "Error: _beginthread(ThreadSocketHandler) failed"; - printf("%s\n", strError.c_str()); - return false; - } - if (_beginthread(ThreadOpenConnections, 0, NULL) == -1) - { - strError = "Error: _beginthread(ThreadOpenConnections) failed"; - printf("%s\n", strError.c_str()); - return false; - } + // Get addresses from IRC and advertise ours + if (!CreateThread(ThreadIRCSeed, NULL)) + printf("Error: CreateThread(ThreadIRCSeed) failed\n"); + + // Send and receive from sockets, accept connections + pthread_t hThreadSocketHandler = CreateThread(ThreadSocketHandler, NULL, true); + + // Initiate outbound connections + if (!CreateThread(ThreadOpenConnections, NULL)) + printf("Error: CreateThread(ThreadOpenConnections) failed\n"); + + // Process messages + if (!CreateThread(ThreadMessageHandler, NULL)) + printf("Error: CreateThread(ThreadMessageHandler) failed\n"); - if (_beginthread(ThreadMessageHandler, 0, NULL) == -1) + // Generate coins in the background + GenerateBitcoins(fGenerateBitcoins); + + // + // Thread monitoring + // + loop { - strError = "Error: _beginthread(ThreadMessageHandler) failed"; - printf("%s\n", strError.c_str()); - return false; + Sleep(15000); + if (GetTime() - nThreadSocketHandlerHeartbeat > 4 * 60) + { + // First see if closing sockets will free it + printf("*** ThreadSocketHandler is stopped ***\n"); + CRITICAL_BLOCK(cs_vNodes) + { + foreach(CNode* pnode, vNodes) + { + bool fGot = false; + TRY_CRITICAL_BLOCK(pnode->cs_vRecv) + TRY_CRITICAL_BLOCK(pnode->cs_vSend) + fGot = true; + if (!fGot) + { + printf("*** closing socket\n"); + closesocket(pnode->hSocket); + pnode->fDisconnect = true; + } + } + } + Sleep(10000); + if (GetTime() - nThreadSocketHandlerHeartbeat < 60) + continue; + + // Hopefully it never comes to this. + // We know it'll always be hung in the recv or send call. + // cs_vRecv or cs_vSend may be left permanently unreleased, + // but we always only use TRY_CRITICAL_SECTION on them. + printf("*** Restarting ThreadSocketHandler ***\n"); + TerminateThread(hThreadSocketHandler, 0); + #ifdef __WXMSW__ + CloseHandle(hThreadSocketHandler); + #endif + vnThreadsRunning[0] = 0; + + // Restart + hThreadSocketHandler = CreateThread(ThreadSocketHandler, NULL, true); + nThreadSocketHandlerHeartbeat = GetTime(); + } } - - return true; } bool StopNode() diff --git a/net.h b/net.h index 7b83d462..6300d3ef 100644 --- a/net.h +++ b/net.h @@ -29,7 +29,7 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout=0); void AbandonRequests(void (*fn)(void*, CDataStream&), void* param1); bool AnySubscribed(unsigned int nChannel); bool BindListenPort(string& strError=REF(string())); -bool StartNode(string& strError=REF(string())); +void StartNode(void* parg); bool StopNode(); @@ -39,7 +39,6 @@ bool StopNode(); - // // Message header // (4) message start @@ -139,7 +138,7 @@ public: unsigned int nTime; // memory only - unsigned int nLastFailed; + unsigned int nLastTry; CAddress() { @@ -183,7 +182,7 @@ public: ip = INADDR_NONE; port = DEFAULT_PORT; nTime = GetAdjustedTime(); - nLastFailed = 0; + nLastTry = 0; } bool SetAddress(const char* pszIn) @@ -458,6 +457,7 @@ extern uint64 nLocalHostNonce; extern bool fShutdown; extern array vnThreadsRunning; extern SOCKET hListenSocket; +extern int64 nThreadSocketHandlerHeartbeat; extern vector vNodes; extern CCriticalSection cs_vNodes; @@ -486,6 +486,10 @@ public: CDataStream vRecv; CCriticalSection cs_vSend; CCriticalSection cs_vRecv; + int64 nLastSend; + int64 nLastRecv; + int64 nLastSendEmpty; + int64 nTimeConnected; unsigned int nPushPos; CAddress addr; int nVersion; @@ -523,6 +527,10 @@ public: hSocket = hSocketIn; vSend.SetType(SER_NETWORK); vRecv.SetType(SER_NETWORK); + nLastSend = 0; + nLastRecv = 0; + nLastSendEmpty = GetTime(); + nTimeConnected = GetTime(); nPushPos = -1; addr = addrIn; nVersion = 0; @@ -542,7 +550,7 @@ public: CAddress addrYou = (fUseProxy ? CAddress("0.0.0.0") : addr); CAddress addrMe = (fUseProxy ? CAddress("0.0.0.0") : addrLocalHost); RAND_bytes((unsigned char*)&nLocalHostNonce, sizeof(nLocalHostNonce)); - PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce); + PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce, "linux-test5"); } ~CNode() @@ -557,11 +565,6 @@ private: public: - bool ReadyToDisconnect() - { - return fDisconnect || GetRefCount() <= 0; - } - int GetRefCount() { return max(nRefCount, 0) + (GetTime() < nReleaseTime ? 1 : 0); @@ -635,6 +638,8 @@ public: AbortMessage(); nPushPos = vSend.size(); vSend << CMessageHeader(pszCommand, 0); + if (fDebug) + printf("%s ", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); printf("sending: %s ", pszCommand); } diff --git a/ui.cpp b/ui.cpp index e12970b8..9d7556cb 100644 --- a/ui.cpp +++ b/ui.cpp @@ -25,6 +25,7 @@ map mapAddressBook; bool fRandSendTest = false; void RandSend(); extern int g_isPainting; +bool fClosedToTray = false; // Settings int fShowGenerated = true; @@ -413,16 +414,17 @@ void Shutdown(void* parg) void CMainFrame::OnClose(wxCloseEvent& event) { - if (fMinimizeToTray && fMinimizeOnClose && event.CanVeto() && !IsIconized()) + if (fMinimizeOnClose && event.CanVeto() && !IsIconized()) { // Divert close to minimize event.Veto(); + fClosedToTray = true; Iconize(true); } else { Destroy(); - _beginthread(Shutdown, 0, NULL); + CreateThread(Shutdown, NULL); } } @@ -430,7 +432,16 @@ void CMainFrame::OnIconize(wxIconizeEvent& event) { // Hide the task bar button when minimized. // Event is sent when the frame is minimized or restored. - Show(!fMinimizeToTray || !event.Iconized()); + if (!event.Iconized()) + fClosedToTray = false; +#ifndef __WXMSW__ + // Tray is not reliable on Linux gnome + fClosedToTray = false; +#endif + if (fMinimizeToTray && event.Iconized()) + fClosedToTray = true; + Show(!fClosedToTray); + ptaskbaricon->Show(fMinimizeToTray || fClosedToTray); } void CMainFrame::OnMouseEvents(wxMouseEvent& event) @@ -527,7 +538,6 @@ bool CMainFrame::DeleteLine(uint256 hashKey) string FormatTxStatus(const CWalletTx& wtx) { // Status - int nDepth = wtx.GetDepthInMainChain(); if (!wtx.IsFinal()) { if (wtx.nLockTime < 500000000) @@ -535,10 +545,16 @@ string FormatTxStatus(const CWalletTx& wtx) else return strprintf("Open until %s", DateTimeStr(wtx.nLockTime).c_str()); } - else if (nDepth < 6) - return strprintf("%d/unconfirmed", nDepth); else - return strprintf("%d blocks", nDepth); + { + int nDepth = wtx.GetDepthInMainChain(); + if (GetAdjustedTime() - wtx.nTimeReceived > 2 * 60 && wtx.GetRequestCount() == 0) + return strprintf("%d/offline?", nDepth); + else if (nDepth < 6) + return strprintf("%d/unconfirmed", nDepth); + else + return strprintf("%d blocks", nDepth); + } } string SingleLine(const string& strIn) @@ -629,9 +645,17 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) foreach(const CTxOut& txout, wtx.vout) nUnmatured += txout.GetCredit(); if (wtx.IsInMainChain()) - strDescription += strprintf(" (%s matures in %d more blocks)", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); + { + strDescription = strprintf("Generated (%s matures in %d more blocks)", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); + + // Check if the block was requested by anyone + if (GetAdjustedTime() - wtx.nTimeReceived > 2 * 60 && wtx.GetRequestCount() == 0) + strDescription = "Generated - Warning: This block was not received by any other nodes and will probably not be accepted!"; + } else - strDescription += " (not accepted)"; + { + strDescription = "Generated (not accepted)"; + } } } else if (!mapValue["from"].empty() || !mapValue["message"].empty()) @@ -701,8 +725,11 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) strStatus, nTime ? DateTimeStr(nTime) : "", "Payment to yourself", - FormatMoney(nNet - nValue, true), - FormatMoney(nValue, true)); + "", + ""); + /// issue: can't tell which is the payment and which is the change anymore + // FormatMoney(nNet - nValue, true), + // FormatMoney(nValue, true)); } else if (fAllFromMe) { @@ -1028,6 +1055,9 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, nTransactionCount); m_statusBar->SetStatusText(strStatus, 2); + if (fDebug && GetTime() - nThreadSocketHandlerHeartbeat > 60) + m_statusBar->SetStatusText(" ERROR: ThreadSocketHandler has stopped", 0); + // Pass through to listctrl to actually do the paint, we're just hooking the message m_listCtrl->Disconnect(wxEVT_PAINT, (wxObjectEventFunction)NULL, NULL, this); m_listCtrl->GetEventHandler()->ProcessEvent(event); @@ -1237,7 +1267,19 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails - strHTML += "Status: " + FormatTxStatus(wtx) + "
"; + strHTML += "Status: " + FormatTxStatus(wtx); + int nRequests = wtx.GetRequestCount(); + if (nRequests != -1) + { + if (nRequests == 0) + strHTML += ", has not been successfully broadcast yet"; + else if (nRequests == 1) + strHTML += strprintf(", broadcast through %d node", nRequests); + else + strHTML += strprintf(", broadcast through %d nodes", nRequests); + } + strHTML += "
"; + strHTML += "Date: " + (nTime ? DateTimeStr(nTime) : "") + "
"; @@ -1366,9 +1408,10 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails if (fAllToMe) { // Payment to self - int64 nValue = wtx.vout[0].nValue; - strHTML += "Debit: " + FormatMoney(-nValue) + "
"; - strHTML += "Credit: " + FormatMoney(nValue) + "
"; + /// issue: can't tell which is the payment and which is the change anymore + //int64 nValue = wtx.vout[0].nValue; + //strHTML += "Debit: " + FormatMoney(-nValue) + "
"; + //strHTML += "Credit: " + FormatMoney(nValue) + "
"; } int64 nTxFee = nDebit - wtx.GetValueOut(); @@ -1469,6 +1512,9 @@ COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) //m_listBox->Append("Test 2"); m_listBox->SetSelection(0); SelectPage(0); +#ifndef __WXMSW__ + m_checkBoxMinimizeOnClose->SetLabel("&Minimize on close"); +#endif // Init values m_textCtrlTransactionFee->SetValue(FormatMoney(nTransactionFee)); @@ -1481,9 +1527,7 @@ COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) m_spinCtrlLimitProcessors->SetRange(1, nProcessors); m_checkBoxStartOnSystemStartup->SetValue(fTmpStartOnSystemStartup = GetStartOnSystemStartup()); m_checkBoxMinimizeToTray->SetValue(fMinimizeToTray); - m_checkBoxMinimizeOnClose->Enable(fMinimizeToTray); - m_checkBoxMinimizeOnClose->SetValue(fMinimizeToTray && fMinimizeOnClose); - fTmpMinimizeOnClose = fMinimizeOnClose; + m_checkBoxMinimizeOnClose->SetValue(fMinimizeOnClose); m_checkBoxUseProxy->SetValue(fUseProxy); m_textCtrlProxyIP->Enable(fUseProxy); m_textCtrlProxyPort->Enable(fUseProxy); @@ -1521,22 +1565,6 @@ void COptionsDialog::OnCheckBoxLimitProcessors(wxCommandEvent& event) m_spinCtrlLimitProcessors->Enable(event.IsChecked()); } -void COptionsDialog::OnCheckBoxMinimizeToTray(wxCommandEvent& event) -{ - m_checkBoxMinimizeOnClose->Enable(event.IsChecked()); - - // Save the value in fTmpMinimizeOnClose so we can - // show the checkbox unchecked when its parent is unchecked - if (event.IsChecked()) - m_checkBoxMinimizeOnClose->SetValue(fTmpMinimizeOnClose); - else - { - fTmpMinimizeOnClose = m_checkBoxMinimizeOnClose->GetValue(); - m_checkBoxMinimizeOnClose->SetValue(false); - } - -} - void COptionsDialog::OnCheckBoxUseProxy(wxCommandEvent& event) { m_textCtrlProxyIP->Enable(event.IsChecked()); @@ -1608,12 +1636,12 @@ void COptionsDialog::OnButtonApply(wxCommandEvent& event) { fMinimizeToTray = m_checkBoxMinimizeToTray->GetValue(); walletdb.WriteSetting("fMinimizeToTray", fMinimizeToTray); - ptaskbaricon->Show(fMinimizeToTray); + ptaskbaricon->Show(fMinimizeToTray || fClosedToTray); } - if (fMinimizeOnClose != (fMinimizeToTray ? m_checkBoxMinimizeOnClose->GetValue() : fTmpMinimizeOnClose)) + if (fMinimizeOnClose != m_checkBoxMinimizeOnClose->GetValue()) { - fMinimizeOnClose = (fMinimizeToTray ? m_checkBoxMinimizeOnClose->GetValue() : fTmpMinimizeOnClose); + fMinimizeOnClose = m_checkBoxMinimizeOnClose->GetValue(); walletdb.WriteSetting("fMinimizeOnClose", fMinimizeOnClose); } @@ -1643,6 +1671,9 @@ CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) if (str.Find('Â') != wxNOT_FOUND) str.Remove(str.Find('Â'), 1); m_staticTextMain->SetLabel(str); +#ifndef __WXMSW__ + SetSize(510, 380); +#endif } void CAboutDialog::OnButtonOK(wxCommandEvent& event) @@ -1849,7 +1880,7 @@ CSendingDialog::CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 n SetTitle(strprintf("Sending %s to %s", FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); m_textCtrlStatus->SetValue(""); - _beginthread(SendingDialogStartTransfer, 0, this); + CreateThread(SendingDialogStartTransfer, this); } CSendingDialog::~CSendingDialog() @@ -2856,7 +2887,7 @@ CViewProductDialog::CViewProductDialog(wxWindow* parent, const CProduct& product this->Layout(); // Request details from seller - _beginthread(ThreadRequestProductDetails, 0, new pair(product, GetEventHandler())); + CreateThread(ThreadRequestProductDetails, new pair(product, GetEventHandler())); } CViewProductDialog::~CViewProductDialog() @@ -3256,6 +3287,7 @@ void CEditReviewDialog::GetReview(CReview& review) enum { ID_TASKBAR_RESTORE = 10001, + ID_TASKBAR_OPTIONS, ID_TASKBAR_GENERATE, ID_TASKBAR_EXIT, }; @@ -3263,6 +3295,7 @@ enum BEGIN_EVENT_TABLE(CMyTaskBarIcon, wxTaskBarIcon) EVT_TASKBAR_LEFT_DCLICK(CMyTaskBarIcon::OnLeftButtonDClick) EVT_MENU(ID_TASKBAR_RESTORE, CMyTaskBarIcon::OnMenuRestore) + EVT_MENU(ID_TASKBAR_OPTIONS, CMyTaskBarIcon::OnMenuOptions) EVT_MENU(ID_TASKBAR_GENERATE, CMyTaskBarIcon::OnMenuGenerate) EVT_UPDATE_UI(ID_TASKBAR_GENERATE, CMyTaskBarIcon::OnUpdateUIGenerate) EVT_MENU(ID_TASKBAR_EXIT, CMyTaskBarIcon::OnMenuExit) @@ -3312,9 +3345,18 @@ void CMyTaskBarIcon::OnMenuRestore(wxCommandEvent& event) Restore(); } +void CMyTaskBarIcon::OnMenuOptions(wxCommandEvent& event) +{ + // Since it's modal, get the main window to do it + wxCommandEvent event2(wxEVT_COMMAND_MENU_SELECTED, wxID_MENUOPTIONSOPTIONS); + pframeMain->AddPendingEvent(event2); +} + void CMyTaskBarIcon::Restore() { pframeMain->Show(); + wxIconizeEvent event(0, false); + pframeMain->AddPendingEvent(event); pframeMain->Iconize(false); pframeMain->Raise(); } @@ -3344,6 +3386,7 @@ wxMenu* CMyTaskBarIcon::CreatePopupMenu() { wxMenu* pmenu = new wxMenu; pmenu->Append(ID_TASKBAR_RESTORE, "&Open Bitcoin"); + pmenu->Append(ID_TASKBAR_OPTIONS, "O&ptions..."); pmenu->AppendCheckItem(ID_TASKBAR_GENERATE, "&Generate Coins")->Check(fGenerateBitcoins); #ifndef __WXMAC_OSX__ // Mac has built-in quit menu pmenu->AppendSeparator(); @@ -3582,7 +3625,7 @@ bool CMyApp::OnInit2() { CBlockIndex* pindex = (*mi).second; CBlock block; - block.ReadFromDisk(pindex, true); + block.ReadFromDisk(pindex); block.BuildMerkleTree(); block.print(); printf("\n"); @@ -3632,20 +3675,20 @@ bool CMyApp::OnInit2() if (mapArgs.count("-min")) pframeMain->Iconize(true); pframeMain->Show(true); // have to show first to get taskbar button to hide - pframeMain->Show(!fMinimizeToTray || !pframeMain->IsIconized()); - ptaskbaricon->Show(fMinimizeToTray); + if (fMinimizeToTray && pframeMain->IsIconized()) + fClosedToTray = true; + pframeMain->Show(!fClosedToTray); + ptaskbaricon->Show(fMinimizeToTray || fClosedToTray); - _beginthread(ThreadDelayedRepaint, 0, NULL); + CreateThread(ThreadDelayedRepaint, NULL); if (!CheckDiskSpace()) return false; RandAddSeedPerfmon(); - if (!StartNode(strErrors)) - wxMessageBox(strErrors, "Bitcoin"); - - GenerateBitcoins(fGenerateBitcoins); + if (!CreateThread(StartNode, NULL)) + wxMessageBox("Error: CreateThread(StartNode) failed", "Bitcoin"); if (fFirstRun) SetStartOnSystemStartup(true); diff --git a/ui.h b/ui.h index a919c366..c4bf8b66 100644 --- a/ui.h +++ b/ui.h @@ -112,7 +112,6 @@ protected: void OnListBox(wxCommandEvent& event); void OnKillFocusTransactionFee(wxFocusEvent& event); void OnCheckBoxLimitProcessors(wxCommandEvent& event); - void OnCheckBoxMinimizeToTray(wxCommandEvent& event); void OnCheckBoxUseProxy(wxCommandEvent& event); void OnKillFocusProxy(wxFocusEvent& event); @@ -447,6 +446,7 @@ protected: // Event handlers void OnLeftButtonDClick(wxTaskBarIconEvent& event); void OnMenuRestore(wxCommandEvent& event); + void OnMenuOptions(wxCommandEvent& event); void OnUpdateUIGenerate(wxUpdateUIEvent& event); void OnMenuGenerate(wxCommandEvent& event); void OnMenuExit(wxCommandEvent& event); diff --git a/uibase.cpp b/uibase.cpp index 7bc8081f..f05f1095 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -45,7 +45,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_menuOptions->Append( m_menuOptionsChangeYourAddress ); wxMenuItem* m_menuOptionsOptions; - m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( wxT("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_MENUOPTIONSOPTIONS, wxString( wxT("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuOptions->Append( m_menuOptionsOptions ); m_menubar->Append( m_menuOptions, wxT("&Options") ); @@ -428,21 +428,13 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer69->Add( m_checkBoxStartOnSystemStartup, 0, wxALL, 5 ); - m_checkBoxMinimizeToTray = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Minimize to the system tray instead of the taskbar"), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkBoxMinimizeToTray = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Minimize to the tray instead of the taskbar"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_checkBoxMinimizeToTray, 0, wxALL, 5 ); - wxBoxSizer* bSizer101; - bSizer101 = new wxBoxSizer( wxHORIZONTAL ); + m_checkBoxMinimizeOnClose = new wxCheckBox( m_panelMain, wxID_ANY, wxT("M&inimize to the tray on close"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer101->Add( 16, 0, 0, 0, 5 ); - - m_checkBoxMinimizeOnClose = new wxCheckBox( m_panelMain, wxID_ANY, wxT("Mi&nimize to system tray on close"), wxDefaultPosition, wxDefaultSize, 0 ); - - bSizer101->Add( m_checkBoxMinimizeOnClose, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - bSizer69->Add( bSizer101, 1, wxEXPAND, 5 ); + bSizer69->Add( m_checkBoxMinimizeOnClose, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); wxBoxSizer* bSizer102; bSizer102 = new wxBoxSizer( wxHORIZONTAL ); diff --git a/uibase.h b/uibase.h index 2cb99e9a..e0c15218 100644 --- a/uibase.h +++ b/uibase.h @@ -43,51 +43,52 @@ #define wxID_MAINFRAME 1000 #define wxID_VIEWSHOWGENERATED 1001 #define wxID_OPTIONSGENERATEBITCOINS 1002 -#define wxID_BUTTONSEND 1003 -#define wxID_BUTTONRECEIVE 1004 -#define wxID_TEXTCTRLADDRESS 1005 -#define wxID_BUTTONCOPY 1006 -#define wxID_BUTTONCHANGE 1007 -#define wxID_TRANSACTIONFEE 1008 -#define wxID_PROXYIP 1009 -#define wxID_PROXYPORT 1010 -#define wxID_TEXTCTRLPAYTO 1011 -#define wxID_BUTTONPASTE 1012 -#define wxID_BUTTONADDRESSBOOK 1013 -#define wxID_TEXTCTRLAMOUNT 1014 -#define wxID_CHOICETRANSFERTYPE 1015 -#define wxID_LISTCTRL 1016 -#define wxID_BUTTONRENAME 1017 -#define wxID_BUTTONNEW 1018 -#define wxID_BUTTONEDIT 1019 -#define wxID_BUTTONDELETE 1020 -#define wxID_DEL0 1021 -#define wxID_DEL1 1022 -#define wxID_DEL2 1023 -#define wxID_DEL3 1024 -#define wxID_DEL4 1025 -#define wxID_DEL5 1026 -#define wxID_DEL6 1027 -#define wxID_DEL7 1028 -#define wxID_DEL8 1029 -#define wxID_DEL9 1030 -#define wxID_DEL10 1031 -#define wxID_DEL11 1032 -#define wxID_DEL12 1033 -#define wxID_DEL13 1034 -#define wxID_DEL14 1035 -#define wxID_DEL15 1036 -#define wxID_DEL16 1037 -#define wxID_DEL17 1038 -#define wxID_DEL18 1039 -#define wxID_DEL19 1040 -#define wxID_BUTTONPREVIEW 1041 -#define wxID_BUTTONSAMPLE 1042 -#define wxID_CANCEL2 1043 -#define wxID_BUTTONBACK 1044 -#define wxID_BUTTONNEXT 1045 -#define wxID_SUBMIT 1046 -#define wxID_TEXTCTRL 1047 +#define wxID_MENUOPTIONSOPTIONS 1003 +#define wxID_BUTTONSEND 1004 +#define wxID_BUTTONRECEIVE 1005 +#define wxID_TEXTCTRLADDRESS 1006 +#define wxID_BUTTONCOPY 1007 +#define wxID_BUTTONCHANGE 1008 +#define wxID_TRANSACTIONFEE 1009 +#define wxID_PROXYIP 1010 +#define wxID_PROXYPORT 1011 +#define wxID_TEXTCTRLPAYTO 1012 +#define wxID_BUTTONPASTE 1013 +#define wxID_BUTTONADDRESSBOOK 1014 +#define wxID_TEXTCTRLAMOUNT 1015 +#define wxID_CHOICETRANSFERTYPE 1016 +#define wxID_LISTCTRL 1017 +#define wxID_BUTTONRENAME 1018 +#define wxID_BUTTONNEW 1019 +#define wxID_BUTTONEDIT 1020 +#define wxID_BUTTONDELETE 1021 +#define wxID_DEL0 1022 +#define wxID_DEL1 1023 +#define wxID_DEL2 1024 +#define wxID_DEL3 1025 +#define wxID_DEL4 1026 +#define wxID_DEL5 1027 +#define wxID_DEL6 1028 +#define wxID_DEL7 1029 +#define wxID_DEL8 1030 +#define wxID_DEL9 1031 +#define wxID_DEL10 1032 +#define wxID_DEL11 1033 +#define wxID_DEL12 1034 +#define wxID_DEL13 1035 +#define wxID_DEL14 1036 +#define wxID_DEL15 1037 +#define wxID_DEL16 1038 +#define wxID_DEL17 1039 +#define wxID_DEL18 1040 +#define wxID_DEL19 1041 +#define wxID_BUTTONPREVIEW 1042 +#define wxID_BUTTONSAMPLE 1043 +#define wxID_CANCEL2 1044 +#define wxID_BUTTONBACK 1045 +#define wxID_BUTTONNEXT 1046 +#define wxID_SUBMIT 1047 +#define wxID_TEXTCTRL 1048 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase @@ -203,7 +204,6 @@ class COptionsDialogBase : public wxDialog wxStaticText* m_staticText35; wxCheckBox* m_checkBoxStartOnSystemStartup; wxCheckBox* m_checkBoxMinimizeToTray; - wxCheckBox* m_checkBoxMinimizeOnClose; wxCheckBox* m_checkBoxUseProxy; diff --git a/uiproject.fbp b/uiproject.fbp index 3aa1c86b..58a99bdd 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -70,7 +70,7 @@ - + 240,240,240 1 @@ -193,7 +193,7 @@ 0 1 - wxID_ANY + wxID_MENUOPTIONSOPTIONS wxITEM_NORMAL &Options... m_menuOptionsOptions @@ -2319,7 +2319,7 @@ 0 wxID_ANY - &Minimize to the system tray instead of the taskbar + &Minimize to the tray instead of the taskbar m_checkBoxMinimizeToTray @@ -2360,75 +2360,54 @@ 5 - wxEXPAND - 1 - + wxALL|wxALIGN_CENTER_VERTICAL + 0 + + + 0 + + 1 + + + 0 + wxID_ANY + M&inimize to the tray on close + - bSizer101 - wxHORIZONTAL - none - - 5 - - 0 - - 0 - protected - 16 - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - 0 - - 1 - - - 0 - wxID_ANY - Mi&nimize to system tray on close - - - m_checkBoxMinimizeOnClose - protected - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + m_checkBoxMinimizeOnClose + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/util.cpp b/util.cpp index 4a5b983d..7a947730 100644 --- a/util.cpp +++ b/util.cpp @@ -85,14 +85,14 @@ void RandAddSeed() void RandAddSeedPerfmon() { -#ifdef __WXMSW__ - // Don't need this on Linux, OpenSSL automatically uses /dev/urandom // This can take up to 2 seconds, so only do it every 10 minutes static int64 nLastPerfmon; if (GetTime() < nLastPerfmon + 10 * 60) return; nLastPerfmon = GetTime(); +#ifdef __WXMSW__ + // Don't need this on Linux, OpenSSL automatically uses /dev/urandom // Seed with the entire set of perfmon data unsigned char pdata[250000]; memset(pdata, 0, sizeof(pdata)); @@ -109,9 +109,30 @@ void RandAddSeedPerfmon() printf("%s RandAddSeed() %d bytes\n", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str(), nSize); } +#else + printf("%s RandAddSeed()\n", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); #endif } +uint64 GetRand(uint64 nMax) +{ + if (nMax == 0) + return 0; + + // The range of the random source must be a multiple of the modulus + // to give every possible output value an equal possibility + uint64 nRange = (UINT64_MAX / nMax) * nMax; + uint64 nRand = 0; + do + RAND_bytes((unsigned char*)&nRand, sizeof(nRand)); + while (nRand >= nRange); + return (nRand % nMax); +} + + + + + @@ -449,28 +470,6 @@ string GetDataDir() -uint64 GetRand(uint64 nMax) -{ - if (nMax == 0) - return 0; - - // The range of the random source must be a multiple of the modulus - // to give every possible output value an equal possibility - uint64 nRange = (_UI64_MAX / nMax) * nMax; - uint64 nRand = 0; - do - RAND_bytes((unsigned char*)&nRand, sizeof(nRand)); - while (nRand >= nRange); - return (nRand % nMax); -} - - - - - - - - // @@ -483,7 +482,6 @@ uint64 GetRand(uint64 nMax) // note: NTP isn't implemented yet, so until then we just use the median // of other nodes clocks to correct ours. // - int64 GetTime() { return time(NULL); diff --git a/util.h b/util.h index 8fcfcd0d..ddac4494 100644 --- a/util.h +++ b/util.h @@ -54,9 +54,13 @@ inline T& REF(const T& val) return (T&)val; } -#ifndef __WXMSW__ -#define _UI64_MAX UINT64_MAX -#define _I64_MAX INT64_MAX +#ifdef __WXMSW__ +#define MSG_NOSIGNAL 0 +#define MSG_DONTWAIT 0 +#define UINT64_MAX _UI64_MAX +#define INT64_MAX _I64_MAX +#define INT64_MIN _I64_MIN +#else #define WSAGetLastError() errno #define WSAEWOULDBLOCK EWOULDBLOCK #define WSAEMSGSIZE EMSGSIZE @@ -74,18 +78,6 @@ typedef u_int SOCKET; #define MAX_PATH 1024 #define Sleep(n) wxMilliSleep(n) #define Beep(n1,n2) (0) -inline int _beginthread(void(*pfn)(void*), unsigned nStack, void* parg) { thread(bind(pfn, parg)); return 0; } -inline void _endthread() { pthread_exit(NULL); } -inline int GetCurrentThread() { return 0; } -// threads are processes on linux, so setpriority affects just the one thread -inline void SetThreadPriority(int nThread, int nPriority) { setpriority(PRIO_PROCESS, getpid(), nPriority); } -#define THREAD_PRIORITY_LOWEST PRIO_MIN -#define THREAD_PRIORITY_BELOW_NORMAL 2 -#define THREAD_PRIORITY_NORMAL 0 -#define THREAD_PRIORITY_ABOVE_NORMAL 0 -#endif -#ifndef MSG_NOSIGNAL -#define MSG_NOSIGNAL 0 #endif @@ -133,6 +125,7 @@ void AddTimeData(unsigned int ip, int64 nTime); + // Wrapper to automatically initialize critical sections class CCriticalSection { @@ -201,8 +194,6 @@ public: - - inline int OutputDebugStringF(const char* pszFormat, ...) { int ret = 0; @@ -498,3 +489,83 @@ inline uint160 Hash160(const vector& vch) RIPEMD160((unsigned char*)&hash1, sizeof(hash1), (unsigned char*)&hash2); return hash2; } + + + + + + + + + + + +// Note: It turns out we might have been able to use boost::thread +// by using TerminateThread(boost::thread.native_handle(), 0); +#ifdef __WXMSW__ +typedef HANDLE pthread_t; + +inline pthread_t CreateThread(void(*pfn)(void*), void* parg, bool fWantHandle=false) +{ + DWORD nUnused = 0; + HANDLE hthread = + CreateThread( + NULL, // default security + 0, // inherit stack size from parent + (LPTHREAD_START_ROUTINE)pfn, // function pointer + parg, // argument + 0, // creation option, start immediately + &nUnused); // thread identifier + if (hthread == NULL) + { + printf("Error: CreateThread() returned %d\n", GetLastError()); + return (pthread_t)0; + } + if (!fWantHandle) + { + CloseHandle(hthread); + return (pthread_t)-1; + } + return hthread; +} + +inline void SetThreadPriority(int nPriority) +{ + SetThreadPriority(GetCurrentThread(), nPriority); +} +#else +inline pthread_t CreateThread(void(*pfn)(void*), void* parg, bool fWantHandle=false) +{ + pthread_t hthread = 0; + int ret = pthread_create(&hthread, NULL, (void*(*)(void*))pfn, parg); + if (ret != 0) + { + printf("Error: pthread_create() returned %d\n", ret); + return (pthread_t)0; + } + if (!fWantHandle) + return (pthread_t)-1; + return hthread; +} + +#define THREAD_PRIORITY_LOWEST PRIO_MIN +#define THREAD_PRIORITY_BELOW_NORMAL 2 +#define THREAD_PRIORITY_NORMAL 0 +#define THREAD_PRIORITY_ABOVE_NORMAL 0 + +inline void SetThreadPriority(int nPriority) +{ + // threads are processes on linux, so PRIO_PROCESS affects just the one thread + setpriority(PRIO_PROCESS, getpid(), nPriority); +} + +inline bool TerminateThread(pthread_t hthread, unsigned int nExitCode) +{ + return (pthread_cancel(hthread) == 0); +} + +inline void ExitThread(unsigned int nExitCode) +{ + pthread_exit((void*)nExitCode); +} +#endif From b349e3dca89dfff5fe70669f9fd3f3a1ec60a498 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sat, 14 Nov 2009 18:04:08 +0000 Subject: [PATCH 028/133] misc compile error git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@39 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- net.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/net.h b/net.h index 6300d3ef..65bbb960 100644 --- a/net.h +++ b/net.h @@ -550,7 +550,7 @@ public: CAddress addrYou = (fUseProxy ? CAddress("0.0.0.0") : addr); CAddress addrMe = (fUseProxy ? CAddress("0.0.0.0") : addrLocalHost); RAND_bytes((unsigned char*)&nLocalHostNonce, sizeof(nLocalHostNonce)); - PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce, "linux-test5"); + PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce, string("test5")); } ~CNode() From c5c7911dab8732861affbe66849a100da62f7464 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 18 Nov 2009 19:19:41 +0000 Subject: [PATCH 029/133] bugfix Db::open/close and zombie sockets bugs fix double-close of socket handle, keep databases open, close db cursors, initial block download in batches of 500 blocks, fix misc warnings, subver linux-test8 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@40 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-unix.txt | 16 ++--- db.cpp | 130 ++++++++++++++++++++++++++++------------- db.h | 17 ++++-- irc.cpp | 14 +++-- main.cpp | 87 +++++++++++++++++---------- main.h | 10 ++-- makefile.unix | 7 +-- net.cpp | 110 +++++++++++++++++++--------------- net.h | 16 +++-- serialize.h | 1 + ui.cpp | 27 ++++----- util.cpp | 10 ++-- util.h | 23 +++++++- xpm/addressbook16.xpm | 2 +- xpm/addressbook20.xpm | 2 +- xpm/bitcoin16.xpm | 2 +- xpm/bitcoin20.xpm | 2 +- xpm/bitcoin32.xpm | 2 +- xpm/bitcoin48.xpm | 2 +- xpm/check.xpm | 2 +- xpm/send16.xpm | 2 +- xpm/send16noshadow.xpm | 2 +- xpm/send20.xpm | 2 +- 23 files changed, 304 insertions(+), 184 deletions(-) diff --git a/build-unix.txt b/build-unix.txt index ceb61ad5..f863b6a9 100644 --- a/build-unix.txt +++ b/build-unix.txt @@ -13,18 +13,18 @@ UNIX BUILD NOTES Dependencies ------------ -Install the dev files for the shared libraries: apt-get install build-essential apt-get install libgtk2.0-dev apt-get install libssl-dev +apt-get install libdb4.7-dev +apt-get install libdb4.7++-dev +apt-get install libboost-dev Libraries you need to obtain separately and build: default path download wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ -Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html -Boost \boost http://www.boost.org/users/download/ -Their licenses: +Licenses: wxWidgets LGPL 2.1 with very liberal exceptions Berkeley DB New BSD license with additional requirement that linked software must be free open source Boost MIT-like license @@ -59,15 +59,9 @@ make install ldconfig -Berkeley DB ------------ -cd /usr/local/db-4.7.25.NC/build_unix -../dist/configure --enable-cxx -make - - Boost ----- +If you download and build Boost yourself cd /usr/local/boost_1_40_0 su ./bootstrap.sh diff --git a/db.cpp b/db.cpp index a9c42880..947aed29 100644 --- a/db.cpp +++ b/db.cpp @@ -20,6 +20,7 @@ static CCriticalSection cs_db; static bool fDbEnvInit = false; DbEnv dbenv(0); static map mapFileUseCount; +static map mapDb; class CDBInit { @@ -39,21 +40,17 @@ public: instance_of_cdbinit; -CDB::CDB(const char* pszFile, const char* pszMode, bool fTxn) : pdb(NULL) +CDB::CDB(const char* pszFile, const char* pszMode) : pdb(NULL) { int ret; if (pszFile == NULL) return; + fReadOnly = (!strchr(pszMode, '+') && !strchr(pszMode, 'w')); bool fCreate = strchr(pszMode, 'c'); - bool fReadOnly = (!strchr(pszMode, '+') && !strchr(pszMode, 'w')); unsigned int nFlags = DB_THREAD; if (fCreate) nFlags |= DB_CREATE; - else if (fReadOnly) - nFlags |= DB_RDONLY; - if (!fReadOnly || fTxn) - nFlags |= DB_AUTO_COMMIT; CRITICAL_BLOCK(cs_db) { @@ -72,7 +69,7 @@ CDB::CDB(const char* pszFile, const char* pszMode, bool fTxn) : pdb(NULL) dbenv.set_lk_max_locks(10000); dbenv.set_lk_max_objects(10000); dbenv.set_errfile(fopen(strErrorFile.c_str(), "a")); /// debug - ///dbenv.log_set_config(DB_LOG_AUTO_REMOVE, 1); /// causes corruption + dbenv.set_flags(DB_AUTO_COMMIT, 1); ret = dbenv.open(strDataDir.c_str(), DB_CREATE | DB_INIT_LOCK | @@ -90,31 +87,39 @@ CDB::CDB(const char* pszFile, const char* pszMode, bool fTxn) : pdb(NULL) strFile = pszFile; ++mapFileUseCount[strFile]; - } - - pdb = new Db(&dbenv, 0); + pdb = mapDb[strFile]; + if (pdb == NULL) + { + pdb = new Db(&dbenv, 0); - ret = pdb->open(NULL, // Txn pointer - pszFile, // Filename - "main", // Logical db name - DB_BTREE, // Database type - nFlags, // Flags - 0); + ret = pdb->open(NULL, // Txn pointer + pszFile, // Filename + "main", // Logical db name + DB_BTREE, // Database type + nFlags, // Flags + 0); - if (ret > 0) - { - delete pdb; - pdb = NULL; - CRITICAL_BLOCK(cs_db) - --mapFileUseCount[strFile]; - strFile = ""; - throw runtime_error(strprintf("CDB() : can't open database file %s, error %d\n", pszFile, ret)); - } + if (ret > 0) + { + delete pdb; + pdb = NULL; + CRITICAL_BLOCK(cs_db) + --mapFileUseCount[strFile]; + strFile = ""; + throw runtime_error(strprintf("CDB() : can't open database file %s, error %d\n", pszFile, ret)); + } - if (fCreate && !Exists(string("version"))) - WriteVersion(VERSION); + if (fCreate && !Exists(string("version"))) + { + bool fTmp = fReadOnly; + fReadOnly = false; + WriteVersion(VERSION); + fReadOnly = fTmp; + } - RandAddSeed(); + mapDb[strFile] = pdb; + } + } } void CDB::Close() @@ -124,8 +129,6 @@ void CDB::Close() if (!vTxn.empty()) vTxn.front()->abort(); vTxn.clear(); - pdb->close(0); - delete pdb; pdb = NULL; dbenv.txn_checkpoint(0, 0, 0); @@ -135,6 +138,21 @@ void CDB::Close() RandAddSeed(); } +void CloseDb(const string& strFile) +{ + CRITICAL_BLOCK(cs_db) + { + if (mapDb[strFile] != NULL) + { + // Close the database handle + Db* pdb = mapDb[strFile]; + pdb->close(0); + delete pdb; + mapDb[strFile] = NULL; + } + } +} + void DBFlush(bool fShutdown) { // Flush log data to the actual data file @@ -144,14 +162,18 @@ void DBFlush(bool fShutdown) return; CRITICAL_BLOCK(cs_db) { - dbenv.txn_checkpoint(0, 0, 0); map::iterator mi = mapFileUseCount.begin(); while (mi != mapFileUseCount.end()) { string strFile = (*mi).first; int nRefCount = (*mi).second; + printf("%s refcount=%d\n", strFile.c_str(), nRefCount); if (nRefCount == 0) { + // Move log data to the dat file + CloseDb(strFile); + dbenv.txn_checkpoint(0, 0, 0); + printf("%s flush\n", strFile.c_str()); dbenv.lsn_reset(strFile.c_str(), 0); mapFileUseCount.erase(mi++); } @@ -238,7 +260,10 @@ bool CTxDB::ReadOwnerTxes(uint160 hash160, int nMinHeight, vector& if (ret == DB_NOTFOUND) break; else if (ret != 0) + { + pcursor->close(); return false; + } // Unserialize string strType; @@ -255,9 +280,14 @@ bool CTxDB::ReadOwnerTxes(uint160 hash160, int nMinHeight, vector& { vtx.resize(vtx.size()+1); if (!vtx.back().ReadFromDisk(pos)) + { + pcursor->close(); return false; + } } } + + pcursor->close(); return true; } @@ -379,6 +409,7 @@ bool CTxDB::LoadBlockIndex() break; } } + pcursor->close(); if (!ReadHashBestChain(hashBestChain)) { @@ -391,7 +422,7 @@ bool CTxDB::LoadBlockIndex() return error("CTxDB::LoadBlockIndex() : blockindex for hashBestChain not found"); pindexBest = mapBlockIndex[hashBestChain]; nBestHeight = pindexBest->nHeight; - printf("LoadBlockIndex(): hashBestChain=%s height=%d\n", hashBestChain.ToString().substr(0,14).c_str(), nBestHeight); + printf("LoadBlockIndex(): hashBestChain=%s height=%d\n", hashBestChain.ToString().substr(0,16).c_str(), nBestHeight); return true; } @@ -456,6 +487,7 @@ bool CAddrDB::LoadAddresses() mapAddresses.insert(make_pair(addr.GetKey(), addr)); } } + pcursor->close(); printf("Loaded %d addresses\n", mapAddresses.size()); @@ -558,7 +590,7 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) //printf(" %12I64d %s %s %s\n", // wtx.vout[0].nValue, // DateTimeStrFormat("%x %H:%M:%S", wtx.nTime).c_str(), - // wtx.hashBlock.ToString().substr(0,14).c_str(), + // wtx.hashBlock.ToString().substr(0,16).c_str(), // wtx.mapValue["message"].c_str()); } else if (strType == "key") @@ -596,6 +628,7 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) } } + pcursor->close(); } printf("fShowGenerated = %d\n", fShowGenerated); @@ -655,6 +688,8 @@ void ThreadFlushWalletDB(void* parg) if (fOneThread) return; fOneThread = true; + if (mapArgs.count("-noflushwallet")) + return; unsigned int nLastSeen = nWalletDBUpdated; unsigned int nLastFlushed = nWalletDBUpdated; @@ -669,24 +704,37 @@ void ThreadFlushWalletDB(void* parg) nLastWalletUpdate = GetTime(); } - if (nLastFlushed != nWalletDBUpdated && nLastWalletUpdate < GetTime() - 1) + if (nLastFlushed != nWalletDBUpdated && GetTime() - nLastWalletUpdate >= 2) { TRY_CRITICAL_BLOCK(cs_db) { - string strFile = "wallet.dat"; - map::iterator mi = mapFileUseCount.find(strFile); - if (mi != mapFileUseCount.end()) + // Don't do this if any databases are in use + int nRefCount = 0; + map::iterator mi = mapFileUseCount.begin(); + while (mi != mapFileUseCount.end()) { - int nRefCount = (*mi).second; - if (nRefCount == 0 && !fShutdown) + nRefCount += (*mi).second; + mi++; + } + + if (nRefCount == 0 && !fShutdown) + { + string strFile = "wallet.dat"; + map::iterator mi = mapFileUseCount.find(strFile); + if (mi != mapFileUseCount.end()) { - // Flush wallet.dat so it's self contained + printf("%s ", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); + printf("Flushing wallet.dat\n"); nLastFlushed = nWalletDBUpdated; int64 nStart = GetTimeMillis(); + + // Flush wallet.dat so it's self contained + CloseDb(strFile); dbenv.txn_checkpoint(0, 0, 0); dbenv.lsn_reset(strFile.c_str(), 0); - printf("Flushed wallet.dat %"PRI64d"ms\n", GetTimeMillis() - nStart); + mapFileUseCount.erase(mi++); + printf("Flushed wallet.dat %"PRI64d"ms\n", GetTimeMillis() - nStart); } } } diff --git a/db.h b/db.h index d11b397e..d32d2d5d 100644 --- a/db.h +++ b/db.h @@ -32,8 +32,9 @@ protected: Db* pdb; string strFile; vector vTxn; + bool fReadOnly; - explicit CDB(const char* pszFile, const char* pszMode="r+", bool fTxn=false); + explicit CDB(const char* pszFile, const char* pszMode="r+"); ~CDB() { Close(); } public: void Close(); @@ -77,6 +78,8 @@ protected: { if (!pdb) return false; + if (fReadOnly) + assert(("Write called on database in read-only mode", false)); // Key CDataStream ssKey(SER_DISK); @@ -104,6 +107,8 @@ protected: { if (!pdb) return false; + if (fReadOnly) + assert(("Erase called on database in read-only mode", false)); // Key CDataStream ssKey(SER_DISK); @@ -254,7 +259,7 @@ public: class CTxDB : public CDB { public: - CTxDB(const char* pszMode="r+", bool fTxn=false) : CDB(!fClient ? "blkindex.dat" : NULL, pszMode, fTxn) { } + CTxDB(const char* pszMode="r+") : CDB(!fClient ? "blkindex.dat" : NULL, pszMode) { } private: CTxDB(const CTxDB&); void operator=(const CTxDB&); @@ -283,7 +288,7 @@ public: class CReviewDB : public CDB { public: - CReviewDB(const char* pszMode="r+", bool fTxn=false) : CDB("reviews.dat", pszMode, fTxn) { } + CReviewDB(const char* pszMode="r+") : CDB("reviews.dat", pszMode) { } private: CReviewDB(const CReviewDB&); void operator=(const CReviewDB&); @@ -309,7 +314,7 @@ public: class CMarketDB : public CDB { public: - CMarketDB(const char* pszMode="r+", bool fTxn=false) : CDB("market.dat", pszMode, fTxn) { } + CMarketDB(const char* pszMode="r+") : CDB("market.dat", pszMode) { } private: CMarketDB(const CMarketDB&); void operator=(const CMarketDB&); @@ -322,7 +327,7 @@ private: class CAddrDB : public CDB { public: - CAddrDB(const char* pszMode="r+", bool fTxn=false) : CDB("addr.dat", pszMode, fTxn) { } + CAddrDB(const char* pszMode="r+") : CDB("addr.dat", pszMode) { } private: CAddrDB(const CAddrDB&); void operator=(const CAddrDB&); @@ -341,7 +346,7 @@ bool LoadAddresses(); class CWalletDB : public CDB { public: - CWalletDB(const char* pszMode="r+", bool fTxn=false) : CDB("wallet.dat", pszMode, fTxn) { } + CWalletDB(const char* pszMode="r+") : CDB("wallet.dat", pszMode) { } private: CWalletDB(const CWalletDB&); void operator=(const CWalletDB&); diff --git a/irc.cpp b/irc.cpp index 8432c6d1..8ac38380 100644 --- a/irc.cpp +++ b/irc.cpp @@ -159,15 +159,12 @@ void ThreadIRCSeed(void* parg) SetThreadPriority(THREAD_PRIORITY_NORMAL); int nErrorWait = 10; int nRetryWait = 10; - - // IRC server blocks TOR users - if (fUseProxy && addrProxy.port == htons(9050)) - return; + bool fTOR = (fUseProxy && addrProxy.port == htons(9050)); while (!fShutdown) { CAddress addrConnect("216.155.130.130:6667"); - if (!(fUseProxy && addrProxy.port == htons(9050))) + if (!fTOR) { struct hostent* phostent = gethostbyname("chat.freenode.net"); if (phostent && phostent->h_addr_list && phostent->h_addr_list[0]) @@ -188,6 +185,7 @@ void ThreadIRCSeed(void* parg) if (!RecvUntil(hSocket, "Found your hostname", "using your IP address instead", "Couldn't look up your hostname")) { closesocket(hSocket); + hSocket = INVALID_SOCKET; nErrorWait = nErrorWait * 11 / 10; if (Wait(nErrorWait += 60)) continue; @@ -208,6 +206,7 @@ void ThreadIRCSeed(void* parg) if (!RecvUntil(hSocket, " 004 ")) { closesocket(hSocket); + hSocket = INVALID_SOCKET; nErrorWait = nErrorWait * 11 / 10; if (Wait(nErrorWait += 60)) continue; @@ -269,6 +268,11 @@ void ThreadIRCSeed(void* parg) } } closesocket(hSocket); + hSocket = INVALID_SOCKET; + + // IRC usually blocks TOR, so only try once + if (fTOR) + return; if (GetTime() - nStart > 20 * 60) { diff --git a/main.cpp b/main.cpp index 13a9f9b7..89b42f76 100644 --- a/main.cpp +++ b/main.cpp @@ -760,7 +760,7 @@ unsigned int GetNextWorkRequired(const CBlockIndex* pindexLast) bnNew = bnProofOfWorkLimit; /// debug print - printf("\n\n\nGetNextWorkRequired RETARGET *****\n"); + printf("GetNextWorkRequired RETARGET\n"); printf("nTargetTimespan = %d nActualTimespan = %d\n", nTargetTimespan, nActualTimespan); printf("Before: %08x %s\n", pindexLast->nBits, CBigNum().SetCompact(pindexLast->nBits).getuint256().ToString().c_str()); printf("After: %08x %s\n", bnNew.GetCompact(), bnNew.getuint256().ToString().c_str()); @@ -1013,7 +1013,7 @@ bool CBlock::ConnectBlock(CTxDB& txdb, CBlockIndex* pindex) bool Reorganize(CTxDB& txdb, CBlockIndex* pindexNew) { - printf("*** REORGANIZE ***\n"); + printf("REORGANIZE\n"); // Find the fork CBlockIndex* pfork = pindexBest; @@ -1114,7 +1114,7 @@ bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) // Check for duplicate uint256 hash = GetHash(); if (mapBlockIndex.count(hash)) - return error("AddToBlockIndex() : %s already exists", hash.ToString().substr(0,14).c_str()); + return error("AddToBlockIndex() : %s already exists", hash.ToString().substr(0,16).c_str()); // Construct new block index object CBlockIndex* pindexNew = new CBlockIndex(nFile, nBlockPos, *this); @@ -1174,7 +1174,7 @@ bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) pindexBest = pindexNew; nBestHeight = pindexBest->nHeight; nTransactionsUpdated++; - printf("AddToBlockIndex: new best=%s height=%d\n", hashBestChain.ToString().substr(0,14).c_str(), nBestHeight); + printf("AddToBlockIndex: new best=%s height=%d\n", hashBestChain.ToString().substr(0,16).c_str(), nBestHeight); } txdb.TxnCommit(); @@ -1294,9 +1294,9 @@ bool ProcessBlock(CNode* pfrom, CBlock* pblock) // Check for duplicate uint256 hash = pblock->GetHash(); if (mapBlockIndex.count(hash)) - return error("ProcessBlock() : already have block %d %s", mapBlockIndex[hash]->nHeight, hash.ToString().substr(0,14).c_str()); + return error("ProcessBlock() : already have block %d %s", mapBlockIndex[hash]->nHeight, hash.ToString().substr(0,16).c_str()); if (mapOrphanBlocks.count(hash)) - return error("ProcessBlock() : already have block (orphan) %s", hash.ToString().substr(0,14).c_str()); + return error("ProcessBlock() : already have block (orphan) %s", hash.ToString().substr(0,16).c_str()); // Preliminary checks if (!pblock->CheckBlock()) @@ -1308,7 +1308,7 @@ bool ProcessBlock(CNode* pfrom, CBlock* pblock) // If don't already have its previous block, shunt it off to holding area until we get it if (!mapBlockIndex.count(pblock->hashPrevBlock)) { - printf("ProcessBlock: ORPHAN BLOCK, prev=%s\n", pblock->hashPrevBlock.ToString().substr(0,14).c_str()); + printf("ProcessBlock: ORPHAN BLOCK, prev=%s\n", pblock->hashPrevBlock.ToString().substr(0,16).c_str()); mapOrphanBlocks.insert(make_pair(hash, pblock)); mapOrphanBlocksByPrev.insert(make_pair(pblock->hashPrevBlock, pblock)); @@ -1503,11 +1503,11 @@ bool LoadBlockIndex(bool fAllowNew) // vMerkleTree: 4a5e1e // Genesis block - char* pszTimestamp = "The Times 03/Jan/2009 Chancellor on brink of second bailout for banks"; + const char* pszTimestamp = "The Times 03/Jan/2009 Chancellor on brink of second bailout for banks"; CTransaction txNew; txNew.vin.resize(1); txNew.vout.resize(1); - txNew.vin[0].scriptSig = CScript() << 486604799 << CBigNum(4) << vector((unsigned char*)pszTimestamp, (unsigned char*)pszTimestamp + strlen(pszTimestamp)); + txNew.vin[0].scriptSig = CScript() << 486604799 << CBigNum(4) << vector((const unsigned char*)pszTimestamp, (const unsigned char*)pszTimestamp + strlen(pszTimestamp)); txNew.vout[0].nValue = 50 * COIN; txNew.vout[0].scriptPubKey = CScript() << CBigNum("0x5F1DF16B2B704C8A578D0BBAF74D385CDE12C11EE50455F3C438EF4C3FBCF649B6DE611FEAE06279A60939E028A8D65C10B73071A6F16719274855FEB0FD8A6704") << OP_CHECKSIG; CBlock block; @@ -1519,7 +1519,7 @@ bool LoadBlockIndex(bool fAllowNew) block.nBits = 0x1d00ffff; block.nNonce = 2083236893; - //// debug print, delete this later + //// debug print printf("%s\n", block.GetHash().ToString().c_str()); printf("%s\n", block.hashMerkleRoot.ToString().c_str()); printf("%s\n", hashGenesisBlock.ToString().c_str()); @@ -1592,7 +1592,7 @@ void PrintBlockTree() pindex->nHeight, pindex->nFile, pindex->nBlockPos, - block.GetHash().ToString().substr(0,14).c_str(), + block.GetHash().ToString().substr(0,16).c_str(), DateTimeStrFormat("%x %H:%M:%S", block.nTime).c_str(), block.vtx.size()); @@ -1912,6 +1912,18 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) CBlock block; block.ReadFromDisk((*mi).second, !pfrom->fClient); pfrom->PushMessage("block", block); + + // Trigger them to send a getblocks request for the next batch of inventory + if (inv.hash == pfrom->hashContinue) + { + // Bypass PushInventory, this must send even if redundant, + // and we want it right after the last block so they don't + // wait for other stuff first. + vector vInv; + vInv.push_back(CInv(MSG_BLOCK, hashBestChain)); + pfrom->PushMessage("inv", vInv); + pfrom->hashContinue = 0; + } } } else if (inv.IsKnownType()) @@ -1948,25 +1960,23 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) // Send the rest of the chain if (pindex) pindex = pindex->pnext; - printf("getblocks %d to %s\n", (pindex ? pindex->nHeight : -1), hashStop.ToString().substr(0,14).c_str()); + printf("getblocks %d to %s\n", (pindex ? pindex->nHeight : -1), hashStop.ToString().substr(0,16).c_str()); + int nLimit = 500; for (; pindex; pindex = pindex->pnext) { if (pindex->GetBlockHash() == hashStop) { - printf(" getblocks stopping at %d %s\n", pindex->nHeight, pindex->GetBlockHash().ToString().substr(0,14).c_str()); + printf(" getblocks stopping at %d %s\n", pindex->nHeight, pindex->GetBlockHash().ToString().substr(0,16).c_str()); break; } - - // Bypass setInventoryKnown in case an inventory message got lost - CRITICAL_BLOCK(pfrom->cs_inventory) + pfrom->PushInventory(CInv(MSG_BLOCK, pindex->GetBlockHash())); + if (--nLimit <= 0) { - CInv inv(MSG_BLOCK, pindex->GetBlockHash()); - // returns true if wasn't already contained in the set - if (pfrom->setInventoryKnown2.insert(inv).second) - { - pfrom->setInventoryKnown.erase(inv); - pfrom->vInventoryToSend.push_back(inv); - } + // When this block is requested, we'll send an inv that'll make them + // getblocks the next batch of inventory. + printf(" getblocks stopping at limit %d %s\n", pindex->nHeight, pindex->GetBlockHash().ToString().substr(0,16).c_str()); + pfrom->hashContinue = pindex->GetBlockHash(); + break; } } } @@ -2049,7 +2059,13 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vRecv >> *pblock; //// debug print - printf("received block:\n"); pblock->print(); + if (false) + { + printf("received block:\n"); + pblock->print(); + } + else + printf("received block %s\n", pblock->GetHash().ToString().substr(0,16).c_str()); CInv inv(MSG_BLOCK, pblock->GetHash()); pfrom->AddInventoryKnown(inv); @@ -2175,9 +2191,13 @@ bool SendMessages(CNode* pto) if (pto->nVersion == 0) return true; + // Keep-alive ping + if (pto->nLastSend && GetTime() - pto->nLastSend > 12 * 60 && pto->vSend.empty()) + pto->PushMessage("ping"); + // Address refresh broadcast static int64 nLastRebroadcast; - if (nLastRebroadcast < GetTime() - 24 * 60 * 60) // every 24 hours + if (GetTime() - nLastRebroadcast > 24 * 60 * 60) // every 24 hours { nLastRebroadcast = GetTime(); CRITICAL_BLOCK(cs_vNodes) @@ -2194,9 +2214,16 @@ bool SendMessages(CNode* pto) } } - // Keep-alive ping - if (pto->nLastSend && GetTime() - pto->nLastSend > 12 * 60 && pto->vSend.empty()) - pto->PushMessage("ping"); + // Clear inventory known periodically in case an inv message was missed, + // although usually they would just get it from another node. + static int64 nLastInventoryKnownClear; + if (GetTime() - nLastInventoryKnownClear > 2 * 60 * 60) // every 2 hours + { + nLastInventoryKnownClear = GetTime(); + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + pnode->setInventoryKnown.clear(); + } // @@ -2243,7 +2270,6 @@ bool SendMessages(CNode* pto) } } pto->vInventoryToSend.clear(); - pto->setInventoryKnown2.clear(); } if (!vInventoryToSend.empty()) pto->PushMessage("inv", vInventoryToSend); @@ -2817,8 +2843,7 @@ bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key) // This is only to keep the database open to defeat the auto-flush for the // duration of this scope. This is the only place where this optimization - // maybe makes sense; please don't do it anywhere else. Keeping databases - // open longer than necessary can create deadlocks. + // maybe makes sense; please don't do it anywhere else. CWalletDB walletdb("r"); // Add the change's private key to wallet diff --git a/main.h b/main.h index 8bb1e19e..79f14c68 100644 --- a/main.h +++ b/main.h @@ -1009,9 +1009,9 @@ public: void print() const { printf("CBlock(hash=%s, ver=%d, hashPrevBlock=%s, hashMerkleRoot=%s, nTime=%u, nBits=%08x, nNonce=%u, vtx=%d)\n", - GetHash().ToString().substr(0,14).c_str(), + GetHash().ToString().substr(0,16).c_str(), nVersion, - hashPrevBlock.ToString().substr(0,14).c_str(), + hashPrevBlock.ToString().substr(0,16).c_str(), hashMerkleRoot.ToString().substr(0,6).c_str(), nTime, nBits, nNonce, vtx.size()); @@ -1159,7 +1159,7 @@ public: return strprintf("CBlockIndex(nprev=%08x, pnext=%08x, nFile=%d, nBlockPos=%-6d nHeight=%d, merkle=%s, hashBlock=%s)", pprev, pnext, nFile, nBlockPos, nHeight, hashMerkleRoot.ToString().substr(0,6).c_str(), - GetBlockHash().ToString().substr(0,14).c_str()); + GetBlockHash().ToString().substr(0,16).c_str()); } void print() const @@ -1229,8 +1229,8 @@ public: str += CBlockIndex::ToString(); str += strprintf("\n hashBlock=%s, hashPrev=%s, hashNext=%s)", GetBlockHash().ToString().c_str(), - hashPrev.ToString().substr(0,14).c_str(), - hashNext.ToString().substr(0,14).c_str()); + hashPrev.ToString().substr(0,16).c_str(), + hashNext.ToString().substr(0,16).c_str()); return str; } diff --git a/makefile.unix b/makefile.unix index 24aa8bfe..c0d0ee1d 100644 --- a/makefile.unix +++ b/makefile.unix @@ -17,24 +17,21 @@ endif INCLUDEPATHS= \ -I"/usr/include" \ - -I"/usr/local/boost_1_40_0" \ - -I"/usr/local/db-4.7.25.NC/build_unix" \ -I"/usr/local/include/wx-2.8" \ -I"/usr/local/lib/wx/include/gtk2-ansi-debug-static-2.8" LIBPATHS= \ -L"/usr/lib" \ -L"/usr/local/lib" \ - -L"/usr/local/db-4.7.25.NC/build_unix" LIBS= \ - -Wl,-Bstatic -l boost_thread -l boost_system -l boost_filesystem -Wl,-Bdynamic \ + -Wl,-Bstatic -l boost_system -l boost_filesystem -Wl,-Bdynamic \ -Wl,-Bstatic -l db_cxx -l wx_gtk2$(D)-2.8 -Wl,-Bdynamic \ -l crypto \ -l gtk-x11-2.0 -l gthread-2.0 -l SM WXDEFS=-D__WXGTK__ -DNOPCH -CFLAGS=-O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h diff --git a/net.cpp b/net.cpp index 71295d5d..9f4060b9 100644 --- a/net.cpp +++ b/net.cpp @@ -148,8 +148,8 @@ bool GetMyExternalIP2(const CAddress& addrConnect, const char* pszGet, const cha bool GetMyExternalIP(unsigned int& ipRet) { CAddress addrConnect; - char* pszGet; - char* pszKeyword; + const char* pszGet; + const char* pszKeyword; if (fUseProxy) return false; @@ -463,14 +463,21 @@ CNode* ConnectNode(CAddress addrConnect, int64 nTimeout) } } -void CNode::DoDisconnect() +void CNode::CloseSocketDisconnect() { - if (fDebug) - printf("%s ", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); - printf("disconnecting node %s\n", addr.ToStringLog().c_str()); - - closesocket(hSocket); + fDisconnect = true; + if (hSocket != INVALID_SOCKET) + { + if (fDebug) + printf("%s ", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); + printf("disconnecting node %s\n", addr.ToStringLog().c_str()); + closesocket(hSocket); + hSocket = INVALID_SOCKET; + } +} +void CNode::Cleanup() +{ // All of a nodes broadcasts and subscriptions are automatically torn down // when it goes down, so a node has to stay up to keep its broadcast going. @@ -540,11 +547,12 @@ void ThreadSocketHandler2(void* parg) // remove from vNodes vNodes.erase(remove(vNodes.begin(), vNodes.end(), pnode), vNodes.end()); - // close socket - pnode->DoDisconnect(); + // close socket and cleanup + pnode->CloseSocketDisconnect(); + pnode->Cleanup(); // hold in disconnected pool until all refs are released - pnode->nReleaseTime = max(pnode->nReleaseTime, GetTime() + 5 * 60); + pnode->nReleaseTime = max(pnode->nReleaseTime, GetTime() + 15 * 60); if (pnode->fNetworkNode || pnode->fInbound) pnode->Release(); vNodesDisconnected.push_back(pnode); @@ -599,6 +607,8 @@ void ThreadSocketHandler2(void* parg) { foreach(CNode* pnode, vNodes) { + if (pnode->hSocket == INVALID_SOCKET || pnode->hSocket < 0) + continue; FD_SET(pnode->hSocket, &fdsetRecv); FD_SET(pnode->hSocket, &fdsetError); hSocketMax = max(hSocketMax, pnode->hSocket); @@ -659,17 +669,22 @@ void ThreadSocketHandler2(void* parg) // vector vNodesCopy; CRITICAL_BLOCK(cs_vNodes) + { vNodesCopy = vNodes; + foreach(CNode* pnode, vNodesCopy) + pnode->AddRef(); + } foreach(CNode* pnode, vNodesCopy) { if (fShutdown) return; - SOCKET hSocket = pnode->hSocket; // // Receive // - if (FD_ISSET(hSocket, &fdsetRecv) || FD_ISSET(hSocket, &fdsetError)) + if (pnode->hSocket == INVALID_SOCKET) + continue; + if (FD_ISSET(pnode->hSocket, &fdsetRecv) || FD_ISSET(pnode->hSocket, &fdsetError)) { TRY_CRITICAL_BLOCK(pnode->cs_vRecv) { @@ -678,7 +693,7 @@ void ThreadSocketHandler2(void* parg) // typical socket buffer is 8K-64K char pchBuf[0x10000]; - int nBytes = recv(hSocket, pchBuf, sizeof(pchBuf), MSG_DONTWAIT); + int nBytes = recv(pnode->hSocket, pchBuf, sizeof(pchBuf), MSG_DONTWAIT); if (nBytes > 0) { vRecv.resize(nPos + nBytes); @@ -690,7 +705,7 @@ void ThreadSocketHandler2(void* parg) // socket closed gracefully if (!pnode->fDisconnect) printf("socket closed\n"); - pnode->fDisconnect = true; + pnode->CloseSocketDisconnect(); } else if (nBytes < 0) { @@ -700,7 +715,7 @@ void ThreadSocketHandler2(void* parg) { if (!pnode->fDisconnect) printf("socket recv error %d\n", nErr); - pnode->fDisconnect = true; + pnode->CloseSocketDisconnect(); } } } @@ -709,14 +724,16 @@ void ThreadSocketHandler2(void* parg) // // Send // - if (FD_ISSET(hSocket, &fdsetSend)) + if (pnode->hSocket == INVALID_SOCKET) + continue; + if (FD_ISSET(pnode->hSocket, &fdsetSend)) { TRY_CRITICAL_BLOCK(pnode->cs_vSend) { CDataStream& vSend = pnode->vSend; if (!vSend.empty()) { - int nBytes = send(hSocket, &vSend[0], vSend.size(), MSG_NOSIGNAL | MSG_DONTWAIT); + int nBytes = send(pnode->hSocket, &vSend[0], vSend.size(), MSG_NOSIGNAL | MSG_DONTWAIT); if (nBytes > 0) { vSend.erase(vSend.begin(), vSend.begin() + nBytes); @@ -729,7 +746,7 @@ void ThreadSocketHandler2(void* parg) if (nErr != WSAEWOULDBLOCK && nErr != WSAEMSGSIZE && nErr != WSAEINTR && nErr != WSAEINPROGRESS) { printf("socket send error %d\n", nErr); - pnode->fDisconnect = true; + pnode->CloseSocketDisconnect(); } } } @@ -760,18 +777,12 @@ void ThreadSocketHandler2(void* parg) } } } - - - //// debug heartbeat - static int64 nHeartbeat1; - if (GetTime() - nHeartbeat1 >= 5 * 60) + CRITICAL_BLOCK(cs_vNodes) { - printf("%s sendrecv\n", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); - nHeartbeat1 = GetTime(); - fDebug = true; + foreach(CNode* pnode, vNodesCopy) + pnode->Release(); } - nThreadSocketHandlerHeartbeat = GetTime(); Sleep(10); } @@ -812,18 +823,21 @@ void ThreadOpenConnections2(void* parg) printf("ThreadOpenConnections started\n"); // Connect to specific addresses - while (mapArgs.count("-connect")) + if (mapArgs.count("-connect")) { - foreach(string strAddr, mapMultiArgs["-connect"]) + for (int64 nLoop = 0;; nLoop++) { - CAddress addr(strAddr, NODE_NETWORK); - if (addr.IsValid()) - OpenNetworkConnection(addr); - for (int i = 0; i < 10; i++) + foreach(string strAddr, mapMultiArgs["-connect"]) { - Sleep(1000); - if (fShutdown) - return; + CAddress addr(strAddr, NODE_NETWORK); + if (addr.IsValid()) + OpenNetworkConnection(addr); + for (int i = 0; i < 10 && i < nLoop; i++) + { + Sleep(500); + if (fShutdown) + return; + } } } } @@ -837,7 +851,7 @@ void ThreadOpenConnections2(void* parg) if (addr.IsValid()) { OpenNetworkConnection(addr); - Sleep(1000); + Sleep(500); if (fShutdown) return; } @@ -898,7 +912,7 @@ void ThreadOpenConnections2(void* parg) // 30 days 27 hours // 90 days 46 hours // 365 days 93 hours - int64 nDelay = 3600.0 * sqrt(fabs(nSinceLastSeen) / 3600.0) + nRandomizer; + int64 nDelay = (int64)(3600.0 * sqrt(fabs(nSinceLastSeen) / 3600.0) + nRandomizer); // Fast reconnect for one hour after last seen if (nSinceLastSeen < 60 * 60) @@ -1013,11 +1027,13 @@ void ThreadMessageHandler2(void* parg) // Poll the connected nodes for messages vector vNodesCopy; CRITICAL_BLOCK(cs_vNodes) + { vNodesCopy = vNodes; + foreach(CNode* pnode, vNodesCopy) + pnode->AddRef(); + } foreach(CNode* pnode, vNodesCopy) { - pnode->AddRef(); - // Receive messages TRY_CRITICAL_BLOCK(pnode->cs_vRecv) ProcessMessages(pnode); @@ -1029,8 +1045,11 @@ void ThreadMessageHandler2(void* parg) SendMessages(pnode); if (fShutdown) return; - - pnode->Release(); + } + CRITICAL_BLOCK(cs_vNodes) + { + foreach(CNode* pnode, vNodesCopy) + pnode->Release(); } // Wait and allow messages to bunch up @@ -1257,8 +1276,7 @@ void StartNode(void* parg) if (!fGot) { printf("*** closing socket\n"); - closesocket(pnode->hSocket); - pnode->fDisconnect = true; + pnode->CloseSocketDisconnect(); } } } @@ -1292,7 +1310,7 @@ bool StopNode() int64 nStart = GetTime(); while (vnThreadsRunning[0] > 0 || vnThreadsRunning[2] > 0 || vnThreadsRunning[3] > 0) { - if (GetTime() - nStart > 15) + if (GetTime() - nStart > 20) break; Sleep(20); } diff --git a/net.h b/net.h index 65bbb960..7d71be49 100644 --- a/net.h +++ b/net.h @@ -414,7 +414,7 @@ public: string ToString() const { - return strprintf("%s %s", GetCommand(), hash.ToString().substr(0,14).c_str()); + return strprintf("%s %s", GetCommand(), hash.ToString().substr(0,16).c_str()); } void print() const @@ -504,6 +504,7 @@ public: int64 nReleaseTime; map mapRequests; CCriticalSection cs_mapRequests; + uint256 hashContinue; // flood vector vAddrToSend; @@ -512,7 +513,6 @@ public: // inventory based relay set setInventoryKnown; - set setInventoryKnown2; vector vInventoryToSend; CCriticalSection cs_inventory; multimap mapAskFor; @@ -541,6 +541,7 @@ public: fDisconnect = false; nRefCount = 0; nReleaseTime = 0; + hashContinue = 0; fGetAddr = false; vfSubscribe.assign(256, false); @@ -550,13 +551,16 @@ public: CAddress addrYou = (fUseProxy ? CAddress("0.0.0.0") : addr); CAddress addrMe = (fUseProxy ? CAddress("0.0.0.0") : addrLocalHost); RAND_bytes((unsigned char*)&nLocalHostNonce, sizeof(nLocalHostNonce)); - PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce, string("test5")); + PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce, string(pszSubVer)); } ~CNode() { if (hSocket != INVALID_SOCKET) + { closesocket(hSocket); + hSocket = INVALID_SOCKET; + } } private: @@ -570,12 +574,13 @@ public: return max(nRefCount, 0) + (GetTime() < nReleaseTime ? 1 : 0); } - void AddRef(int64 nTimeout=0) + CNode* AddRef(int64 nTimeout=0) { if (nTimeout != 0) nReleaseTime = max(nReleaseTime, GetTime() + nTimeout); else nRefCount++; + return this; } void Release() @@ -899,7 +904,8 @@ public: bool IsSubscribed(unsigned int nChannel); void Subscribe(unsigned int nChannel, unsigned int nHops=0); void CancelSubscribe(unsigned int nChannel); - void DoDisconnect(); + void CloseSocketDisconnect(); + void Cleanup(); }; diff --git a/serialize.h b/serialize.h index 9b20e2a0..aae821b6 100644 --- a/serialize.h +++ b/serialize.h @@ -20,6 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 106; +static const char* pszSubVer = " linux-test8"; diff --git a/ui.cpp b/ui.cpp index 9d7556cb..aaa26adf 100644 --- a/ui.cpp +++ b/ui.cpp @@ -1664,7 +1664,7 @@ void COptionsDialog::OnButtonApply(wxCommandEvent& event) CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) { - m_staticTextVersion->SetLabel(strprintf("version 0.%d.%d Beta", VERSION/100, VERSION%100)); + m_staticTextVersion->SetLabel(strprintf("version 0.%d.%d beta", VERSION/100, VERSION%100)); // Workaround until upgrade to wxWidgets supporting UTF-8 wxString str = m_staticTextMain->GetLabel(); @@ -2030,7 +2030,7 @@ void CSendingDialog::StartTransfer() // We may have connected already for product details if (!Status("Connecting...")) return; - CNode* pnode = ConnectNode(addr, 5 * 60); + CNode* pnode = ConnectNode(addr, 15 * 60); if (!pnode) { Error("Unable to connect"); @@ -2075,14 +2075,6 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) return; } - // Should already be connected - CNode* pnode = ConnectNode(addr, 5 * 60); - if (!pnode) - { - Error("Lost connection"); - return; - } - // Pause to give the user a chance to cancel while (wxDateTime::UNow() < start + wxTimeSpan(0, 0, 0, 2 * 1000)) { @@ -2112,6 +2104,14 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) return; } + // Make sure we're still connected + CNode* pnode = ConnectNode(addr, 2 * 60 * 60); + if (!pnode) + { + Error("Lost connection, transaction cancelled"); + return; + } + // Last chance to cancel Sleep(50); if (!Status()) @@ -3495,12 +3495,14 @@ bool CMyApp::OnInit2() if (mapArgs.count("-debug")) fDebug = true; + if (strstr(pszSubVer, "test")) + fDebug = true; if (mapArgs.count("-printtodebugger")) fPrintToDebugger = true; printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); - printf("Bitcoin version %d, OS version %s\n", VERSION, wxGetOsDescription().mb_str()); + printf("Bitcoin version %d%s, OS version %s\n", VERSION, pszSubVer, wxGetOsDescription().mb_str()); if (mapArgs.count("-loadblockindextest")) { @@ -3843,9 +3845,8 @@ void SetStartOnSystemStartup(bool fAutoStart) CoInitialize(NULL); // Get a pointer to the IShellLink interface. - HRESULT hres = NULL; IShellLink* psl = NULL; - hres = CoCreateInstance(CLSID_ShellLink, NULL, + HRESULT hres = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, reinterpret_cast(&psl)); diff --git a/util.cpp b/util.cpp index 7a947730..305db5ce 100644 --- a/util.cpp +++ b/util.cpp @@ -56,9 +56,11 @@ public: // Close sockets foreach(CNode* pnode, vNodes) - closesocket(pnode->hSocket); - if (closesocket(hListenSocket) == SOCKET_ERROR) - printf("closesocket(hListenSocket) failed with error %d\n", WSAGetLastError()); + if (pnode->hSocket != INVALID_SOCKET) + closesocket(pnode->hSocket); + if (hListenSocket != INVALID_SOCKET) + if (closesocket(hListenSocket) == SOCKET_ERROR) + printf("closesocket(hListenSocket) failed with error %d\n", WSAGetLastError()); #ifdef __WXMSW__ // Shutdown Windows Sockets @@ -348,7 +350,7 @@ void ParseParameters(int argc, char* argv[]) { char psz[10000]; strlcpy(psz, argv[i], sizeof(psz)); - char* pszValue = ""; + char* pszValue = (char*)""; if (strchr(psz, '=')) { pszValue = strchr(psz, '='); diff --git a/util.h b/util.h index ddac4494..9366e66e 100644 --- a/util.h +++ b/util.h @@ -57,9 +57,11 @@ inline T& REF(const T& val) #ifdef __WXMSW__ #define MSG_NOSIGNAL 0 #define MSG_DONTWAIT 0 +#ifndef UINT64_MAX #define UINT64_MAX _UI64_MAX #define INT64_MAX _I64_MAX #define INT64_MIN _I64_MIN +#endif #else #define WSAGetLastError() errno #define WSAEWOULDBLOCK EWOULDBLOCK @@ -67,7 +69,7 @@ inline T& REF(const T& val) #define WSAEINTR EINTR #define WSAEINPROGRESS EINPROGRESS #define WSAEADDRINUSE EADDRINUSE -#define closesocket(s) close(s) +#define WSAENOTSOCK EBADF #define INVALID_SOCKET (SOCKET)(~0) #define SOCKET_ERROR -1 typedef u_int SOCKET; @@ -80,6 +82,23 @@ typedef u_int SOCKET; #define Beep(n1,n2) (0) #endif +inline int myclosesocket(SOCKET& hSocket) +{ + if (hSocket == INVALID_SOCKET) + return WSAENOTSOCK; +#ifdef __WXMSW__ + int ret = closesocket(hSocket); +#else + int ret = close(hSocket); +#endif + hSocket = INVALID_SOCKET; + return ret; +} +#define closesocket(s) myclosesocket(s) + + + + @@ -149,7 +168,7 @@ public: bool TryEnter() { return mutex.TryLock() == wxMUTEX_NO_ERROR; } #endif public: - char* pszFile; + const char* pszFile; int nLine; }; diff --git a/xpm/addressbook16.xpm b/xpm/addressbook16.xpm index 471f700c..e00944ef 100644 --- a/xpm/addressbook16.xpm +++ b/xpm/addressbook16.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * addressbook16_xpm[] = { +static const char * addressbook16_xpm[] = { /* columns rows colors chars-per-pixel */ "16 16 256 2", " c #FFFFFF", diff --git a/xpm/addressbook20.xpm b/xpm/addressbook20.xpm index 48df12d8..7ebd73fb 100644 --- a/xpm/addressbook20.xpm +++ b/xpm/addressbook20.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * addressbook20_xpm[] = { +static const char * addressbook20_xpm[] = { /* columns rows colors chars-per-pixel */ "20 20 256 2", " c #FFFFFF", diff --git a/xpm/bitcoin16.xpm b/xpm/bitcoin16.xpm index 8bec142c..a1397522 100644 --- a/xpm/bitcoin16.xpm +++ b/xpm/bitcoin16.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * bitcoin16_xpm[] = { +static const char * bitcoin16_xpm[] = { /* columns rows colors chars-per-pixel */ "16 16 181 2", " c #775605", diff --git a/xpm/bitcoin20.xpm b/xpm/bitcoin20.xpm index 2dd61a59..93b34ba7 100644 --- a/xpm/bitcoin20.xpm +++ b/xpm/bitcoin20.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * bitcoin20_xpm[] = { +static const char * bitcoin20_xpm[] = { /* columns rows colors chars-per-pixel */ "20 20 200 2", " c #7B5500", diff --git a/xpm/bitcoin32.xpm b/xpm/bitcoin32.xpm index 25da102f..0ac49f61 100644 --- a/xpm/bitcoin32.xpm +++ b/xpm/bitcoin32.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * bitcoin32_xpm[] = { +static const char * bitcoin32_xpm[] = { /* columns rows colors chars-per-pixel */ "32 32 185 2", " c #715103", diff --git a/xpm/bitcoin48.xpm b/xpm/bitcoin48.xpm index 788e855d..bc388bdc 100644 --- a/xpm/bitcoin48.xpm +++ b/xpm/bitcoin48.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * bitcoin48_xpm[] = { +static const char * bitcoin48_xpm[] = { /* columns rows colors chars-per-pixel */ "48 48 224 2", " c #715103", diff --git a/xpm/check.xpm b/xpm/check.xpm index 8f0b9d28..e62b6569 100644 --- a/xpm/check.xpm +++ b/xpm/check.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * check_xpm[] = { +static const char * check_xpm[] = { /* columns rows colors chars-per-pixel */ "32 32 3 1", " c #008000", diff --git a/xpm/send16.xpm b/xpm/send16.xpm index 1eeceb4e..7da44d9c 100644 --- a/xpm/send16.xpm +++ b/xpm/send16.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * send16_xpm[] = { +static const char * send16_xpm[] = { /* columns rows colors chars-per-pixel */ "16 16 256 2", " c #ADF7AD", diff --git a/xpm/send16noshadow.xpm b/xpm/send16noshadow.xpm index d1b482ef..f6cef45e 100644 --- a/xpm/send16noshadow.xpm +++ b/xpm/send16noshadow.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * send16noshadow_xpm[] = { +static const char * send16noshadow_xpm[] = { /* columns rows colors chars-per-pixel */ "16 16 256 2", " c #ADF7AD", diff --git a/xpm/send20.xpm b/xpm/send20.xpm index 597ea146..68e7b137 100644 --- a/xpm/send20.xpm +++ b/xpm/send20.xpm @@ -1,5 +1,5 @@ /* XPM */ -static char * send20_xpm[] = { +static const char * send20_xpm[] = { /* columns rows colors chars-per-pixel */ "20 20 256 2", " c #CEFFCE", From 52f4cb48590a706caf7a492e8d94b85620d5cd33 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Tue, 24 Nov 2009 21:04:50 +0000 Subject: [PATCH 030/133] minor fix to batched initial download in case requester has more than 500 block non-main branch git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@41 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 4 ++-- main.h | 21 +++++++++++++++++++++ serialize.h | 2 +- 3 files changed, 24 insertions(+), 3 deletions(-) diff --git a/main.cpp b/main.cpp index 89b42f76..ae20e08d 100644 --- a/main.cpp +++ b/main.cpp @@ -1960,8 +1960,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) // Send the rest of the chain if (pindex) pindex = pindex->pnext; - printf("getblocks %d to %s\n", (pindex ? pindex->nHeight : -1), hashStop.ToString().substr(0,16).c_str()); - int nLimit = 500; + int nLimit = 500 + locator.GetDistanceBack(); + printf("getblocks %d to %s limit %d\n", (pindex ? pindex->nHeight : -1), hashStop.ToString().substr(0,16).c_str(), nLimit); for (; pindex; pindex = pindex->pnext) { if (pindex->GetBlockHash() == hashStop) diff --git a/main.h b/main.h index 79f14c68..822045a5 100644 --- a/main.h +++ b/main.h @@ -1298,6 +1298,27 @@ public: vHave.push_back(hashGenesisBlock); } + int GetDistanceBack() + { + // Retrace how far back it was in the sender's branch + int nDistance = 0; + int nStep = 1; + foreach(const uint256& hash, vHave) + { + map::iterator mi = mapBlockIndex.find(hash); + if (mi != mapBlockIndex.end()) + { + CBlockIndex* pindex = (*mi).second; + if (pindex->IsInMainChain()) + return nDistance; + } + nDistance += nStep; + if (nDistance > 10) + nStep *= 2; + } + return nDistance; + } + CBlockIndex* GetBlockIndex() { // Find the first block the caller has in the main chain diff --git a/serialize.h b/serialize.h index aae821b6..b2d948df 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 106; -static const char* pszSubVer = " linux-test8"; +static const char* pszSubVer = " linux-test9"; From 107d9e288df8207e83f4273a8dcd631412f89889 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 6 Dec 2009 00:29:09 +0000 Subject: [PATCH 031/133] fix transaction fee bug in CreateTransaction, higher size cutoff for free transactions in GetMinFee git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@42 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 18 ++++++++---------- main.h | 11 ++++++++--- serialize.h | 2 +- 3 files changed, 17 insertions(+), 14 deletions(-) diff --git a/main.cpp b/main.cpp index ae20e08d..6575099f 100644 --- a/main.cpp +++ b/main.cpp @@ -2479,10 +2479,8 @@ void BitcoinMiner() if (tx.IsCoinBase() || !tx.IsFinal()) continue; - // Transaction fee requirements, mainly only needed for flood control - // Under 10K (about 80 inputs) is free for first 100 transactions - // Base rate is 0.01 per KB - int64 nMinFee = tx.GetMinFee(pblock->vtx.size() < 100); + // Transaction fee based on block size + int64 nMinFee = tx.GetMinFee(nBlockSize); map mapTestPoolTmp(mapTestPool); if (!tx.ConnectInputs(txdb, mapTestPoolTmp, CDiskTxPos(1,1,1), 0, nFees, false, true, nMinFee)) @@ -2768,11 +2766,11 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CK if (nValue < 0) return false; int64 nValueOut = nValue; - nValue += nFee; + int64 nTotalValue = nValue + nFee; // Choose coins to use set setCoins; - if (!SelectCoins(nValue, setCoins)) + if (!SelectCoins(nTotalValue, setCoins)) return false; int64 nValueIn = 0; foreach(CWalletTx* pcoin, setCoins) @@ -2784,7 +2782,7 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CK wtxNew.vout.push_back(CTxOut(nValueOut, scriptPubKey)); // Fill a vout back to self with any change - if (nValueIn > nValue) + if (nValueIn > nTotalValue) { // New private key if (keyRet.IsNull()) @@ -2793,7 +2791,7 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CK // Fill a vout to ourself CScript scriptPubKey; scriptPubKey << keyRet.GetPubKey() << OP_CHECKSIG; - wtxNew.vout.push_back(CTxOut(nValueIn - nValue, scriptPubKey)); + wtxNew.vout.push_back(CTxOut(nValueIn - nTotalValue, scriptPubKey)); } // Fill a vout to the payee @@ -2814,9 +2812,9 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CK SignSignature(*pcoin, wtxNew, nIn++); // Check that enough fee is included - if (nFee < wtxNew.GetMinFee(true)) + if (nFee < wtxNew.GetMinFee()) { - nFee = nFeeRequiredRet = wtxNew.GetMinFee(true); + nFee = nFeeRequiredRet = wtxNew.GetMinFee(); continue; } diff --git a/main.h b/main.h index 822045a5..716485e9 100644 --- a/main.h +++ b/main.h @@ -512,14 +512,19 @@ public: return nValueOut; } - int64 GetMinFee(bool fDiscount=false) const + int64 GetMinFee(unsigned int nBlockSize=1) const { // Base fee is 1 cent per kilobyte unsigned int nBytes = ::GetSerializeSize(*this, SER_NETWORK); int64 nMinFee = (1 + (int64)nBytes / 1000) * CENT; - // First 100 transactions in a block are free - if (fDiscount && nBytes < 10000) + // Transactions under 60K are free as long as block size is under 80K + // (about 27,000bc if made of 50bc inputs) + if (nBytes < 60000 && nBlockSize < 80000) + nMinFee = 0; + + // Transactions under 3K are free as long as block size is under 200K + if (nBytes < 3000 && nBlockSize < 200000) nMinFee = 0; // To limit dust spam, require a 0.01 fee if any output is less than 0.01 diff --git a/serialize.h b/serialize.h index b2d948df..14a2bed7 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 106; -static const char* pszSubVer = " linux-test9"; +static const char* pszSubVer = " test10"; From b075bbf9862df41cdf5265026ba787b7d0fecb07 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 6 Dec 2009 00:38:11 +0000 Subject: [PATCH 032/133] misc git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@43 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 14 +++++--------- 1 file changed, 5 insertions(+), 9 deletions(-) diff --git a/main.cpp b/main.cpp index 6575099f..46e74752 100644 --- a/main.cpp +++ b/main.cpp @@ -1254,8 +1254,8 @@ bool CBlock::AcceptBlock() if (nTime <= pindexPrev->GetMedianTimePast()) return error("AcceptBlock() : block's timestamp is too early"); - // Check that all transactions are finalized (starting around Dec 2009) - if (nBestHeight > 31000) + // Check that all transactions are finalized (starting around Mar 2010) + if (nBestHeight > 36000) foreach(const CTransaction& tx, vtx) if (!tx.IsFinal(nTime)) return error("AcceptBlock() : contains a non-final transaction"); @@ -2059,13 +2059,9 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vRecv >> *pblock; //// debug print - if (false) - { - printf("received block:\n"); - pblock->print(); - } - else - printf("received block %s\n", pblock->GetHash().ToString().substr(0,16).c_str()); + // printf("received block:\n"); + // pblock->print(); + printf("received block %s\n", pblock->GetHash().ToString().substr(0,16).c_str()); CInv inv(MSG_BLOCK, pblock->GetHash()); pfrom->AddInventoryKnown(inv); From 4ea3f3da1a0c00ea74e85c31a22ea94d18bbdf06 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 11 Dec 2009 16:49:21 +0000 Subject: [PATCH 033/133] retry IRC if name in use, resize to fit ubuntu's giant default font, scroll debug.log, pause gen during initial block download git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@44 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-msw.txt | 6 +++--- build-unix.txt | 13 ++++++------ db.cpp | 21 ++++++++++++++++--- irc.cpp | 23 ++++++++++++++------- main.cpp | 21 +++++++++++++++---- makefile | 26 +++++++++++------------- makefile.unix | 35 +++++++++++++++++--------------- net.cpp | 25 ++++++++++++++++++++++- net.h | 1 - serialize.h | 2 +- ui.cpp | 46 ++++++++++++++++++++++++++++++++--------- util.cpp | 55 +++++++++++++++++++++++++++++--------------------- util.h | 2 ++ 13 files changed, 187 insertions(+), 89 deletions(-) diff --git a/build-msw.txt b/build-msw.txt index a159a71a..56a38b02 100644 --- a/build-msw.txt +++ b/build-msw.txt @@ -9,7 +9,7 @@ cryptographic software written by Eric Young (eay@cryptsoft.com). WINDOWS BUILD NOTES - +=================== Compilers Supported ------------------- @@ -19,7 +19,7 @@ Microsoft Visual C++ 6.0 SP6 Dependencies ------------ -Libraries you need to obtain separately to build: +Libraries you need to download separately and build: default path download wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ @@ -44,7 +44,7 @@ Boost 1.34.1 Notes ----- -The UI layout is edited with wxFormBuilder. Open the project file +The UI layout is edited with wxFormBuilder. The project file is uiproject.fbp. It generates uibase.cpp and uibase.h, which define base classes that do the rote work of constructing all the UI elements. diff --git a/build-unix.txt b/build-unix.txt index f863b6a9..7c650589 100644 --- a/build-unix.txt +++ b/build-unix.txt @@ -9,7 +9,7 @@ cryptographic software written by Eric Young (eay@cryptsoft.com). UNIX BUILD NOTES - +================ Dependencies ------------ @@ -20,11 +20,10 @@ apt-get install libdb4.7-dev apt-get install libdb4.7++-dev apt-get install libboost-dev -Libraries you need to obtain separately and build: - default path download -wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ +You need to download wxWidgets from http://www.wxwidgets.org/downloads/ +and build it yourself. -Licenses: +Licenses of statically linked libraries: wxWidgets LGPL 2.1 with very liberal exceptions Berkeley DB New BSD license with additional requirement that linked software must be free open source Boost MIT-like license @@ -39,7 +38,7 @@ Boost 1.40.0 Notes ----- -The UI layout is edited with wxFormBuilder. Open the project file +The UI layout is edited with wxFormBuilder. The project file is uiproject.fbp. It generates uibase.cpp and uibase.h, which define base classes that do the rote work of constructing all the UI elements. @@ -61,7 +60,7 @@ ldconfig Boost ----- -If you download and build Boost yourself +If you want to build Boost yourself, cd /usr/local/boost_1_40_0 su ./bootstrap.sh diff --git a/db.cpp b/db.cpp index 947aed29..6fbd4b9c 100644 --- a/db.cpp +++ b/db.cpp @@ -445,7 +445,7 @@ bool CAddrDB::LoadAddresses() CRITICAL_BLOCK(cs_mapAddresses) { // Load user provided addresses - CAutoFile filein = fopen("addr.txt", "rt"); + CAutoFile filein = fopen((GetDataDir() + "/addr.txt").c_str(), "rt"); if (filein) { try @@ -536,10 +536,11 @@ bool CReviewDB::WriteReviews(uint256 hash, const vector& vReviews) bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) { vchDefaultKeyRet.clear(); + int nFileVersion = 0; // Modify defaults #ifndef __WXMSW__ - // Reports that tray icon can disappear on gnome, leaving no way to access the program + // Tray icon sometimes disappears on 9.10 karmic koala 64-bit, leaving no way to access the program fMinimizeToTray = false; fMinimizeOnClose = false; #endif @@ -607,6 +608,10 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) { ssValue >> vchDefaultKeyRet; } + else if (strType == "version") + { + ssValue >> nFileVersion; + } else if (strType == "setting") { string strKey; @@ -649,6 +654,16 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) WriteSetting("nTransactionFee", nTransactionFee); } + // Upgrade + if (nFileVersion < VERSION) + { + // Get rid of old debug.log file in current directory + if (nFileVersion <= 105 && !pszSetDataDir[0]) + unlink("debug.log"); + + WriteVersion(VERSION); + } + return true; } @@ -656,7 +671,7 @@ bool LoadWallet(bool& fFirstRunRet) { fFirstRunRet = false; vector vchDefaultKey; - if (!CWalletDB("cr").LoadWallet(vchDefaultKey)) + if (!CWalletDB("cr+").LoadWallet(vchDefaultKey)) return false; fFirstRunRet = vchDefaultKey.empty(); diff --git a/irc.cpp b/irc.cpp index 8ac38380..8e851085 100644 --- a/irc.cpp +++ b/irc.cpp @@ -121,20 +121,20 @@ bool RecvLineIRC(SOCKET hSocket, string& strLine) } } -bool RecvUntil(SOCKET hSocket, const char* psz1, const char* psz2=NULL, const char* psz3=NULL) +int RecvUntil(SOCKET hSocket, const char* psz1, const char* psz2=NULL, const char* psz3=NULL) { loop { string strLine; if (!RecvLineIRC(hSocket, strLine)) - return false; + return 0; printf("IRC %s\n", strLine.c_str()); if (psz1 && strLine.find(psz1) != -1) - return true; + return 1; if (psz2 && strLine.find(psz2) != -1) - return true; + return 2; if (psz3 && strLine.find(psz3) != -1) - return true; + return 3; } } @@ -159,6 +159,7 @@ void ThreadIRCSeed(void* parg) SetThreadPriority(THREAD_PRIORITY_NORMAL); int nErrorWait = 10; int nRetryWait = 10; + bool fNameInUse = false; bool fTOR = (fUseProxy && addrProxy.port == htons(9050)); while (!fShutdown) @@ -194,7 +195,7 @@ void ThreadIRCSeed(void* parg) } string strMyName; - if (addrLocalHost.IsRoutable() && !fUseProxy) + if (addrLocalHost.IsRoutable() && !fUseProxy && !fNameInUse) strMyName = EncodeAddress(addrLocalHost); else strMyName = strprintf("x%u", GetRand(1000000000)); @@ -203,10 +204,18 @@ void ThreadIRCSeed(void* parg) Send(hSocket, strprintf("NICK %s\r", strMyName.c_str()).c_str()); Send(hSocket, strprintf("USER %s 8 * : %s\r", strMyName.c_str(), strMyName.c_str()).c_str()); - if (!RecvUntil(hSocket, " 004 ")) + int nRet = RecvUntil(hSocket, " 004 ", " 433 "); + if (nRet != 1) { closesocket(hSocket); hSocket = INVALID_SOCKET; + if (nRet == 2) + { + printf("IRC name already in use\n"); + fNameInUse = true; + Wait(10); + continue; + } nErrorWait = nErrorWait * 11 / 10; if (Wait(nErrorWait += 60)) continue; diff --git a/main.cpp b/main.cpp index 46e74752..cb93e548 100644 --- a/main.cpp +++ b/main.cpp @@ -2530,7 +2530,7 @@ void BitcoinMiner() // // Search // - unsigned int nStart = GetTime(); + int64 nStart = GetTime(); uint256 hashTarget = CBigNum().SetCompact(pblock->nBits).getuint256(); uint256 hash; loop @@ -2582,14 +2582,27 @@ void BitcoinMiner() return; if (fLimitProcessors && vnThreadsRunning[3] > nLimitProcessors) return; - if (tmp.block.nNonce == 0) + if (vNodes.empty()) break; - if (pindexPrev != pindexBest) + if (tmp.block.nNonce == 0) break; if (nTransactionsUpdated != nTransactionsUpdatedLast && GetTime() - nStart > 60) break; - if (vNodes.empty()) + if (pindexPrev != pindexBest) + { + // Pause generating during initial download + if (GetTime() - nStart < 20) + { + CBlockIndex* pindexTmp; + do + { + pindexTmp = pindexBest; + Sleep(10000); + } + while (pindexTmp != pindexBest); + } break; + } tmp.block.nTime = pblock->nTime = max(pindexPrev->GetMedianTimePast()+1, GetAdjustedTime()); } } diff --git a/makefile b/makefile index 2d932f02..0dd62210 100644 --- a/makefile +++ b/makefile @@ -10,8 +10,6 @@ endif endif ifeq "$(BUILD)" "debug" D=d -# note: gcc 3.x profile doesn't work -#DEBUGFLAGS=-O0 -g -pg -D__WXDEBUG__ DEBUGFLAGS=-g -D__WXDEBUG__ endif @@ -36,34 +34,34 @@ all: bitcoin.exe headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h g++ -c $(CFLAGS) -o $@ $< -obj/util.o: util.cpp $(HEADERS) +obj/util.o: util.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/script.o: script.cpp $(HEADERS) +obj/script.o: script.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/db.o: db.cpp $(HEADERS) market.h +obj/db.o: db.cpp $(HEADERS) market.h g++ -c $(CFLAGS) -o $@ $< -obj/net.o: net.cpp $(HEADERS) net.h +obj/net.o: net.cpp $(HEADERS) net.h g++ -c $(CFLAGS) -o $@ $< -obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h +obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) market.h +obj/market.o: market.cpp $(HEADERS) market.h g++ -c $(CFLAGS) -o $@ $< -obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h +obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h g++ -c $(CFLAGS) -o $@ $< -obj/uibase.o: uibase.cpp uibase.h +obj/uibase.o: uibase.cpp uibase.h g++ -c $(CFLAGS) -o $@ $< -obj/sha.o: sha.cpp sha.h +obj/sha.o: sha.cpp sha.h g++ -c $(CFLAGS) -O3 -o $@ $< -obj/irc.o: irc.cpp $(HEADERS) +obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp @@ -71,8 +69,8 @@ obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp -OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o +OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o bitcoin.exe: headers.h.gch $(OBJS) -kill /f bitcoin.exe diff --git a/makefile.unix b/makefile.unix index c0d0ee1d..b9826d6c 100644 --- a/makefile.unix +++ b/makefile.unix @@ -22,13 +22,16 @@ INCLUDEPATHS= \ LIBPATHS= \ -L"/usr/lib" \ - -L"/usr/local/lib" \ + -L"/usr/local/lib" LIBS= \ - -Wl,-Bstatic -l boost_system -l boost_filesystem -Wl,-Bdynamic \ - -Wl,-Bstatic -l db_cxx -l wx_gtk2$(D)-2.8 -Wl,-Bdynamic \ - -l crypto \ - -l gtk-x11-2.0 -l gthread-2.0 -l SM + -Wl,-Bstatic \ + -l boost_system -l boost_filesystem \ + -l db_cxx \ + -l wx_gtk2$(D)-2.8 \ + -Wl,-Bdynamic \ + -l crypto \ + -l gtk-x11-2.0 -l gthread-2.0 -l SM WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) @@ -42,41 +45,41 @@ all: bitcoin headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h g++ -c $(CFLAGS) -o $@ $< -obj/util.o: util.cpp $(HEADERS) +obj/util.o: util.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/script.o: script.cpp $(HEADERS) +obj/script.o: script.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/db.o: db.cpp $(HEADERS) market.h +obj/db.o: db.cpp $(HEADERS) market.h g++ -c $(CFLAGS) -o $@ $< -obj/net.o: net.cpp $(HEADERS) net.h +obj/net.o: net.cpp $(HEADERS) net.h g++ -c $(CFLAGS) -o $@ $< -obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h +obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) market.h +obj/market.o: market.cpp $(HEADERS) market.h g++ -c $(CFLAGS) -o $@ $< -obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h +obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h g++ -c $(CFLAGS) -o $@ $< -obj/uibase.o: uibase.cpp uibase.h +obj/uibase.o: uibase.cpp uibase.h g++ -c $(CFLAGS) -o $@ $< -obj/sha.o: sha.cpp sha.h +obj/sha.o: sha.cpp sha.h g++ -c $(CFLAGS) -O3 -o $@ $< -obj/irc.o: irc.cpp $(HEADERS) +obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/net.cpp b/net.cpp index 9f4060b9..436a5abf 100644 --- a/net.cpp +++ b/net.cpp @@ -21,7 +21,6 @@ uint64 nLocalServices = (fClient ? 0 : NODE_NETWORK); CAddress addrLocalHost(0, DEFAULT_PORT, nLocalServices); CNode* pnodeLocalHost = NULL; uint64 nLocalHostNonce = 0; -bool fShutdown = false; array vnThreadsRunning; SOCKET hListenSocket = INVALID_SOCKET; int64 nThreadSocketHandlerHeartbeat = INT64_MAX; @@ -1324,3 +1323,27 @@ bool StopNode() return true; } + +class CNetCleanup +{ +public: + CNetCleanup() + { + } + ~CNetCleanup() + { + // Close sockets + foreach(CNode* pnode, vNodes) + if (pnode->hSocket != INVALID_SOCKET) + closesocket(pnode->hSocket); + if (hListenSocket != INVALID_SOCKET) + if (closesocket(hListenSocket) == SOCKET_ERROR) + printf("closesocket(hListenSocket) failed with error %d\n", WSAGetLastError()); + +#ifdef __WXMSW__ + // Shutdown Windows Sockets + WSACleanup(); +#endif + } +} +instance_of_cnetcleanup; diff --git a/net.h b/net.h index 7d71be49..7fe4a7c7 100644 --- a/net.h +++ b/net.h @@ -454,7 +454,6 @@ extern uint64 nLocalServices; extern CAddress addrLocalHost; extern CNode* pnodeLocalHost; extern uint64 nLocalHostNonce; -extern bool fShutdown; extern array vnThreadsRunning; extern SOCKET hListenSocket; extern int64 nThreadSocketHandlerHeartbeat; diff --git a/serialize.h b/serialize.h index 14a2bed7..63291c6b 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 106; -static const char* pszSubVer = " test10"; +static const char* pszSubVer = " test11"; diff --git a/ui.cpp b/ui.cpp index aaa26adf..262f2a88 100644 --- a/ui.cpp +++ b/ui.cpp @@ -317,6 +317,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) fOnSetFocusAddress = false; fRefresh = false; m_choiceFilter->SetSelection(0); + double dResize = 1.0; #ifdef __WXMSW__ SetIcon(wxICON(bitcoin)); #else @@ -330,6 +331,10 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_toolBar->AddTool(wxID_BUTTONSEND, "Send Coins", wxBitmap(send20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); m_toolBar->AddTool(wxID_BUTTONRECEIVE, "Address Book", wxBitmap(addressbook20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); m_toolBar->Realize(); + // resize to fit ubuntu's huge default font + dResize = 1.19; + SetSize(dResize * GetSize().GetWidth(), 1.1 * GetSize().GetHeight()); + dResize = 1.20; #endif m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); @@ -339,13 +344,13 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) int nDateWidth = DateTimeStr(1229413914).size() * 6 + 8; if (!strstr(DateTimeStr(1229413914).c_str(), "2008")) nDateWidth += 12; - m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, 0); - m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, 0); - m_listCtrl->InsertColumn(2, "Status", wxLIST_FORMAT_LEFT, 90); - m_listCtrl->InsertColumn(3, "Date", wxLIST_FORMAT_LEFT, nDateWidth); - m_listCtrl->InsertColumn(4, "Description", wxLIST_FORMAT_LEFT, 409 - nDateWidth); - m_listCtrl->InsertColumn(5, "Debit", wxLIST_FORMAT_RIGHT, 79); - m_listCtrl->InsertColumn(6, "Credit", wxLIST_FORMAT_RIGHT, 79); + m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, dResize * 0); + m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, dResize * 0); + m_listCtrl->InsertColumn(2, "Status", wxLIST_FORMAT_LEFT, dResize * 90); + m_listCtrl->InsertColumn(3, "Date", wxLIST_FORMAT_LEFT, dResize * nDateWidth); + m_listCtrl->InsertColumn(4, "Description", wxLIST_FORMAT_LEFT, dResize * 409 - nDateWidth); + m_listCtrl->InsertColumn(5, "Debit", wxLIST_FORMAT_RIGHT, dResize * 79); + m_listCtrl->InsertColumn(6, "Credit", wxLIST_FORMAT_RIGHT, dResize * 79); //m_listCtrlProductsSent->InsertColumn(0, "Category", wxLIST_FORMAT_LEFT, 100); //m_listCtrlProductsSent->InsertColumn(1, "Title", wxLIST_FORMAT_LEFT, 100); @@ -367,6 +372,10 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) // Init status bar int pnWidths[3] = { -100, 88, 290 }; +#ifndef __WXMSW__ + pnWidths[1] = pnWidths[1] * 1.1 * dResize; + pnWidths[2] = pnWidths[2] * 1.1 * dResize; +#endif m_statusBar->SetFieldsCount(3, pnWidths); // Fill your address text box @@ -1514,6 +1523,7 @@ COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) SelectPage(0); #ifndef __WXMSW__ m_checkBoxMinimizeOnClose->SetLabel("&Minimize on close"); + m_checkBoxStartOnSystemStartup->Enable(false); // not implemented yet #endif // Init values @@ -1876,6 +1886,9 @@ CSendingDialog::CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 n fSuccess = false; fUIDone = false; fWorkDone = false; +#ifndef __WXMSW__ + SetSize(1.2 * GetSize().GetWidth(), 1.05 * GetSize().GetHeight()); +#endif SetTitle(strprintf("Sending %s to %s", FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); m_textCtrlStatus->SetValue(""); @@ -3475,6 +3488,7 @@ bool CMyApp::OnInit2() ParseParameters(argc, argv); if (mapArgs.count("-?") || mapArgs.count("--help")) { +#ifdef __WXMSW__ string strUsage = "Usage: bitcoin [options]\t\t\t\t\t\t\n" "Options:\n" @@ -3487,6 +3501,20 @@ bool CMyApp::OnInit2() " -connect=\t Connect only to the specified node\n" " -?\t\t This help message\n"; wxMessageBox(strUsage, "Bitcoin", wxOK); +#else + string strUsage = + "Usage: bitcoin [options]\n" + "Options:\n" + " -gen Generate coins\n" + " -gen=0 Don't generate coins\n" + " -min Start minimized\n" + " -datadir= Specify data directory\n" + " -proxy= Connect through socks4 proxy\n" + " -addnode= Add a node to connect to\n" + " -connect= Connect only to the specified node\n" + " -? This help message\n"; + fprintf(stderr, "%s", strUsage.c_str()); +#endif return false; } @@ -3495,12 +3523,12 @@ bool CMyApp::OnInit2() if (mapArgs.count("-debug")) fDebug = true; - if (strstr(pszSubVer, "test")) - fDebug = true; if (mapArgs.count("-printtodebugger")) fPrintToDebugger = true; + if (!fDebug && !pszSetDataDir[0]) + ShrinkDebugFile(); printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); printf("Bitcoin version %d%s, OS version %s\n", VERSION, pszSubVer, wxGetOsDescription().mb_str()); diff --git a/util.cpp b/util.cpp index 305db5ce..f5f2797f 100644 --- a/util.cpp +++ b/util.cpp @@ -11,6 +11,7 @@ bool fDebug = false; bool fPrintToDebugger = false; bool fPrintToConsole = false; char pszSetDataDir[MAX_PATH] = ""; +bool fShutdown = false; @@ -53,19 +54,6 @@ public: for (int i = 0; i < CRYPTO_num_locks(); i++) delete ppmutexOpenSSL[i]; OPENSSL_free(ppmutexOpenSSL); - - // Close sockets - foreach(CNode* pnode, vNodes) - if (pnode->hSocket != INVALID_SOCKET) - closesocket(pnode->hSocket); - if (hListenSocket != INVALID_SOCKET) - if (closesocket(hListenSocket) == SOCKET_ERROR) - printf("closesocket(hListenSocket) failed with error %d\n", WSAGetLastError()); - -#ifdef __WXMSW__ - // Shutdown Windows Sockets - WSACleanup(); -#endif } } instance_of_cinit; @@ -416,16 +404,6 @@ void PrintException(std::exception* pex, const char* pszThread) -int GetFilesize(FILE* file) -{ - int nSavePos = ftell(file); - int nFilesize = -1; - if (fseek(file, 0, SEEK_END) == 0) - nFilesize = ftell(file); - fseek(file, nSavePos, SEEK_SET); - return nFilesize; -} - void GetDataDir(char* pszDir) { // pszDir must be at least MAX_PATH length. @@ -465,6 +443,37 @@ string GetDataDir() return pszDir; } +int GetFilesize(FILE* file) +{ + int nSavePos = ftell(file); + int nFilesize = -1; + if (fseek(file, 0, SEEK_END) == 0) + nFilesize = ftell(file); + fseek(file, nSavePos, SEEK_SET); + return nFilesize; +} + +void ShrinkDebugFile() +{ + // Scroll debug.log if it's getting too big + string strFile = GetDataDir() + "/debug.log"; + FILE* file = fopen(strFile.c_str(), "r"); + if (file && GetFilesize(file) > 10 * 1000000) + { + // Restart the file with some of the end + char pch[200000]; + fseek(file, -sizeof(pch), SEEK_END); + int nBytes = fread(pch, 1, sizeof(pch), file); + fclose(file); + if (file = fopen(strFile.c_str(), "w")) + { + fwrite(pch, 1, nBytes, file); + fclose(file); + } + } +} + + diff --git a/util.h b/util.h index 9366e66e..ae53bfed 100644 --- a/util.h +++ b/util.h @@ -111,6 +111,7 @@ extern bool fDebug; extern bool fPrintToDebugger; extern bool fPrintToConsole; extern char pszSetDataDir[MAX_PATH]; +extern bool fShutdown; void RandAddSeed(); void RandAddSeedPerfmon(); @@ -128,6 +129,7 @@ void ParseParameters(int argc, char* argv[]); int GetFilesize(FILE* file); void GetDataDir(char* pszDirRet); string GetDataDir(); +void ShrinkDebugFile(); uint64 GetRand(uint64 nMax); int64 GetTime(); int64 GetAdjustedTime(); From 651d33556982d038d39f04373b9f63913bf03187 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 11 Dec 2009 17:19:51 +0000 Subject: [PATCH 034/133] misc git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@45 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/db.cpp b/db.cpp index 6fbd4b9c..1d68bef6 100644 --- a/db.cpp +++ b/db.cpp @@ -454,6 +454,7 @@ bool CAddrDB::LoadAddresses() while (fgets(psz, sizeof(psz), filein)) { CAddress addr(psz, NODE_NETWORK); + addr.nTime = 0; // so it won't relay unless successfully connected if (addr.IsValid()) AddAddress(*this, addr); } From e39bc50eb4dd4f0a34ee4c264af83043f13543dd Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 13 Dec 2009 00:13:16 +0000 Subject: [PATCH 035/133] misc exit code, updated setup.nsi git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@46 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 1 + net.cpp | 11 ++++++++--- serialize.h | 4 ++-- setup.nsi | 24 ++++++++++++++++-------- ui.cpp | 18 ++++++++++++++++-- 5 files changed, 43 insertions(+), 15 deletions(-) diff --git a/db.cpp b/db.cpp index 1d68bef6..b21d1e3a 100644 --- a/db.cpp +++ b/db.cpp @@ -637,6 +637,7 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) pcursor->close(); } + printf("nFileVersion = %d\n", nFileVersion); printf("fShowGenerated = %d\n", fShowGenerated); printf("fGenerateBitcoins = %d\n", fGenerateBitcoins); printf("nTransactionFee = %"PRI64d"\n", nTransactionFee); diff --git a/net.cpp b/net.cpp index 436a5abf..941e7c4d 100644 --- a/net.cpp +++ b/net.cpp @@ -929,7 +929,7 @@ void ThreadOpenConnections2(void* parg) // Only try the old stuff if we don't have enough connections if (vNodes.size() >= 2 && nSinceLastSeen > 7 * 24 * 60 * 60) continue; - if (vNodes.size() >= 4 && nSinceLastSeen > 24 * 60 * 60) + if (vNodes.size() >= 5 && nSinceLastSeen > 24 * 60 * 60) continue; // If multiple addresses are ready, prioritize by time since @@ -1256,11 +1256,14 @@ void StartNode(void* parg) // // Thread monitoring + // Not really needed anymore, the cause of the hanging was fixed // loop { - Sleep(15000); - if (GetTime() - nThreadSocketHandlerHeartbeat > 4 * 60) + Sleep(1000); + if (fShutdown) + return; + if (GetTime() - nThreadSocketHandlerHeartbeat > 15 * 60) { // First see if closing sockets will free it printf("*** ThreadSocketHandler is stopped ***\n"); @@ -1280,6 +1283,8 @@ void StartNode(void* parg) } } Sleep(10000); + if (fShutdown) + return; if (GetTime() - nThreadSocketHandlerHeartbeat < 60) continue; diff --git a/serialize.h b/serialize.h index 63291c6b..8f925103 100644 --- a/serialize.h +++ b/serialize.h @@ -19,8 +19,8 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 106; -static const char* pszSubVer = " test11"; +static const int VERSION = 200; +static const char* pszSubVer = " rc1"; diff --git a/setup.nsi b/setup.nsi index e30ff13c..a5a73366 100644 --- a/setup.nsi +++ b/setup.nsi @@ -7,12 +7,12 @@ RequestExecutionLevel highest # General Symbol Definitions !define REGKEY "SOFTWARE\$(^Name)" -!define VERSION 0.1.6 +!define VERSION 0.2.0 !define COMPANY "Bitcoin project" -!define URL http://bitcoin.sourceforge.net/ +!define URL http://www.bitcoin.org/ # MUI Symbol Definitions -!define MUI_ICON "rc\bitcoin.ico" +!define MUI_ICON "src\rc\bitcoin.ico" !define MUI_FINISHPAGE_NOAUTOCLOSE !define MUI_STARTMENUPAGE_REGISTRY_ROOT HKLM !define MUI_STARTMENUPAGE_REGISTRY_KEY ${REGKEY} @@ -42,12 +42,12 @@ Var StartMenuGroup !insertmacro MUI_LANGUAGE English # Installer attributes -OutFile Bitcoin_0.1.6_setup.exe +OutFile bitcoin-0.2.0-setup.exe InstallDir $PROGRAMFILES\Bitcoin CRCCheck on XPStyle on ShowInstDetails show -VIProductVersion 0.1.6.0 +VIProductVersion 0.2.0.0 VIAddVersionKey ProductName Bitcoin VIAddVersionKey ProductVersion "${VERSION}" VIAddVersionKey CompanyName "${COMPANY}" @@ -65,6 +65,11 @@ Section -Main SEC0000 File bitcoin.exe File libeay32.dll File mingwm10.dll + File license.txt + File readme.txt + SetOutPath $INSTDIR\src + File /r src\*.* + SetOutPath $INSTDIR WriteRegStr HKCU "${REGKEY}\Components" Main 1 SectionEnd @@ -102,9 +107,12 @@ done${UNSECTION_ID}: # Uninstaller sections Section /o -un.Main UNSEC0000 - Delete /REBOOTOK $INSTDIR\mingwm10.dll - Delete /REBOOTOK $INSTDIR\libeay32.dll Delete /REBOOTOK $INSTDIR\bitcoin.exe + Delete /REBOOTOK $INSTDIR\libeay32.dll + Delete /REBOOTOK $INSTDIR\mingwm10.dll + Delete /REBOOTOK $INSTDIR\license.txt + Delete /REBOOTOK $INSTDIR\readme.txt + RMDir /r /REBOOTOK $INSTDIR\src DeleteRegValue HKCU "${REGKEY}\Components" Main SectionEnd @@ -114,6 +122,7 @@ Section -un.post UNSEC0001 Delete /REBOOTOK "$SMPROGRAMS\$StartMenuGroup\Bitcoin.lnk" Delete /REBOOTOK "$SMSTARTUP\Bitcoin.lnk" Delete /REBOOTOK $INSTDIR\uninstall.exe + Delete /REBOOTOK $INSTDIR\debug.log Delete /REBOOTOK $INSTDIR\db.log DeleteRegValue HKCU "${REGKEY}" StartMenuGroup DeleteRegValue HKCU "${REGKEY}" Path @@ -139,4 +148,3 @@ Function un.onInit !insertmacro MUI_STARTMENU_GETFOLDER Application $StartMenuGroup !insertmacro SELECT_UNSECTION Main ${UNSEC0000} FunctionEnd - diff --git a/ui.cpp b/ui.cpp index 262f2a88..74323028 100644 --- a/ui.cpp +++ b/ui.cpp @@ -394,6 +394,14 @@ CMainFrame::~CMainFrame() ptaskbaricon = NULL; } +void ExitTimeout(void* parg) +{ +#ifdef __WXMSW__ + Sleep(5000); + ExitProcess(0); +#endif +} + void Shutdown(void* parg) { static CCriticalSection cs_Shutdown; @@ -404,6 +412,7 @@ void Shutdown(void* parg) fFirstThread = !fTaken; fTaken = true; } + static bool fExit; if (fFirstThread) { fShutdown = true; @@ -411,13 +420,18 @@ void Shutdown(void* parg) DBFlush(false); StopNode(); DBFlush(true); + CreateThread(ExitTimeout, NULL); + Sleep(10); printf("Bitcoin exiting\n\n"); + fExit = true; exit(0); } else { - loop - Sleep(100000); + while (!fExit) + Sleep(500); + Sleep(100); + ExitThread(0); } } From e899779450bcd940dcdb0108d27c35eae92cc1e1 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 13 Dec 2009 02:13:43 +0000 Subject: [PATCH 036/133] restrict file permissions on linux to the user only git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@47 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 2 +- ui.cpp | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/db.cpp b/db.cpp index b21d1e3a..c33f71ef 100644 --- a/db.cpp +++ b/db.cpp @@ -79,7 +79,7 @@ CDB::CDB(const char* pszFile, const char* pszMode) : pdb(NULL) DB_THREAD | DB_PRIVATE | DB_RECOVER, - 0); + S_IRUSR | S_IWUSR); if (ret > 0) throw runtime_error(strprintf("CDB() : error %d opening database environment\n", ret)); fDbEnvInit = true; diff --git a/ui.cpp b/ui.cpp index 74323028..6d65720d 100644 --- a/ui.cpp +++ b/ui.cpp @@ -3494,6 +3494,7 @@ bool CMyApp::OnInit2() SetAppName("Bitcoin"); #else SetAppName("bitcoin"); + umask(077); #endif // From adb50ffe32c5c37c84ff35d94bb486004e14a7cc Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 13 Dec 2009 02:34:45 +0000 Subject: [PATCH 037/133] misc git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@48 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- util.h | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/util.h b/util.h index ae53bfed..b9c3658f 100644 --- a/util.h +++ b/util.h @@ -62,6 +62,10 @@ inline T& REF(const T& val) #define INT64_MAX _I64_MAX #define INT64_MIN _I64_MIN #endif +#ifndef S_IRUSR +#define S_IRUSR 0400 +#define S_IWUSR 0200 +#endif #else #define WSAGetLastError() errno #define WSAEWOULDBLOCK EWOULDBLOCK From e2c2648c14f4b87d331dbc30f0f2bd4aab9ce7e6 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Mon, 14 Dec 2009 02:12:01 +0000 Subject: [PATCH 038/133] filter out duplicate getblocks and don't relay inv messages during initial block download -- 0.2 rc2 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@49 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 25 ++++++++++--------------- net.cpp | 21 +++++++++++++++++++-- net.h | 7 +++++++ serialize.h | 2 +- setup.nsi | 2 +- 5 files changed, 38 insertions(+), 19 deletions(-) diff --git a/main.cpp b/main.cpp index cb93e548..4cc91cb1 100644 --- a/main.cpp +++ b/main.cpp @@ -1274,7 +1274,7 @@ bool CBlock::AcceptBlock() if (!AddToBlockIndex(nFile, nBlockPos)) return error("AcceptBlock() : AddToBlockIndex failed"); - if (hashBestChain == hash) + if (hashBestChain == hash && nBestHeight > 28000) RelayInventory(CInv(MSG_BLOCK, hash)); // // Add atoms to user reviews for coins created @@ -1314,7 +1314,7 @@ bool ProcessBlock(CNode* pfrom, CBlock* pblock) // Ask this guy to fill in what we're missing if (pfrom) - pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), GetOrphanRoot(pblock)); + pfrom->PushGetBlocks(pindexBest, GetOrphanRoot(pblock)); return true; } @@ -1816,7 +1816,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) if (!fAskedForBlocks && !pfrom->fClient) { fAskedForBlocks = true; - pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), uint256(0)); + pfrom->PushGetBlocks(pindexBest, uint256(0)); } pfrom->fSuccessfullyConnected = true; @@ -1836,6 +1836,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) { vector vAddr; vRecv >> vAddr; + if (vAddr.size() > 50000) // lower this to 1000 later + return error("message addr size() = %d", vAddr.size()); // Store the new addresses CAddrDB addrdb; @@ -1864,6 +1866,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) { vector vInv; vRecv >> vInv; + if (vInv.size() > 50000) + return error("message inv size() = %d", vInv.size()); CTxDB txdb("r"); foreach(const CInv& inv, vInv) @@ -1878,7 +1882,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) if (!fAlreadyHave) pfrom->AskFor(inv); else if (inv.type == MSG_BLOCK && mapOrphanBlocks.count(inv.hash)) - pfrom->PushMessage("getblocks", CBlockLocator(pindexBest), GetOrphanRoot(mapOrphanBlocks[inv.hash])); + pfrom->PushGetBlocks(pindexBest, GetOrphanRoot(mapOrphanBlocks[inv.hash])); // Track requests for our stuff CRITICAL_BLOCK(cs_mapRequestCount) @@ -1895,6 +1899,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) { vector vInv; vRecv >> vInv; + if (vInv.size() > 50000) + return error("message getdata size() = %d", vInv.size()); foreach(const CInv& inv, vInv) { @@ -2210,17 +2216,6 @@ bool SendMessages(CNode* pto) } } - // Clear inventory known periodically in case an inv message was missed, - // although usually they would just get it from another node. - static int64 nLastInventoryKnownClear; - if (GetTime() - nLastInventoryKnownClear > 2 * 60 * 60) // every 2 hours - { - nLastInventoryKnownClear = GetTime(); - CRITICAL_BLOCK(cs_vNodes) - foreach(CNode* pnode, vNodes) - pnode->setInventoryKnown.clear(); - } - // // Message: addr diff --git a/net.cpp b/net.cpp index 941e7c4d..168651ca 100644 --- a/net.cpp +++ b/net.cpp @@ -40,6 +40,23 @@ CAddress addrProxy("127.0.0.1:9050"); + + +void CNode::PushGetBlocks(CBlockIndex* pindexBegin, uint256 hashEnd) +{ + // Filter out duplicate requests + if (pindexBegin == pindexLastGetBlocksBegin && hashEnd == hashLastGetBlocksEnd) + return; + pindexLastGetBlocksBegin = pindexBegin; + hashLastGetBlocksEnd = hashEnd; + + PushMessage("getblocks", CBlockLocator(pindexBegin), hashEnd); +} + + + + + bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet) { hSocketRet = INVALID_SOCKET; @@ -764,12 +781,12 @@ void ThreadSocketHandler2(void* parg) printf("socket no message in first 60 seconds, %d %d\n", pnode->nLastRecv != 0, pnode->nLastSend != 0); pnode->fDisconnect = true; } - else if (GetTime() - pnode->nLastSend > 10 * 60 && GetTime() - pnode->nLastSendEmpty > 10 * 60) + else if (GetTime() - pnode->nLastSend > 90*60 && GetTime() - pnode->nLastSendEmpty > 90*60) { printf("socket not sending\n"); pnode->fDisconnect = true; } - else if (GetTime() - pnode->nLastRecv > (pnode->nVersion >= 107 ? 15*60 : 90*60)) + else if (GetTime() - pnode->nLastRecv > 90*60) { printf("socket inactivity timeout\n"); pnode->fDisconnect = true; diff --git a/net.h b/net.h index 7fe4a7c7..90af3b4b 100644 --- a/net.h +++ b/net.h @@ -7,6 +7,7 @@ class CAddress; class CInv; class CRequestTracker; class CNode; +class CBlockIndex; @@ -504,6 +505,8 @@ public: map mapRequests; CCriticalSection cs_mapRequests; uint256 hashContinue; + CBlockIndex* pindexLastGetBlocksBegin; + uint256 hashLastGetBlocksEnd; // flood vector vAddrToSend; @@ -541,6 +544,8 @@ public: nRefCount = 0; nReleaseTime = 0; hashContinue = 0; + pindexLastGetBlocksBegin = 0; + hashLastGetBlocksEnd = 0; fGetAddr = false; vfSubscribe.assign(256, false); @@ -635,6 +640,7 @@ public: } + void BeginMessage(const char* pszCommand) { cs_vSend.Enter(); @@ -900,6 +906,7 @@ public: + void PushGetBlocks(CBlockIndex* pindexBegin, uint256 hashEnd); bool IsSubscribed(unsigned int nChannel); void Subscribe(unsigned int nChannel, unsigned int nHops=0); void CancelSubscribe(unsigned int nChannel); diff --git a/serialize.h b/serialize.h index 8f925103..ce4aff3c 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 200; -static const char* pszSubVer = " rc1"; +static const char* pszSubVer = " rc2"; diff --git a/setup.nsi b/setup.nsi index a5a73366..03322877 100644 --- a/setup.nsi +++ b/setup.nsi @@ -42,7 +42,7 @@ Var StartMenuGroup !insertmacro MUI_LANGUAGE English # Installer attributes -OutFile bitcoin-0.2.0-setup.exe +OutFile bitcoin-0.2.0-win32-setup.exe InstallDir $PROGRAMFILES\Bitcoin CRCCheck on XPStyle on From 312c2c42b6d38dc7bfeef849722c8e231c38e8e4 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 17 Dec 2009 18:15:50 +0000 Subject: [PATCH 039/133] a few header changes for freebsd from madhatter git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@51 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/headers.h b/headers.h index 22bb830f..73a0b43a 100644 --- a/headers.h +++ b/headers.h @@ -35,9 +35,7 @@ #include #include #include -#include #include -#define BOUNDSCHECK 1 #include #include #include @@ -64,6 +62,7 @@ #include #include #include +#include #else #include #include @@ -77,6 +76,10 @@ #include #include #endif +#ifdef __BSD__ +#include +#endif + #pragma hdrstop using namespace std; From 9a36562347122482fd1e6e77c74df66ad8cf2c3f Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 17 Dec 2009 23:16:27 +0000 Subject: [PATCH 040/133] command line switch -noui git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@52 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- ui.cpp | 34 +++++++++++++++------------------- 1 file changed, 15 insertions(+), 19 deletions(-) diff --git a/ui.cpp b/ui.cpp index 6d65720d..67895c12 100644 --- a/ui.cpp +++ b/ui.cpp @@ -190,6 +190,9 @@ void CalledMessageBox(const string& message, const string& caption, int style, w int ThreadSafeMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) { + if (mapArgs.count("-noui")) + return wxOK; + #ifdef __WXMSW__ return wxMessageBox(message, caption, style, parent, x, y); #else @@ -3716,16 +3719,19 @@ bool CMyApp::OnInit2() // // Create the main frame window // - pframeMain = new CMainFrame(NULL); - if (mapArgs.count("-min")) - pframeMain->Iconize(true); - pframeMain->Show(true); // have to show first to get taskbar button to hide - if (fMinimizeToTray && pframeMain->IsIconized()) - fClosedToTray = true; - pframeMain->Show(!fClosedToTray); - ptaskbaricon->Show(fMinimizeToTray || fClosedToTray); + if (!mapArgs.count("-noui")) + { + pframeMain = new CMainFrame(NULL); + if (mapArgs.count("-min")) + pframeMain->Iconize(true); + pframeMain->Show(true); // have to show first to get taskbar button to hide + if (fMinimizeToTray && pframeMain->IsIconized()) + fClosedToTray = true; + pframeMain->Show(!fClosedToTray); + ptaskbaricon->Show(fMinimizeToTray || fClosedToTray); - CreateThread(ThreadDelayedRepaint, NULL); + CreateThread(ThreadDelayedRepaint, NULL); + } if (!CheckDiskSpace()) return false; @@ -3928,13 +3934,3 @@ void SetStartOnSystemStartup(bool fAutoStart) bool GetStartOnSystemStartup() { return false; } void SetStartOnSystemStartup(bool fAutoStart) { } #endif - - - - - - - - - - From cb0f89646f065800d38d7cfc10ba1e855563296a Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 28 Jan 2010 00:31:00 +0000 Subject: [PATCH 041/133] simplify AddAddress, readcompactsize limit, fixed a 64-bit compile error in serialize.h, change status "# blocks" to "# confirmations" and widen the column. git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@53 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 4 +- irc.cpp | 3 +- main.cpp | 8 +++- net.cpp | 10 ++--- net.h | 4 +- serialize.h | 16 +++++--- ui.cpp | 17 +++++++-- uibase.h | 2 +- uiproject.fbp | 4 +- util.cpp | 77 +++++++++++++++++++++++++++++++++++++- util.h | 101 ++++++-------------------------------------------- 11 files changed, 129 insertions(+), 117 deletions(-) diff --git a/db.cpp b/db.cpp index c33f71ef..77f8e1e0 100644 --- a/db.cpp +++ b/db.cpp @@ -134,8 +134,6 @@ void CDB::Close() CRITICAL_BLOCK(cs_db) --mapFileUseCount[strFile]; - - RandAddSeed(); } void CloseDb(const string& strFile) @@ -456,7 +454,7 @@ bool CAddrDB::LoadAddresses() CAddress addr(psz, NODE_NETWORK); addr.nTime = 0; // so it won't relay unless successfully connected if (addr.IsValid()) - AddAddress(*this, addr); + AddAddress(addr); } } catch (...) { } diff --git a/irc.cpp b/irc.cpp index 8e851085..f38db6bb 100644 --- a/irc.cpp +++ b/irc.cpp @@ -265,8 +265,7 @@ void ThreadIRCSeed(void* parg) if (DecodeAddress(pszName, addr)) { addr.nTime = GetAdjustedTime() - 51 * 60; - CAddrDB addrdb; - if (AddAddress(addrdb, addr)) + if (AddAddress(addr)) printf("IRC got new address\n"); nGotIRCAddresses++; } diff --git a/main.cpp b/main.cpp index 4cc91cb1..416c616a 100644 --- a/main.cpp +++ b/main.cpp @@ -1734,6 +1734,11 @@ bool ProcessMessages(CNode* pfrom) // Allow exceptions from underlength message on vRecv printf("ProcessMessage(%s, %d bytes) : Exception '%s' caught, normally caused by a message being shorter than its stated length\n", strCommand.c_str(), nMessageSize, e.what()); } + else if (strstr(e.what(), ": size too large")) + { + // Allow exceptions from overlong size + printf("ProcessMessage(%s, %d bytes) : Exception '%s' caught\n", strCommand.c_str(), nMessageSize, e.what()); + } else { PrintException(&e, "ProcessMessage()"); @@ -1840,7 +1845,6 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) return error("message addr size() = %d", vAddr.size()); // Store the new addresses - CAddrDB addrdb; foreach(CAddress& addr, vAddr) { if (fShutdown) @@ -1848,7 +1852,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) addr.nTime = GetAdjustedTime() - 2 * 60 * 60; if (pfrom->fGetAddr) addr.nTime -= 5 * 24 * 60 * 60; - AddAddress(addrdb, addr, false); + AddAddress(addr, false); pfrom->AddAddressKnown(addr); if (!pfrom->fGetAddr && addr.IsRoutable()) { diff --git a/net.cpp b/net.cpp index 168651ca..38d05de1 100644 --- a/net.cpp +++ b/net.cpp @@ -223,7 +223,7 @@ bool GetMyExternalIP(unsigned int& ipRet) -bool AddAddress(CAddrDB& addrdb, CAddress addr, bool fCurrentlyOnline) +bool AddAddress(CAddress addr, bool fCurrentlyOnline) { if (!addr.IsRoutable()) return false; @@ -239,7 +239,7 @@ bool AddAddress(CAddrDB& addrdb, CAddress addr, bool fCurrentlyOnline) // New address printf("AddAddress(%s)\n", addr.ToStringLog().c_str()); mapAddresses.insert(make_pair(addr.GetKey(), addr)); - addrdb.WriteAddress(addr); + CAddrDB().WriteAddress(addr); return true; } else @@ -260,7 +260,7 @@ bool AddAddress(CAddrDB& addrdb, CAddress addr, bool fCurrentlyOnline) fUpdated = true; } if (fUpdated) - addrdb.WriteAddress(addrFound); + CAddrDB().WriteAddress(addrFound); } } return false; @@ -881,11 +881,11 @@ void ThreadOpenConnections2(void* parg) vnThreadsRunning[1]--; Sleep(500); const int nMaxConnections = 15; - while (vNodes.size() >= nMaxConnections || vNodes.size() >= mapAddresses.size()) + while (vNodes.size() >= nMaxConnections) { + Sleep(2000); if (fShutdown) return; - Sleep(2000); } vnThreadsRunning[1]++; if (fShutdown) diff --git a/net.h b/net.h index 90af3b4b..ba4607a5 100644 --- a/net.h +++ b/net.h @@ -23,7 +23,7 @@ enum bool ConnectSocket(const CAddress& addrConnect, SOCKET& hSocketRet); bool GetMyExternalIP(unsigned int& ipRet); -bool AddAddress(CAddrDB& addrdb, CAddress addr, bool fCurrentlyOnline=true); +bool AddAddress(CAddress addr, bool fCurrentlyOnline=true); void AddressCurrentlyConnected(const CAddress& addr); CNode* FindNode(unsigned int ip); CNode* ConnectNode(CAddress addrConnect, int64 nTimeout=0); @@ -627,7 +627,7 @@ public: // We're using mapAskFor as a priority queue, // the key is the earliest time the request can be sent int64& nRequestTime = mapAlreadyAskedFor[inv]; - printf("askfor %s %"PRI64d"\n", inv.ToString().c_str(), nRequestTime); + printf("askfor %s %"PRI64d"\n", inv.ToString().c_str(), nRequestTime); // Make sure not to reuse time indexes to keep things in the same order int64 nNow = (GetTime() - 1) * 1000000; diff --git a/serialize.h b/serialize.h index ce4aff3c..263a2267 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 200; -static const char* pszSubVer = " rc2"; +static const char* pszSubVer = " test1"; @@ -194,28 +194,32 @@ uint64 ReadCompactSize(Stream& is) { unsigned char chSize; READDATA(is, chSize); + uint64 nSizeRet = 0; if (chSize < UCHAR_MAX-2) { - return chSize; + nSizeRet = chSize; } else if (chSize == UCHAR_MAX-2) { unsigned short nSize; READDATA(is, nSize); - return nSize; + nSizeRet = nSize; } else if (chSize == UCHAR_MAX-1) { unsigned int nSize; READDATA(is, nSize); - return nSize; + nSizeRet = nSize; } else { uint64 nSize; READDATA(is, nSize); - return nSize; + nSizeRet = nSize; } + if (nSizeRet > (uint64)INT_MAX) + throw std::ios_base::failure("ReadCompactSize() : size too large"); + return nSizeRet; } @@ -460,7 +464,7 @@ void Unserialize_impl(Stream& is, std::vector& v, int nType, int nVersion, unsigned int i = 0; while (i < nSize) { - unsigned int blk = min(nSize - i, 1 + 4999999 / sizeof(T)); + unsigned int blk = min(nSize - i, (unsigned int)(1 + 4999999 / sizeof(T))); v.resize(i + blk); is.read((char*)&v[i], blk * sizeof(T)); i += blk; diff --git a/ui.cpp b/ui.cpp index 67895c12..994cc7e2 100644 --- a/ui.cpp +++ b/ui.cpp @@ -349,7 +349,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) nDateWidth += 12; m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, dResize * 0); m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, dResize * 0); - m_listCtrl->InsertColumn(2, "Status", wxLIST_FORMAT_LEFT, dResize * 90); + m_listCtrl->InsertColumn(2, "Status", wxLIST_FORMAT_LEFT, dResize * 110); m_listCtrl->InsertColumn(3, "Date", wxLIST_FORMAT_LEFT, dResize * nDateWidth); m_listCtrl->InsertColumn(4, "Description", wxLIST_FORMAT_LEFT, dResize * 409 - nDateWidth); m_listCtrl->InsertColumn(5, "Debit", wxLIST_FORMAT_RIGHT, dResize * 79); @@ -579,7 +579,7 @@ string FormatTxStatus(const CWalletTx& wtx) else if (nDepth < 6) return strprintf("%d/unconfirmed", nDepth); else - return strprintf("%d blocks", nDepth); + return strprintf("%d confirmations", nDepth); } } @@ -3706,13 +3706,12 @@ bool CMyApp::OnInit2() if (mapArgs.count("-addnode")) { - CAddrDB addrdb; foreach(string strAddr, mapMultiArgs["-addnode"]) { CAddress addr(strAddr, NODE_NETWORK); addr.nTime = 0; // so it won't relay unless successfully connected if (addr.IsValid()) - AddAddress(addrdb, addr); + AddAddress(addr); } } @@ -3934,3 +3933,13 @@ void SetStartOnSystemStartup(bool fAutoStart) bool GetStartOnSystemStartup() { return false; } void SetStartOnSystemStartup(bool fAutoStart) { } #endif + + + + + + + + + + diff --git a/uibase.h b/uibase.h index e0c15218..7620a24e 100644 --- a/uibase.h +++ b/uibase.h @@ -158,7 +158,7 @@ class CMainFrameBase : public wxFrame wxListCtrl* m_listCtrlOrdersSent; wxListCtrl* m_listCtrlProductsSent; wxListCtrl* m_listCtrlOrdersReceived; - CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 705,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 725,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); ~CMainFrameBase(); }; diff --git a/uiproject.fbp b/uiproject.fbp index 58a99bdd..34fde781 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -32,7 +32,7 @@ CMainFrameBase - 705,484 + 725,484 wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER Bitcoin @@ -1737,7 +1737,7 @@ - + bSizer55 wxVERTICAL diff --git a/util.cpp b/util.cpp index f5f2797f..42922467 100644 --- a/util.cpp +++ b/util.cpp @@ -8,8 +8,8 @@ map mapArgs; map > mapMultiArgs; bool fDebug = false; -bool fPrintToDebugger = false; bool fPrintToConsole = false; +bool fPrintToDebugger = false; char pszSetDataDir[MAX_PATH] = ""; bool fShutdown = false; @@ -75,6 +75,8 @@ void RandAddSeed() void RandAddSeedPerfmon() { + RandAddSeed(); + // This can take up to 2 seconds, so only do it every 10 minutes static int64 nLastPerfmon; if (GetTime() < nLastPerfmon + 10 * 60) @@ -129,6 +131,79 @@ uint64 GetRand(uint64 nMax) +inline int OutputDebugStringF(const char* pszFormat, ...) +{ + int ret = 0; + if (fPrintToConsole || wxTheApp == NULL) + { + // print to console + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + ret = vprintf(pszFormat, arg_ptr); + va_end(arg_ptr); + } + else + { + // print to debug.log + char pszFile[MAX_PATH+100]; + GetDataDir(pszFile); + strlcat(pszFile, "/debug.log", sizeof(pszFile)); + FILE* fileout = fopen(pszFile, "a"); + if (fileout) + { + //// Debug print useful for profiling + //fprintf(fileout, " %"PRI64d" ", wxGetLocalTimeMillis().GetValue()); + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + ret = vfprintf(fileout, pszFormat, arg_ptr); + va_end(arg_ptr); + fclose(fileout); + } + } + +#ifdef __WXMSW__ + if (fPrintToDebugger) + { + // accumulate a line at a time + static CCriticalSection cs_OutputDebugStringF; + CRITICAL_BLOCK(cs_OutputDebugStringF) + { + static char pszBuffer[50000]; + static char* pend; + if (pend == NULL) + pend = pszBuffer; + va_list arg_ptr; + va_start(arg_ptr, pszFormat); + int limit = END(pszBuffer) - pend - 2; + int ret = _vsnprintf(pend, limit, pszFormat, arg_ptr); + va_end(arg_ptr); + if (ret < 0 || ret >= limit) + { + pend = END(pszBuffer) - 2; + *pend++ = '\n'; + } + else + pend += ret; + *pend = '\0'; + char* p1 = pszBuffer; + char* p2; + while (p2 = strchr(p1, '\n')) + { + p2++; + char c = *p2; + *p2 = '\0'; + OutputDebugString(p1); + *p2 = c; + p1 = p2; + } + if (p1 != pszBuffer) + memmove(pszBuffer, p1, pend - p1 + 1); + pend -= (p1 - pszBuffer); + } + } +#endif + return ret; +} // Safer snprintf diff --git a/util.h b/util.h index b9c3658f..d20648bc 100644 --- a/util.h +++ b/util.h @@ -112,13 +112,14 @@ inline int myclosesocket(SOCKET& hSocket) extern map mapArgs; extern map > mapMultiArgs; extern bool fDebug; -extern bool fPrintToDebugger; extern bool fPrintToConsole; +extern bool fPrintToDebugger; extern char pszSetDataDir[MAX_PATH]; extern bool fShutdown; void RandAddSeed(); void RandAddSeedPerfmon(); +int OutputDebugStringF(const char* pszFormat, ...); int my_snprintf(char* buffer, size_t limit, const char* format, ...); string strprintf(const char* format, ...); bool error(const char* format, ...); @@ -219,92 +220,6 @@ public: -inline int OutputDebugStringF(const char* pszFormat, ...) -{ - int ret = 0; -#ifdef __WXDEBUG__ - if (!fPrintToConsole) - { - // print to debug.log - char pszFile[MAX_PATH+100]; - GetDataDir(pszFile); - strlcat(pszFile, "/debug.log", sizeof(pszFile)); - FILE* fileout = fopen(pszFile, "a"); - if (fileout) - { - //// Debug print useful for profiling - //fprintf(fileout, " %"PRI64d" ", wxGetLocalTimeMillis().GetValue()); - va_list arg_ptr; - va_start(arg_ptr, pszFormat); - ret = vfprintf(fileout, pszFormat, arg_ptr); - va_end(arg_ptr); - fclose(fileout); - } - } - -#ifdef __WXMSW__ - if (fPrintToDebugger) - { - // accumulate a line at a time - static CCriticalSection cs_OutputDebugStringF; - CRITICAL_BLOCK(cs_OutputDebugStringF) - { - static char pszBuffer[50000]; - static char* pend; - if (pend == NULL) - pend = pszBuffer; - va_list arg_ptr; - va_start(arg_ptr, pszFormat); - int limit = END(pszBuffer) - pend - 2; - int ret = _vsnprintf(pend, limit, pszFormat, arg_ptr); - va_end(arg_ptr); - if (ret < 0 || ret >= limit) - { - pend = END(pszBuffer) - 2; - *pend++ = '\n'; - } - else - pend += ret; - *pend = '\0'; - char* p1 = pszBuffer; - char* p2; - while (p2 = strchr(p1, '\n')) - { - p2++; - char c = *p2; - *p2 = '\0'; - OutputDebugString(p1); - *p2 = c; - p1 = p2; - } - if (p1 != pszBuffer) - memmove(pszBuffer, p1, pend - p1 + 1); - pend -= (p1 - pszBuffer); - } - } -#endif -#endif - - if (fPrintToConsole) - { - // print to console - va_list arg_ptr; - va_start(arg_ptr, pszFormat); - ret = vprintf(pszFormat, arg_ptr); - va_end(arg_ptr); - } - return ret; -} - - - - - - - - - - inline string i64tostr(int64 n) { return strprintf("%"PRI64d, n); @@ -415,11 +330,19 @@ inline string DateTimeStrFormat(const char* pszFormat, int64 nTime) + + + + + + + inline void heapchk() { #ifdef __WXMSW__ - if (_heapchk() != _HEAPOK) - DebugBreak(); + /// for debugging + //if (_heapchk() != _HEAPOK) + // DebugBreak(); #endif } From 8be979d9aef606f2a196a2addcf80473d571185b Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 28 Jan 2010 21:49:20 +0000 Subject: [PATCH 042/133] now compiles on 64-bit Ubuntu Karmic with wxWidgets-2.9.0, fixes for wxWidgets-2.9.0, added superfluous GetEventHandler()'s, there's still a bug on wx2.9.0 that the status number is mashed up for some reason otherwise seems to run fine git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@54 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-unix.txt | 25 +++++--- makefile.unix => makefile.unix.wx2.8 | 0 makefile.unix.wx2.9 | 89 ++++++++++++++++++++++++++++ ui.cpp | 54 ++++++++++------- uibase.cpp | 6 +- uibase.h | 6 +- uiproject.fbp | 14 ++--- 7 files changed, 153 insertions(+), 41 deletions(-) rename makefile.unix => makefile.unix.wx2.8 (100%) create mode 100644 makefile.unix.wx2.9 diff --git a/build-unix.txt b/build-unix.txt index 7c650589..5467bbea 100644 --- a/build-unix.txt +++ b/build-unix.txt @@ -1,6 +1,6 @@ Bitcoin v0.2.0 BETA -Copyright (c) 2009 Satoshi Nakamoto +Copyright (c) 2009-2010 Satoshi Nakamoto Distributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php. This product includes software developed by the OpenSSL Project for use in @@ -13,15 +13,22 @@ UNIX BUILD NOTES Dependencies ------------ -apt-get install build-essential -apt-get install libgtk2.0-dev -apt-get install libssl-dev -apt-get install libdb4.7-dev -apt-get install libdb4.7++-dev -apt-get install libboost-dev +sudo apt-get install build-essential +sudo apt-get install libgtk2.0-dev +sudo apt-get install libssl-dev +sudo apt-get install libdb4.7-dev +sudo apt-get install libdb4.7++-dev +sudo apt-get install libboost-dev + +There is currently no libwxgtk2.8-ansi-dev debian package for Karmic. +libwxgtk2.8-dev is the "unicode" build, but for wxWidgets 2.8 "unicode" +means wchar, not UTF-8. wchar wxString doesn't convert to std::string. + +In wxWidgets 2.9, unicode is UTF-8 and that's the only version. You need to download wxWidgets from http://www.wxwidgets.org/downloads/ -and build it yourself. +and build it yourself. See the build instructions and configure parameters +below. Licenses of statically linked libraries: wxWidgets LGPL 2.1 with very liberal exceptions @@ -48,7 +55,7 @@ symbols, which reduces the executable size by about 90%. wxWidgets --------- -cd /usr/local/wxWidgets-2.8.9 +cd /usr/local/wxWidgets-2.8.9 or 2.9.0 mkdir buildgtk cd buildgtk ../configure --with-gtk --enable-debug --disable-shared --enable-monolithic diff --git a/makefile.unix b/makefile.unix.wx2.8 similarity index 100% rename from makefile.unix rename to makefile.unix.wx2.8 diff --git a/makefile.unix.wx2.9 b/makefile.unix.wx2.9 new file mode 100644 index 00000000..81dcbd70 --- /dev/null +++ b/makefile.unix.wx2.9 @@ -0,0 +1,89 @@ +# Copyright (c) 2009 Satoshi Nakamoto +# Distributed under the MIT/X11 software license, see the accompanying +# file license.txt or http://www.opensource.org/licenses/mit-license.php. + + +ifneq "$(BUILD)" "debug" +ifneq "$(BUILD)" "release" +BUILD=debug +endif +endif +ifeq "$(BUILD)" "debug" +D=d +DEBUGFLAGS=-g -D__WXDEBUG__ +endif + + + +INCLUDEPATHS= \ + -I"/usr/include" \ + -I"/usr/local/include/wx-2.9" \ + -I"/usr/local/lib/wx/include/gtk2-unicode-debug-static-2.9" + +LIBPATHS= \ + -L"/usr/lib" \ + -L"/usr/local/lib" + +LIBS= \ + -Wl,-Bstatic \ + -l boost_system-mt -l boost_filesystem-mt \ + -l db_cxx \ + -l wx_gtk2u$(D)-2.9 \ + -Wl,-Bdynamic \ + -l crypto \ + -l gtk-x11-2.0 -l gthread-2.0 -l SM + +WXDEFS=-D__WXGTK__ -DNOPCH +CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h + + + +all: bitcoin + + +headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h + g++ -c $(CFLAGS) -o $@ $< + +obj/util.o: util.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/script.o: script.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/db.o: db.cpp $(HEADERS) market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/net.o: net.cpp $(HEADERS) net.h + g++ -c $(CFLAGS) -o $@ $< + +obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h + g++ -c $(CFLAGS) -o $@ $< + +obj/market.o: market.cpp $(HEADERS) market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h + g++ -c $(CFLAGS) -o $@ $< + +obj/uibase.o: uibase.cpp uibase.h + g++ -c $(CFLAGS) -o $@ $< + +obj/sha.o: sha.cpp sha.h + g++ -c $(CFLAGS) -O3 -o $@ $< + +obj/irc.o: irc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + + + + +OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + +bitcoin: headers.h.gch $(OBJS) + g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) + +clean: + -rm obj/* + -rm headers.h.gch diff --git a/ui.cpp b/ui.cpp index 994cc7e2..d1163aa5 100644 --- a/ui.cpp +++ b/ui.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -252,18 +252,23 @@ bool IsCallbackAvailable(void* p) template void AddPendingCustomEvent(wxEvtHandler* pevthandler, int nEventID, const T pbeginIn, const T pendIn) { - if (!pevthandler) - return; - - const char* pbegin = (pendIn != pbeginIn) ? &pbeginIn[0] : NULL; - const char* pend = pbegin + (pendIn - pbeginIn) * sizeof(pbeginIn[0]); - wxCommandEvent event(nEventID); - wxString strData(wxChar(0), (pend - pbegin) / sizeof(wxChar) + 1); - memcpy(&strData[0], pbegin, pend - pbegin); - event.SetString(strData); - event.SetInt(pend - pbegin); - - pevthandler->AddPendingEvent(event); + // Need to rewrite with something like UIThreadCall + // I'm tired of maintaining this hack that's only called by unfinished unused code, + // but I'm not willing to delete it because it serves as documentation of what the + // unfinished code was trying to do. + assert(("Unimplemented", 0)); + //if (!pevthandler) + // return; + // + //const char* pbegin = (pendIn != pbeginIn) ? &pbeginIn[0] : NULL; + //const char* pend = pbegin + (pendIn - pbeginIn) * sizeof(pbeginIn[0]); + //wxCommandEvent event(nEventID); + //wxString strData(wxChar(0), (pend - pbegin) / sizeof(wxChar) + 1); + //memcpy(&strData[0], pbegin, pend - pbegin); + //event.SetString(strData); + //event.SetInt(pend - pbegin); + // + //pevthandler->AddPendingEvent(event); } template @@ -335,9 +340,8 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_toolBar->AddTool(wxID_BUTTONRECEIVE, "Address Book", wxBitmap(addressbook20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); m_toolBar->Realize(); // resize to fit ubuntu's huge default font - dResize = 1.19; - SetSize(dResize * GetSize().GetWidth(), 1.1 * GetSize().GetHeight()); dResize = 1.20; + SetSize(dResize * GetSize().GetWidth(), 1.1 * GetSize().GetHeight()); #endif m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); @@ -458,6 +462,8 @@ void CMainFrame::OnIconize(wxIconizeEvent& event) { // Hide the task bar button when minimized. // Event is sent when the frame is minimized or restored. + // wxWidgets 2.8.9 doesn't have IsIconized() so there's no way + // to get rid of the deprecated warning. Just ignore it. if (!event.Iconized()) fClosedToTray = false; #ifndef __WXMSW__ @@ -985,7 +991,7 @@ void ThreadDelayedRepaint(void* parg) printf("DelayedRepaint\n"); wxPaintEvent event; pframeMain->fRefresh = true; - pframeMain->AddPendingEvent(event); + pframeMain->GetEventHandler()->AddPendingEvent(event); } } Sleep(nRepaintInterval); @@ -1010,7 +1016,7 @@ void MainFrameRepaint() printf("MainFrameRepaint\n"); wxPaintEvent event; pframeMain->fRefresh = true; - pframeMain->AddPendingEvent(event); + pframeMain->GetEventHandler()->AddPendingEvent(event); } } @@ -1695,12 +1701,14 @@ CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) // Workaround until upgrade to wxWidgets supporting UTF-8 wxString str = m_staticTextMain->GetLabel(); +#if !wxUSE_UNICODE if (str.Find('Â') != wxNOT_FOUND) str.Remove(str.Find('Â'), 1); - m_staticTextMain->SetLabel(str); +#endif #ifndef __WXMSW__ SetSize(510, 380); #endif + m_staticTextMain->SetLabel(str); } void CAboutDialog::OnButtonOK(wxCommandEvent& event) @@ -1732,7 +1740,7 @@ CSendDialog::CSendDialog(wxWindow* parent, const wxString& strAddress) : CSendDi if (fontTmp.GetPointSize() > 9); fontTmp.SetPointSize(9); m_staticTextInstructions->SetFont(fontTmp); - SetSize(725, wxDefaultCoord); + SetSize(725, 380); #endif // Set Icon @@ -2000,7 +2008,7 @@ void CSendingDialog::Repaint() { Refresh(); wxPaintEvent event; - AddPendingEvent(event); + GetEventHandler()->AddPendingEvent(event); } bool CSendingDialog::Status() @@ -3379,14 +3387,14 @@ void CMyTaskBarIcon::OnMenuOptions(wxCommandEvent& event) { // Since it's modal, get the main window to do it wxCommandEvent event2(wxEVT_COMMAND_MENU_SELECTED, wxID_MENUOPTIONSOPTIONS); - pframeMain->AddPendingEvent(event2); + pframeMain->GetEventHandler()->AddPendingEvent(event2); } void CMyTaskBarIcon::Restore() { pframeMain->Show(); wxIconizeEvent event(0, false); - pframeMain->AddPendingEvent(event); + pframeMain->GetEventHandler()->AddPendingEvent(event); pframeMain->Iconize(false); pframeMain->Raise(); } @@ -3548,7 +3556,7 @@ bool CMyApp::OnInit2() if (!fDebug && !pszSetDataDir[0]) ShrinkDebugFile(); printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); - printf("Bitcoin version %d%s, OS version %s\n", VERSION, pszSubVer, wxGetOsDescription().mb_str()); + printf("Bitcoin version %d%s, OS version %s\n", VERSION, pszSubVer, ((string)wxGetOsDescription()).c_str()); if (mapArgs.count("-loadblockindextest")) { diff --git a/uibase.cpp b/uibase.cpp index f05f1095..fb4f1ed0 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -1,3 +1,7 @@ +// Copyright (c) 2009-2010 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -639,7 +643,7 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin bSizer21->Add( 0, 5, 0, wxEXPAND, 5 ); wxFlexGridSizer* fgSizer1; - fgSizer1 = new wxFlexGridSizer( 3, 2, 0, 0 ); + fgSizer1 = new wxFlexGridSizer( 0, 2, 0, 0 ); fgSizer1->AddGrowableCol( 1 ); fgSizer1->SetFlexibleDirection( wxBOTH ); fgSizer1->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED ); diff --git a/uibase.h b/uibase.h index 7620a24e..97bb1bdc 100644 --- a/uibase.h +++ b/uibase.h @@ -1,3 +1,7 @@ +// Copyright (c) 2009-2010 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -158,7 +162,7 @@ class CMainFrameBase : public wxFrame wxListCtrl* m_listCtrlOrdersSent; wxListCtrl* m_listCtrlProductsSent; wxListCtrl* m_listCtrlOrdersReceived; - CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 725,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 727,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); ~CMainFrameBase(); }; diff --git a/uiproject.fbp b/uiproject.fbp index 34fde781..3c489ae6 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -32,7 +32,7 @@ CMainFrameBase - 725,484 + 727,484 wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER Bitcoin @@ -3509,7 +3509,7 @@ - + @@ -3580,7 +3580,7 @@ 5 wxEXPAND|wxLEFT 0 - + 2 wxBOTH 1 @@ -3590,7 +3590,7 @@ fgSizer1 wxFLEX_GROWMODE_SPECIFIED none - 3 + 0 0 5 @@ -4189,7 +4189,7 @@ 5 wxEXPAND 0 - + bSizer672 wxHORIZONTAL @@ -4317,7 +4317,7 @@ 5 wxEXPAND 1 - + bSizer67 wxHORIZONTAL @@ -4445,7 +4445,7 @@ 5 wxEXPAND 0 - + bSizer23 wxHORIZONTAL From 53d508072b02d522371bde148dcc3e925f472be7 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 3 Feb 2010 22:58:40 +0000 Subject: [PATCH 043/133] update fSpent flag on wallet transactions if they're seen spent in case copy of wallet.dat was used elsewhere or restored from backup, better error dialog box if try to spend already spent coins, got rid of unused notebook with only one tab on main dialog, nicer looking About dialog, resize About dialog better on linux git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@55 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 13 +- headers.h | 3 +- main.cpp | 67 ++- main.h | 28 +- script.cpp | 12 +- serialize.h | 4 +- ui.cpp | 46 +- ui.h | 4 +- uibase.cpp | 124 +---- uibase.h | 23 +- uiproject.fbp | 1449 +++++++++++++++---------------------------------- xpm/about.xpm | 665 +++++++++++++++++++++++ 12 files changed, 1275 insertions(+), 1163 deletions(-) create mode 100644 xpm/about.xpm diff --git a/db.cpp b/db.cpp index 77f8e1e0..ff9ece52 100644 --- a/db.cpp +++ b/db.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -593,14 +593,17 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) // wtx.hashBlock.ToString().substr(0,16).c_str(), // wtx.mapValue["message"].c_str()); } - else if (strType == "key") + else if (strType == "key" || strType == "wkey") { vector vchPubKey; ssKey >> vchPubKey; - CPrivKey vchPrivKey; - ssValue >> vchPrivKey; + CWalletKey wkey; + if (strType == "key") + ssValue >> wkey.vchPrivKey; + else + ssValue >> wkey; - mapKeys[vchPubKey] = vchPrivKey; + mapKeys[vchPubKey] = wkey.vchPrivKey; mapPubKeys[Hash160(vchPubKey)] = vchPubKey; } else if (strType == "defaultkey") diff --git a/headers.h b/headers.h index 73a0b43a..45be4b65 100644 --- a/headers.h +++ b/headers.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -113,3 +113,4 @@ using namespace boost; #include "xpm/send16.xpm" #include "xpm/send16noshadow.xpm" #include "xpm/send20.xpm" +#include "xpm/about.xpm" diff --git a/main.cpp b/main.cpp index 416c616a..53acf8a7 100644 --- a/main.cpp +++ b/main.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -168,6 +168,27 @@ bool EraseFromWallet(uint256 hash) return true; } +void WalletUpdateSpent(const COutPoint& prevout) +{ + // Anytime a signature is successfully verified, it's proof the outpoint is spent. + // Update the wallet spent flag if it doesn't know due to wallet.dat being + // restored from backup or the user making copies of wallet.dat. + CRITICAL_BLOCK(cs_mapWallet) + { + map::iterator mi = mapWallet.find(prevout.hash); + if (mi != mapWallet.end()) + { + CWalletTx& wtx = (*mi).second; + if (!wtx.fSpent && wtx.vout[prevout.n].IsMine()) + { + printf("WalletUpdateSpent found spent coin %sbc %s\n", FormatMoney(wtx.GetCredit()).c_str(), wtx.GetHash().ToString().c_str()); + wtx.fSpent = true; + wtx.WriteToDisk(); + vWalletUpdated.push_back(prevout.hash); + } + } + } +} @@ -622,15 +643,44 @@ bool CWalletTx::AcceptWalletTransaction(CTxDB& txdb, bool fCheckInputs) void ReacceptWalletTransactions() { - // Reaccept any txes of ours that aren't already in a block CTxDB txdb("r"); CRITICAL_BLOCK(cs_mapWallet) { foreach(PAIRTYPE(const uint256, CWalletTx)& item, mapWallet) { CWalletTx& wtx = item.second; - if (!wtx.IsCoinBase() && !txdb.ContainsTx(wtx.GetHash())) - wtx.AcceptWalletTransaction(txdb, false); + if (wtx.fSpent && wtx.IsCoinBase()) + continue; + + CTxIndex txindex; + if (txdb.ReadTxIndex(wtx.GetHash(), txindex)) + { + // Update fSpent if a tx got spent somewhere else by a copy of wallet.dat + if (!wtx.fSpent) + { + if (txindex.vSpent.size() != wtx.vout.size()) + { + printf("ERROR: ReacceptWalletTransactions() : txindex.vSpent.size() %d != wtx.vout.size() %d\n", txindex.vSpent.size(), wtx.vout.size()); + continue; + } + for (int i = 0; i < txindex.vSpent.size(); i++) + { + if (!txindex.vSpent[i].IsNull() && wtx.vout[i].IsMine()) + { + printf("ReacceptWalletTransactions found spent coin %sbc %s\n", FormatMoney(wtx.GetCredit()).c_str(), wtx.GetHash().ToString().c_str()); + wtx.fSpent = true; + wtx.WriteToDisk(); + break; + } + } + } + } + else + { + // Reaccept any txes of ours that aren't already in a block + if (!wtx.IsCoinBase()) + wtx.AcceptWalletTransaction(txdb, false); + } } } } @@ -2843,9 +2893,13 @@ bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key) CRITICAL_BLOCK(cs_main) CRITICAL_BLOCK(cs_mapWallet) { - //// todo: eventually should make this transactional, never want to add a + //// old: eventually should make this transactional, never want to add a //// transaction without marking spent transactions, although the risk of //// interruption during this step is remote. + //// update: This matters even less now that fSpent can get corrected + //// when transactions are seen in VerifySignature. The remote chance of + //// unmarked fSpent will be handled by that. Don't need to make this + //// transactional. // This is only to keep the database open to defeat the auto-flush for the // duration of this scope. This is the only place where this optimization @@ -2910,8 +2964,7 @@ bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) if (!wtxNew.AcceptTransaction()) { // This must not fail. The transaction has already been signed and recorded. - throw runtime_error("SendMoney() : wtxNew.AcceptTransaction() failed\n"); - wxMessageBox("Error: Transaction not valid ", "Sending..."); + wxMessageBox("Error: The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here.", "Sending..."); return error("SendMoney() : Error: Transaction not valid"); } wtxNew.RelayWalletTransaction(); diff --git a/main.h b/main.h index 716485e9..14c445c7 100644 --- a/main.h +++ b/main.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -56,6 +56,7 @@ FILE* AppendBlockFile(unsigned int& nFileRet); bool AddKey(const CKey& key); vector GenerateNewKey(); bool AddToWallet(const CWalletTx& wtxIn); +void WalletUpdateSpent(const COutPoint& prevout); void ReacceptWalletTransactions(); void RelayWalletTransactions(); bool LoadBlockIndex(bool fAllowNew=true); @@ -1370,6 +1371,31 @@ public: +// +// Private key that includes an expiration date in case it never gets used. +// +class CWalletKey +{ +public: + CPrivKey vchPrivKey; + int64 nTimeCreated; + int64 nTimeExpires; + + CWalletKey(int64 nTimeExpiresIn=0) + { + nTimeCreated = (nTimeExpiresIn ? GetTime() : 0); + nTimeExpires = nTimeExpiresIn; + } + + IMPLEMENT_SERIALIZE + ( + if (!(nType & SER_GETHASH)) + READWRITE(nVersion); + READWRITE(vchPrivKey); + READWRITE(nTimeCreated); + READWRITE(nTimeExpires); + ) +}; diff --git a/script.cpp b/script.cpp index 0e95af50..a41de2aa 100644 --- a/script.cpp +++ b/script.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -1123,5 +1123,13 @@ bool VerifySignature(const CTransaction& txFrom, const CTransaction& txTo, unsig if (txin.prevout.hash != txFrom.GetHash()) return false; - return EvalScript(txin.scriptSig + CScript(OP_CODESEPARATOR) + txout.scriptPubKey, txTo, nIn, nHashType); + if (!EvalScript(txin.scriptSig + CScript(OP_CODESEPARATOR) + txout.scriptPubKey, txTo, nIn, nHashType)) + return false; + + // Anytime a signature is successfully verified, it's proof the outpoint is spent, + // so lets update the wallet spent flag if it doesn't know due to wallet.dat being + // restored from backup or the user making copies of wallet.dat. + WalletUpdateSpent(txin.prevout); + + return true; } diff --git a/serialize.h b/serialize.h index 263a2267..439ef641 100644 --- a/serialize.h +++ b/serialize.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 200; -static const char* pszSubVer = " test1"; +static const char* pszSubVer = " test2"; diff --git a/ui.cpp b/ui.cpp index d1163aa5..5d93ad22 100644 --- a/ui.cpp +++ b/ui.cpp @@ -134,7 +134,6 @@ int GetSelection(wxListCtrl* listCtrl) return -1; } - string HtmlEscape(const char* psz, bool fMultiLine=false) { int len = 0; @@ -170,18 +169,6 @@ string HtmlEscape(const string& str, bool fMultiLine=false) return HtmlEscape(str.c_str(), fMultiLine); } -void AddToMyProducts(CProduct product) -{ - CProduct& productInsert = mapMyProducts[product.GetHash()]; - productInsert = product; - InsertLine(pframeMain->m_listCtrlProductsSent, &productInsert, - product.mapValue["category"], - product.mapValue["title"].substr(0, 100), - product.mapValue["description"].substr(0, 100), - product.mapValue["price"], - ""); -} - void CalledMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y, int* pnRet, bool* pfDone) { *pnRet = wxMessageBox(message, caption, style, parent, x, y); @@ -253,9 +240,7 @@ template void AddPendingCustomEvent(wxEvtHandler* pevthandler, int nEventID, const T pbeginIn, const T pendIn) { // Need to rewrite with something like UIThreadCall - // I'm tired of maintaining this hack that's only called by unfinished unused code, - // but I'm not willing to delete it because it serves as documentation of what the - // unfinished code was trying to do. + // I'm tired of maintaining this hack that's only called by unfinished unused code. assert(("Unimplemented", 0)); //if (!pevthandler) // return; @@ -1233,7 +1218,7 @@ void CMainFrame::OnButtonChange(wxCommandEvent& event) } } -void CMainFrame::OnListItemActivatedAllTransactions(wxListEvent& event) +void CMainFrame::OnListItemActivated(wxListEvent& event) { uint256 hash((string)GetItemText(m_listCtrl, event.GetIndex(), 1)); CWalletTx wtx; @@ -1242,7 +1227,7 @@ void CMainFrame::OnListItemActivatedAllTransactions(wxListEvent& event) map::iterator mi = mapWallet.find(hash); if (mi == mapWallet.end()) { - printf("CMainFrame::OnListItemActivatedAllTransactions() : tx not found in mapWallet\n"); + printf("CMainFrame::OnListItemActivated() : tx not found in mapWallet\n"); return; } wtx = (*mi).second; @@ -1699,16 +1684,23 @@ CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) { m_staticTextVersion->SetLabel(strprintf("version 0.%d.%d beta", VERSION/100, VERSION%100)); +#if !wxUSE_UNICODE // Workaround until upgrade to wxWidgets supporting UTF-8 wxString str = m_staticTextMain->GetLabel(); -#if !wxUSE_UNICODE if (str.Find('Â') != wxNOT_FOUND) str.Remove(str.Find('Â'), 1); + m_staticTextMain->SetLabel(str); #endif #ifndef __WXMSW__ - SetSize(510, 380); + // Resize on Linux to make the window fit the text. + // The text was wrapped manually rather than using the Wrap setting because + // the wrap would be too small on Linux and it can't be changed at this point. + wxFont fontTmp = m_staticTextMain->GetFont(); + if (fontTmp.GetPointSize() > 8); + fontTmp.SetPointSize(8); + m_staticTextMain->SetFont(fontTmp); + SetSize(GetSize().GetWidth() + 44, GetSize().GetHeight() - 4); #endif - m_staticTextMain->SetLabel(str); } void CAboutDialog::OnButtonOK(wxCommandEvent& event) @@ -2789,6 +2781,18 @@ void CEditProductDialog::OnButtonAddField(wxCommandEvent& event) } } +void AddToMyProducts(CProduct product) +{ + CProduct& productInsert = mapMyProducts[product.GetHash()]; + productInsert = product; + //InsertLine(pframeMain->m_listCtrlProductsSent, &productInsert, + // product.mapValue["category"], + // product.mapValue["title"].substr(0, 100), + // product.mapValue["description"].substr(0, 100), + // product.mapValue["price"], + // ""); +} + void CEditProductDialog::OnButtonSend(wxCommandEvent& event) { CProduct product; diff --git a/ui.h b/ui.h index c4bf8b66..43e05659 100644 --- a/ui.h +++ b/ui.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -60,7 +60,7 @@ protected: void OnButtonCopy(wxCommandEvent& event); void OnButtonChange(wxCommandEvent& event); void OnListColBeginDrag(wxListEvent& event); - void OnListItemActivatedAllTransactions(wxListEvent& event); + void OnListItemActivated(wxListEvent& event); void OnListItemActivatedProductsSent(wxListEvent& event); void OnListItemActivatedOrdersSent(wxListEvent& event); void OnListItemActivatedOrdersReceived(wxListEvent& event); diff --git a/uibase.cpp b/uibase.cpp index fb4f1ed0..08c025fa 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -1,7 +1,6 @@ // Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. - /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -11,6 +10,7 @@ #include "uibase.h" +#include "xpm/about.xpm" #include "xpm/addressbook20.xpm" #include "xpm/check.xpm" #include "xpm/send20.xpm" @@ -142,81 +142,8 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& bSizer2->Add( bSizer3, 0, wxEXPAND, 5 ); - m_notebook = new wxNotebook( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 ); - m_panel7 = new wxPanel( m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); - wxBoxSizer* bSizer157; - bSizer157 = new wxBoxSizer( wxVERTICAL ); - - m_listCtrl = new wxListCtrl( m_panel7, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); - bSizer157->Add( m_listCtrl, 1, wxEXPAND|wxALL, 5 ); - - m_panel7->SetSizer( bSizer157 ); - m_panel7->Layout(); - bSizer157->Fit( m_panel7 ); - m_notebook->AddPage( m_panel7, wxT("All Transactions"), false ); - - bSizer2->Add( m_notebook, 1, wxEXPAND, 5 ); - - wxBoxSizer* bSizer_TabsForFutureUse; - bSizer_TabsForFutureUse = new wxBoxSizer( wxVERTICAL ); - - m_panel9 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); - m_panel9->Hide(); - - wxBoxSizer* bSizer159; - bSizer159 = new wxBoxSizer( wxVERTICAL ); - - m_listCtrlEscrows = new wxListCtrl( m_panel9, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); - bSizer159->Add( m_listCtrlEscrows, 1, wxALL|wxEXPAND, 5 ); - - m_panel9->SetSizer( bSizer159 ); - m_panel9->Layout(); - bSizer159->Fit( m_panel9 ); - bSizer_TabsForFutureUse->Add( m_panel9, 1, wxEXPAND | wxALL, 5 ); - - m_panel8 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); - m_panel8->Hide(); - - wxBoxSizer* bSizer158; - bSizer158 = new wxBoxSizer( wxVERTICAL ); - - m_listCtrlOrdersSent = new wxListCtrl( m_panel8, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); - bSizer158->Add( m_listCtrlOrdersSent, 1, wxALL|wxEXPAND, 5 ); - - m_panel8->SetSizer( bSizer158 ); - m_panel8->Layout(); - bSizer158->Fit( m_panel8 ); - bSizer_TabsForFutureUse->Add( m_panel8, 1, wxEXPAND | wxALL, 5 ); - - m_panel10 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); - m_panel10->Hide(); - - wxBoxSizer* bSizer160; - bSizer160 = new wxBoxSizer( wxVERTICAL ); - - m_listCtrlProductsSent = new wxListCtrl( m_panel10, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); - bSizer160->Add( m_listCtrlProductsSent, 1, wxALL|wxEXPAND, 5 ); - - m_panel10->SetSizer( bSizer160 ); - m_panel10->Layout(); - bSizer160->Fit( m_panel10 ); - bSizer_TabsForFutureUse->Add( m_panel10, 1, wxEXPAND | wxALL, 5 ); - - m_panel11 = new wxPanel( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); - m_panel11->Hide(); - - wxBoxSizer* bSizer161; - bSizer161 = new wxBoxSizer( wxVERTICAL ); - - m_listCtrlOrdersReceived = new wxListCtrl( m_panel11, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); - bSizer161->Add( m_listCtrlOrdersReceived, 1, wxALL|wxEXPAND, 5 ); - - m_panel11->SetSizer( bSizer161 ); - m_panel11->Layout(); - bSizer161->Fit( m_panel11 ); - bSizer_TabsForFutureUse->Add( m_panel11, 1, wxEXPAND | wxALL, 5 ); - - bSizer2->Add( bSizer_TabsForFutureUse, 1, wxEXPAND, 5 ); + m_listCtrl = new wxListCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + bSizer2->Add( m_listCtrl, 1, wxEXPAND, 5 ); this->SetSizer( bSizer2 ); this->Layout(); @@ -267,11 +194,8 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_buttonCopy->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); m_button91->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); m_listCtrl->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedAllTransactions ), NULL, this ); + m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); m_listCtrl->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); - m_listCtrlOrdersSent->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersSent ), NULL, this ); - m_listCtrlProductsSent->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedProductsSent ), NULL, this ); - m_listCtrlOrdersReceived->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersReceived ), NULL, this ); } CMainFrameBase::~CMainFrameBase() @@ -322,11 +246,8 @@ CMainFrameBase::~CMainFrameBase() m_buttonCopy->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); m_button91->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedAllTransactions ), NULL, this ); + m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); m_listCtrl->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); - m_listCtrlOrdersSent->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersSent ), NULL, this ); - m_listCtrlProductsSent->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedProductsSent ), NULL, this ); - m_listCtrlOrdersReceived->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivatedOrdersReceived ), NULL, this ); } CTxDetailsDialogBase::CTxDetailsDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) @@ -562,20 +483,23 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr { this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + wxBoxSizer* bSizer63; + bSizer63 = new wxBoxSizer( wxHORIZONTAL ); + + m_bitmap = new wxStaticBitmap( this, wxID_ANY, wxBitmap( about_xpm ), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer63->Add( m_bitmap, 0, 0, 5 ); + wxBoxSizer* bSizer60; bSizer60 = new wxBoxSizer( wxVERTICAL ); wxBoxSizer* bSizer62; bSizer62 = new wxBoxSizer( wxHORIZONTAL ); + wxBoxSizer* bSizer631; + bSizer631 = new wxBoxSizer( wxVERTICAL ); - bSizer62->Add( 60, 0, 0, wxEXPAND, 5 ); - wxBoxSizer* bSizer63; - bSizer63 = new wxBoxSizer( wxVERTICAL ); - - - bSizer63->Add( 0, 50, 0, wxEXPAND, 5 ); + bSizer631->Add( 0, 65, 0, wxEXPAND, 5 ); wxBoxSizer* bSizer64; bSizer64 = new wxBoxSizer( wxHORIZONTAL ); @@ -592,19 +516,19 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr bSizer64->Add( m_staticTextVersion, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxRIGHT, 5 ); - bSizer63->Add( bSizer64, 0, wxEXPAND, 5 ); + bSizer631->Add( bSizer64, 0, wxEXPAND, 5 ); - bSizer63->Add( 0, 4, 0, wxEXPAND, 5 ); + bSizer631->Add( 0, 4, 0, wxEXPAND, 5 ); - m_staticTextMain = new wxStaticText( this, wxID_ANY, wxT("Copyright © 2009 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticTextMain->Wrap( 400 ); - bSizer63->Add( m_staticTextMain, 0, wxALL, 5 ); + m_staticTextMain = new wxStaticText( this, wxID_ANY, wxT("Copyright © 2009-2010 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file \nlicense.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the \nOpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by \nEric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMain->Wrap( -1 ); + bSizer631->Add( m_staticTextMain, 0, wxALL, 5 ); - bSizer63->Add( 0, 0, 1, wxEXPAND, 5 ); + bSizer631->Add( 0, 0, 1, wxEXPAND, 5 ); - bSizer62->Add( bSizer63, 1, wxEXPAND, 5 ); + bSizer62->Add( bSizer631, 1, wxEXPAND, 5 ); bSizer60->Add( bSizer62, 1, wxEXPAND, 5 ); @@ -617,9 +541,11 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer61->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - bSizer60->Add( bSizer61, 0, wxALIGN_RIGHT|wxEXPAND, 5 ); + bSizer60->Add( bSizer61, 0, wxALIGN_RIGHT|wxEXPAND|wxRIGHT, 5 ); + + bSizer63->Add( bSizer60, 1, wxEXPAND|wxLEFT, 5 ); - this->SetSizer( bSizer60 ); + this->SetSizer( bSizer63 ); this->Layout(); // Connect Events diff --git a/uibase.h b/uibase.h index 97bb1bdc..faef9829 100644 --- a/uibase.h +++ b/uibase.h @@ -1,7 +1,6 @@ // Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. - /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -30,7 +29,6 @@ #include #include #include -#include #include #include #include @@ -119,12 +117,6 @@ class CMainFrameBase : public wxFrame wxStaticText* m_staticTextBalance; wxChoice* m_choiceFilter; - wxNotebook* m_notebook; - wxPanel* m_panel7; - wxPanel* m_panel9; - wxPanel* m_panel8; - wxPanel* m_panel10; - wxPanel* m_panel11; // Virtual event handlers, overide them in your derived class virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } @@ -148,21 +140,14 @@ class CMainFrameBase : public wxFrame virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonChange( wxCommandEvent& event ){ event.Skip(); } virtual void OnListColBeginDrag( wxListEvent& event ){ event.Skip(); } - virtual void OnListItemActivatedAllTransactions( wxListEvent& event ){ event.Skip(); } + virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } virtual void OnPaintListCtrl( wxPaintEvent& event ){ event.Skip(); } - virtual void OnListItemActivatedOrdersSent( wxListEvent& event ){ event.Skip(); } - virtual void OnListItemActivatedProductsSent( wxListEvent& event ){ event.Skip(); } - virtual void OnListItemActivatedOrdersReceived( wxListEvent& event ){ event.Skip(); } public: wxMenu* m_menuOptions; wxListCtrl* m_listCtrl; - wxListCtrl* m_listCtrlEscrows; - wxListCtrl* m_listCtrlOrdersSent; - wxListCtrl* m_listCtrlProductsSent; - wxListCtrl* m_listCtrlOrdersReceived; - CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 727,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 712,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); ~CMainFrameBase(); }; @@ -251,7 +236,7 @@ class CAboutDialogBase : public wxDialog private: protected: - + wxStaticBitmap* m_bitmap; wxStaticText* m_staticText40; @@ -266,7 +251,7 @@ class CAboutDialogBase : public wxDialog public: wxStaticText* m_staticTextVersion; - CAboutDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("About Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 507,298 ), long style = wxDEFAULT_DIALOG_STYLE ); + CAboutDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("About Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 532,329 ), long style = wxDEFAULT_DIALOG_STYLE ); ~CAboutDialogBase(); }; diff --git a/uiproject.fbp b/uiproject.fbp index 3c489ae6..c24f6962 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -18,7 +18,7 @@ 1 0 0 - + wxSYS_COLOUR_BTNFACE @@ -32,7 +32,7 @@ CMainFrameBase - 727,484 + 712,484 wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER Bitcoin @@ -824,9 +824,8 @@ 5 wxEXPAND 1 - + - 1 @@ -835,16 +834,16 @@ wxID_ANY - m_notebook - protected + m_listCtrl + public - + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING - + wxVSCROLL @@ -855,26 +854,114 @@ + + + + + OnListColBeginDrag + + + + + + + + + OnListItemActivated + + + + + + - - - + OnPaintListCtrl - - - All Transactions - 0 - + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + CTxDetailsDialogBase + + 620,450 + wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER + + Transaction Details + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer64 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + bSizer66 + wxVERTICAL + none + + 5 + wxALL|wxEXPAND + 1 + 1 @@ -884,18 +971,22 @@ wxID_ANY - m_panel7 + m_htmlWin protected + wxHW_SCROLLBAR_AUTO - wxTAB_TRAVERSAL + + + + @@ -916,117 +1007,46 @@ - - - bSizer157 - wxVERTICAL - none - - 5 - wxEXPAND|wxALL - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_listCtrl - public - - - wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING - - - - - wxVSCROLL - - - - - - - - - - - - - - - OnListColBeginDrag - - - - - - - - - OnListItemActivatedAllTransactions - - - - - - - - - - - - - OnPaintListCtrl - - - - - - - - - 5 - wxEXPAND - 1 + wxALIGN_RIGHT + 0 - bSizer_TabsForFutureUse - wxVERTICAL + bSizer65 + wxHORIZONTAL none 5 - wxEXPAND | wxALL - 1 - + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND + 0 + + 0 1 - 1 - wxID_ANY + 0 + wxID_OK + OK - m_panel9 + m_buttonOK protected - + -1,-1 + - wxTAB_TRAVERSAL + + OnButtonOK @@ -1050,229 +1070,103 @@ - - - bSizer159 - wxVERTICAL - none - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_listCtrlEscrows - public - - - wxLC_NO_SORT_HEADER|wxLC_REPORT - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxEXPAND | wxALL - 1 - - - - 1 - - - 1 - wxID_ANY - - - m_panel8 - protected - - - - - - - wxTAB_TRAVERSAL - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer158 - wxVERTICAL - none - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_listCtrlOrdersSent - public - - - wxLC_NO_SORT_HEADER|wxLC_REPORT - - - - - - - - - - - - - - - - - - - - - - - - - - - - - OnListItemActivatedOrdersSent - - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + 1 + + + + 0 + wxID_ANY + + + COptionsDialogBase + + 540,360 + wxDEFAULT_DIALOG_STYLE + + Options + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer55 + wxVERTICAL + none + + 9 + wxEXPAND|wxALL + 1 + + + bSizer66 + wxHORIZONTAL + none 5 - wxEXPAND | wxALL - 1 - + wxEXPAND|wxRIGHT + 0 + + 1 - 1 + 0 wxID_ANY - m_panel10 + m_listBox protected - + 110,-1 + wxLB_NEEDED_SB|wxLB_SINGLE - wxTAB_TRAVERSAL + @@ -1283,6 +1177,8 @@ + OnListBox + @@ -1296,106 +1192,33 @@ - - - bSizer160 - wxVERTICAL - none - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_listCtrlProductsSent - public - - - wxLC_NO_SORT_HEADER|wxLC_REPORT - - - - - - - - - - - - - - - - - - - - - - - - - - - - - OnListItemActivatedProductsSent - - - - - - - - - - - - - - - - - - - - - - 5 - wxEXPAND | wxALL + wxEXPAND|wxLEFT 1 - + 1 - 1 + 0 wxID_ANY - m_panel11 + m_scrolledWindow protected + 5 + 5 - wxTAB_TRAVERSAL + @@ -1421,14 +1244,14 @@ - bSizer161 + bSizer63 wxVERTICAL none 5 - wxALL|wxEXPAND - 1 - + wxEXPAND + 0 + 1 @@ -1438,447 +1261,15 @@ wxID_ANY - m_listCtrlOrdersReceived - public + m_panelMain + protected - wxLC_NO_SORT_HEADER|wxLC_REPORT - - - - - - - - - - - - - - - - - - - - - - - - - OnListItemActivatedOrdersReceived - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 1 - - - - 0 - wxID_ANY - - - CTxDetailsDialogBase - - 620,450 - wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER - - Transaction Details - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer64 - wxVERTICAL - none - - 5 - wxEXPAND - 1 - - - bSizer66 - wxVERTICAL - none - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_htmlWin - protected - - - wxHW_SCROLLBAR_AUTO - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_RIGHT - 0 - - - bSizer65 - wxHORIZONTAL - none - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_OK - OK - - - m_buttonOK - protected - - -1,-1 - - - - - - - OnButtonOK - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 1 - - - - 0 - wxID_ANY - - - COptionsDialogBase - - 540,360 - wxDEFAULT_DIALOG_STYLE - - Options - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer55 - wxVERTICAL - none - - 9 - wxEXPAND|wxALL - 1 - - - bSizer66 - wxHORIZONTAL - none - - 5 - wxEXPAND|wxRIGHT - 0 - - - - - 1 - - - 0 - wxID_ANY - - - m_listBox - protected - - 110,-1 - wxLB_NEEDED_SB|wxLB_SINGLE - - - - - - - - - - - - - - - - OnListBox - - - - - - - - - - - - - - - - - - 5 - wxEXPAND|wxLEFT - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_scrolledWindow - protected - - 5 - 5 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer63 - wxVERTICAL - none - - 5 - wxEXPAND - 0 - - - - 1 - - - 0 - wxID_ANY - - - m_panelMain - protected - - - - - - - wxTAB_TRAVERSAL + wxTAB_TRAVERSAL @@ -3165,7 +2556,7 @@ CAboutDialogBase - 507,298 + 532,329 wxDEFAULT_DIALOG_STYLE About Bitcoin @@ -3203,126 +2594,237 @@ - + - bSizer60 - wxVERTICAL + bSizer63 + wxHORIZONTAL none 5 - wxEXPAND + + 0 + + + xpm/about.xpm; Load From File + + 1 + + + 0 + wxID_ANY + + + m_bitmap + protected + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND|wxLEFT 1 - bSizer62 - wxHORIZONTAL + bSizer60 + wxVERTICAL none - - 5 - wxEXPAND - 0 - - 0 - protected - 60 - - 5 wxEXPAND 1 - bSizer63 - wxVERTICAL - none - - 5 - wxEXPAND - 0 - - 50 - protected - 0 - - + bSizer62 + wxHORIZONTAL + none 5 wxEXPAND - 0 + 1 - bSizer64 - wxHORIZONTAL + bSizer631 + wxVERTICAL none 5 - wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxLEFT + wxEXPAND 0 - - - - 1 - - Tahoma,90,92,10,74,0 - 0 - wxID_ANY - Bitcoin - + + 65 + protected + 0 + + + + 5 + wxEXPAND + 0 + - m_staticText40 + bSizer64 + wxHORIZONTAL + none + + 5 + wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxLEFT + 0 + + + + 1 + + Tahoma,90,92,10,74,0 + 0 + wxID_ANY + Bitcoin + + + m_staticText40 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxRIGHT + 0 + + + + 1 + + Tahoma,90,90,10,74,0 + 0 + wxID_ANY + version + + + m_staticTextVersion + public + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + 4 protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - + 0 5 - wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxRIGHT + wxALL 0 1 - Tahoma,90,90,10,74,0 + 0 wxID_ANY - version + Copyright © 2009-2010 Satoshi Nakamoto. This is experimental software. Do not rely on it for actual financial transactions. Distributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php. This product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com). - m_staticTextVersion - public + m_staticTextMain + protected @@ -3357,44 +2859,66 @@ + + 5 + wxEXPAND + 1 + + 0 + protected + 0 + + + + + + 5 + wxALIGN_RIGHT|wxEXPAND|wxRIGHT + 0 + + + bSizer61 + wxHORIZONTAL + none 5 wxEXPAND - 0 + 1 - 4 + 0 protected 0 5 - wxALL + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND 0 - + + 0 1 0 - wxID_ANY - Copyright © 2009 Satoshi Nakamoto. This is experimental software. Do not rely on it for actual financial transactions. Distributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php. This product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com). + wxID_OK + OK - m_staticTextMain + m_buttonOK protected - + -1,-1 - 400 + OnButtonOK @@ -3420,96 +2944,13 @@ - - 5 - wxEXPAND - 1 - - 0 - protected - 0 - - - - - - - - 5 - wxALIGN_RIGHT|wxEXPAND - 0 - - - bSizer61 - wxHORIZONTAL - none - - 5 - wxEXPAND - 1 - - 0 - protected - 0 - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_OK - OK - - - m_buttonOK - protected - - -1,-1 - - - - - - - OnButtonOK - - - - - - - - - - - - - - - - - - - - - - - - + diff --git a/xpm/about.xpm b/xpm/about.xpm new file mode 100644 index 00000000..3fa868ca --- /dev/null +++ b/xpm/about.xpm @@ -0,0 +1,665 @@ +// Copyright (c) 2009-2010 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. +/* XPM */ +static const char * about_xpm[] = { +/* columns rows colors chars-per-pixel */ +"96 564 92 1", +" c #001269", +". c #000C72", +"X c #00057F", +"o c #001175", +"O c #000B6A", +"+ c #000E84", +"@ c #000489", +"# c #001583", +"$ c #001B89", +"% c #001B99", +"& c #000B92", +"* c #00208B", +"= c #002B97", +"- c #0004A6", +"; c #001DA7", +": c #0014BC", +"> c #0019BB", +", c #0017B4", +"< c #0023A3", +"1 c #002CAA", +"2 c #0030A4", +"3 c #003BA3", +"4 c #0033AB", +"5 c #003FA8", +"6 c #0027B8", +"7 c #0035BB", +"8 c #003CBA", +"9 c #004ABD", +"0 c #001DC4", +"q c #0017CC", +"w c #000CD0", +"e c #0026C7", +"r c #0035C4", +"t c #003DC5", +"y c #0032CB", +"u c #003BCC", +"i c #002BD3", +"p c #0021DC", +"a c #0025D5", +"s c #0034D5", +"d c #003ADB", +"f c #0016F6", +"g c #0008F9", +"h c #0027E3", +"j c #003CE9", +"k c #002BF5", +"l c #0024F9", +"z c #0033F4", +"x c #0035F8", +"c c #0048CA", +"v c #0055C5", +"b c #0059C3", +"n c #0053CB", +"m c #005ACC", +"M c #004FD4", +"N c #004CDC", +"B c #0047D0", +"V c #005BD6", +"C c #0049E5", +"Z c #0042EA", +"A c #0052E4", +"S c #005CE4", +"D c #0054EC", +"F c #005EEB", +"G c #004AF5", +"H c #0051F2", +"J c #005CFA", +"K c #0058F9", +"L c #0066E4", +"P c #006BE3", +"I c #0064EC", +"U c #006DEF", +"Y c #0074EB", +"T c #0078EC", +"R c #0073E7", +"E c #0065F4", +"W c #006BF5", +"Q c #006BFB", +"! c #0066FD", +"~ c #0073F5", +"^ c #007CF3", +"/ c #0075FB", +"( c #007DFC", +") c #0084FF", +"_ c #008AFF", +"` c #0092FF", +"' c #339CFF", +"] c #33A3FF", +"[ c #33AAFF", +"{ c #66B5FF", +"} c #66BBFF", +"| c #66C0FF", +/* pixels */ +"kkkkkkkkkkkk<<<<<<<<<<<>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"hhhhhhhhhhhh>>>>>>>>>>>>rrrrrrrrrrrrVVVVVVVVVVVVLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::;;;;;;;;;;;;HHHHHHHHHHHHccccccccccccQQQQQQQQQQQQ))))))))))))''''''''''''}}}}}}}}}}}}", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"qqqqqqqqqqqqkkkkkkkkkkkk333333333333AAAAAAAAAAAARRRRRRRRRRRR))))))))))))''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"############ppppppppppppssssssssssssIIIIIIIIIIII^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"++++++++++++rrrrrrrrrrrr777777777777MMMMMMMMMMMMIIIIIIIIIIII````````````''''''''''''{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"------------$$$$$$$$$$$$999999999999JJJJJJJJJJJJTTTTTTTTTTTT____________]]]]]]]]]]]]{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@666666666666ttttttttttttWWWWWWWWWWWWPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaazzzzzzzzzzzzBBBBBBBBBBBBbbbbbbbbbbbbPPPPPPPPPPPP____________''''''''''''{{{{{{{{{{{{", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +"------------%%%%%%%%%%%%ttttttttttttNNNNNNNNNNNN^^^^^^^^^^^^))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +" 000000000000888888888888FFFFFFFFFFFF~~~~~~~~~~~~))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"++++++++++++222222222222xxxxxxxxxxxxNNNNNNNNNNNNEEEEEEEEEEEE))))))))))))''''''''''''}}}}}}}}}}}}", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"$$$$$$$$$$$$000000000000GGGGGGGGGGGGnnnnnnnnnnnnLLLLLLLLLLLL))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"ooooooooooooffffffffffffccccccccccccbbbbbbbbbbbbRRRRRRRRRRRR____________''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +"@@@@@@@@@@@@111111111111777777777777JJJJJJJJJJJJPPPPPPPPPPPP((((((((((((''''''''''''{{{{{{{{{{{{", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +" iiiiiiiiiiiiGGGGGGGGGGGGVVVVVVVVVVVV~~~~~~~~~~~~____________''''''''''''}}}}}}}}}}}}", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"------------222222222222KKKKKKKKKKKKIIIIIIIIIIIIQQQQQQQQQQQQ____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"&&&&&&&&&&&&222222222222333333333333WWWWWWWWWWWW~~~~~~~~~~~~____________''''''''''''{{{{{{{{{{{{", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"wwwwwwwwwwww============555555555555EEEEEEEEEEEEEEEEEEEEEEEE____________''''''''''''||||||||||||", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"ffffffffffff>>>>>>>>>>>>rrrrrrrrrrrrnnnnnnnnnnnn~~~~~~~~~~~~____________]]]]]]]]]]]]{{{{{{{{{{{{", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +"############$$$$$$$$$$$$CCCCCCCCCCCCEEEEEEEEEEEE(((((((((((())))))))))))''''''''''''}}}}}}}}}}}}", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +",,,,,,,,,,,,666666666666ddddddddddddHHHHHHHHHHHHEEEEEEEEEEEE____________]]]]]]]]]]]]{{{{{{{{{{{{", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"xxxxxxxxxxxxjjjjjjjjjjjjccccccccccccSSSSSSSSSSSSPPPPPPPPPPPP))))))))))))]]]]]]]]]]]]}}}}}}}}}}}}", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +"000000000000%%%%%%%%%%%%ttttttttttttmmmmmmmmmmmm////////////````````````''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +">>>>>>>>>>>>uuuuuuuuuuuuZZZZZZZZZZZZmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"OOOOOOOOOOOO444444444444888888888888KKKKKKKKKKKKTTTTTTTTTTTT))))))))))))''''''''''''{{{{{{{{{{{{", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"++++++++++++666666666666CCCCCCCCCCCCQQQQQQQQQQQQYYYYYYYYYYYY____________''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"oooooooooooo,,,,,,,,,,,,DDDDDDDDDDDDmmmmmmmmmmmmLLLLLLLLLLLL))))))))))))''''''''''''}}}}}}}}}}}}", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"::::::::::::eeeeeeeeeeee444444444444mmmmmmmmmmmm^^^^^^^^^^^^____________''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"............666666666666ZZZZZZZZZZZZbbbbbbbbbbbbPPPPPPPPPPPP````````````''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"aaaaaaaaaaaaiiiiiiiiiiiizzzzzzzzzzzzJJJJJJJJJJJJPPPPPPPPPPPP))))))))))))''''''''''''{{{{{{{{{{{{", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"............eeeeeeeeeeee444444444444IIIIIIIIIIIIWWWWWWWWWWWW))))))))))))''''''''''''}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"llllllllllll444444444444HHHHHHHHHHHHvvvvvvvvvvvv((((((((((((____________]]]]]]]]]]]]}}}}}}}}}}}}", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"qqqqqqqqqqqq111111111111ssssssssssssGGGGGGGGGGGGQQQQQQQQQQQQ____________[[[[[[[[[[[[{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppkkkkkkkkkkkkttttttttttttSSSSSSSSSSSS!!!!!!!!!!!!))))))))))))]]]]]]]]]]]]{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"ppppppppppppzzzzzzzzzzzzddddddddddddFFFFFFFFFFFFLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{", +"666666666666************777777777777MMMMMMMMMMMMLLLLLLLLLLLL````````````''''''''''''{{{{{{{{{{{{" +}; From 082e725b33addda32ec4b31eba60b47f0dc6879f Mon Sep 17 00:00:00 2001 From: sirius-m Date: Thu, 4 Feb 2010 15:31:46 +0000 Subject: [PATCH 044/133] Added some basic IPC functionality using wxServer, wxClient and wxConnection. Added the -blockamount command line option for an example of usage. git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@56 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 3 +++ ipc.cpp | 33 +++++++++++++++++++++++++++++++++ ipc.h | 28 ++++++++++++++++++++++++++++ makefile | 4 +++- makefile.unix.wx2.8 | 5 +++-- makefile.unix.wx2.9 | 7 ++++--- ui.cpp | 29 +++++++++++++++++++++++++++-- 7 files changed, 101 insertions(+), 8 deletions(-) create mode 100644 ipc.cpp create mode 100644 ipc.h diff --git a/headers.h b/headers.h index 45be4b65..fb5e3ecc 100644 --- a/headers.h +++ b/headers.h @@ -24,6 +24,8 @@ #include #include #include +#include +#include #include #include #include @@ -102,6 +104,7 @@ using namespace boost; #include "market.h" #include "uibase.h" #include "ui.h" +#include "ipc.h" #include "xpm/addressbook16.xpm" #include "xpm/addressbook20.xpm" diff --git a/ipc.cpp b/ipc.cpp new file mode 100644 index 00000000..86e57674 --- /dev/null +++ b/ipc.cpp @@ -0,0 +1,33 @@ +/* + * Inter-process calling functionality + */ + +#include "headers.h" + +wxConnectionBase * CServer::OnAcceptConnection (const wxString &topic) { + return new CServerConnection; +} + +wxConnectionBase * CClient::OnMakeConnection () { + return new CClientConnection; +} + +// For request based handling +const void * CServerConnection::OnRequest (const wxString &topic, const wxString &item, size_t *size, wxIPCFormat format) { + const char * output; + + if (item == "blockamount") { + stringstream stream; + stream << nBestHeight + 1; + output = stream.str().c_str(); + } + else + output = "Unknown identifier"; + + return output; +} + +// For event based handling +bool CClientConnection::OnAdvise (const wxString &topic, const wxString &item, const void *data, size_t size, wxIPCFormat format) { + return false; +} \ No newline at end of file diff --git a/ipc.h b/ipc.h new file mode 100644 index 00000000..777d31b2 --- /dev/null +++ b/ipc.h @@ -0,0 +1,28 @@ +#ifndef _IPC_H +#define _IPC_H + +class CServer : public wxServer { +public: + wxConnectionBase * OnAcceptConnection (const wxString &topic); +}; + +class CClient : public wxClient { +public: + wxConnectionBase * OnMakeConnection (); +}; + +class CServerConnection : public wxConnection { +public: + const void * OnRequest (const wxString &topic, const wxString &item, size_t *size, wxIPCFormat format); +}; + +class CClientConnection : public wxConnection { +public: + CClientConnection() : wxConnection() {} + ~CClientConnection() {} + + bool OnAdvise (const wxString &topic, const wxString &item, const void *data, size_t size, wxIPCFormat format); +}; + +#endif /* _IPC_H */ + diff --git a/makefile b/makefile index 0dd62210..efdfc7ae 100644 --- a/makefile +++ b/makefile @@ -67,10 +67,12 @@ obj/irc.o: irc.cpp $(HEADERS) obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp windres $(WXDEFS) $(INCLUDEPATHS) -o $@ -i $< +obj/ipc.o: ipc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o obj/ipc.o bitcoin.exe: headers.h.gch $(OBJS) -kill /f bitcoin.exe diff --git a/makefile.unix.wx2.8 b/makefile.unix.wx2.8 index b9826d6c..899ce295 100644 --- a/makefile.unix.wx2.8 +++ b/makefile.unix.wx2.8 @@ -75,11 +75,12 @@ obj/sha.o: sha.cpp sha.h obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< - +obj/ipc.o: ipc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ipc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/makefile.unix.wx2.9 b/makefile.unix.wx2.9 index 81dcbd70..d2c4eb5e 100644 --- a/makefile.unix.wx2.9 +++ b/makefile.unix.wx2.9 @@ -31,7 +31,7 @@ LIBS= \ -l wx_gtk2u$(D)-2.9 \ -Wl,-Bdynamic \ -l crypto \ - -l gtk-x11-2.0 -l gthread-2.0 -l SM + -l gtk-x11-2.0 -l gthread-2.0 -l SM \ WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) @@ -75,11 +75,12 @@ obj/sha.o: sha.cpp sha.h obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< - +obj/ipc.o: ipc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ipc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/ui.cpp b/ui.cpp index 5d93ad22..1f221703 100644 --- a/ui.cpp +++ b/ui.cpp @@ -21,6 +21,7 @@ DEFINE_EVENT_TYPE(wxEVT_REPLY3) CMainFrame* pframeMain = NULL; CMyTaskBarIcon* ptaskbaricon = NULL; +CServer* pserver = NULL; map mapAddressBook; bool fRandSendTest = false; void RandSend(); @@ -384,6 +385,8 @@ CMainFrame::~CMainFrame() pframeMain = NULL; delete ptaskbaricon; ptaskbaricon = NULL; + delete pserver; + pserver = NULL; } void ExitTimeout(void* parg) @@ -1687,8 +1690,8 @@ CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) #if !wxUSE_UNICODE // Workaround until upgrade to wxWidgets supporting UTF-8 wxString str = m_staticTextMain->GetLabel(); - if (str.Find('Â') != wxNOT_FOUND) - str.Remove(str.Find('Â'), 1); + if (str.Find('�') != wxNOT_FOUND) + str.Remove(str.Find('�'), 1); m_staticTextMain->SetLabel(str); #endif #ifndef __WXMSW__ @@ -3548,6 +3551,26 @@ bool CMyApp::OnInit2() return false; } + if (mapArgs.count("-blockamount")) { + CClient client; + wxString hostname = "localhost"; + wxString server = GetDataDir() + "service"; + CClientConnection * pconnection = (CClientConnection *)client.MakeConnection(hostname, server, "ipc test"); + string output = ""; + if (pconnection) { + char * pbuffer = (char *)pconnection->Request("blockamount"); + while (*pbuffer != '\n') { + output += *pbuffer; + pbuffer++; + } + } + else { + output = "Cannot access Bitcoin. Are you sure the program is running?\n"; + } + fprintf(stderr, "%s", output.c_str()); + return false; + } + if (mapArgs.count("-datadir")) strlcpy(pszSetDataDir, mapArgs["-datadir"].c_str(), sizeof(pszSetDataDir)); @@ -3755,6 +3778,8 @@ bool CMyApp::OnInit2() if (fFirstRun) SetStartOnSystemStartup(true); + pserver = new CServer; + pserver->Create(GetDataDir() + "service"); // // Tests From c41226d847c7d32ade7f02c0d53c85b0800fc71b Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 10 Feb 2010 19:33:04 +0000 Subject: [PATCH 045/133] revert revision 56, going in different direction with boost::asio and JSON-RPC git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@57 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 3 --- ipc.cpp | 33 --------------------------------- ipc.h | 28 ---------------------------- makefile | 4 +--- makefile.unix.wx2.8 | 5 ++--- makefile.unix.wx2.9 | 7 +++---- ui.cpp | 29 ++--------------------------- 7 files changed, 8 insertions(+), 101 deletions(-) delete mode 100644 ipc.cpp delete mode 100644 ipc.h diff --git a/headers.h b/headers.h index fb5e3ecc..45be4b65 100644 --- a/headers.h +++ b/headers.h @@ -24,8 +24,6 @@ #include #include #include -#include -#include #include #include #include @@ -104,7 +102,6 @@ using namespace boost; #include "market.h" #include "uibase.h" #include "ui.h" -#include "ipc.h" #include "xpm/addressbook16.xpm" #include "xpm/addressbook20.xpm" diff --git a/ipc.cpp b/ipc.cpp deleted file mode 100644 index 86e57674..00000000 --- a/ipc.cpp +++ /dev/null @@ -1,33 +0,0 @@ -/* - * Inter-process calling functionality - */ - -#include "headers.h" - -wxConnectionBase * CServer::OnAcceptConnection (const wxString &topic) { - return new CServerConnection; -} - -wxConnectionBase * CClient::OnMakeConnection () { - return new CClientConnection; -} - -// For request based handling -const void * CServerConnection::OnRequest (const wxString &topic, const wxString &item, size_t *size, wxIPCFormat format) { - const char * output; - - if (item == "blockamount") { - stringstream stream; - stream << nBestHeight + 1; - output = stream.str().c_str(); - } - else - output = "Unknown identifier"; - - return output; -} - -// For event based handling -bool CClientConnection::OnAdvise (const wxString &topic, const wxString &item, const void *data, size_t size, wxIPCFormat format) { - return false; -} \ No newline at end of file diff --git a/ipc.h b/ipc.h deleted file mode 100644 index 777d31b2..00000000 --- a/ipc.h +++ /dev/null @@ -1,28 +0,0 @@ -#ifndef _IPC_H -#define _IPC_H - -class CServer : public wxServer { -public: - wxConnectionBase * OnAcceptConnection (const wxString &topic); -}; - -class CClient : public wxClient { -public: - wxConnectionBase * OnMakeConnection (); -}; - -class CServerConnection : public wxConnection { -public: - const void * OnRequest (const wxString &topic, const wxString &item, size_t *size, wxIPCFormat format); -}; - -class CClientConnection : public wxConnection { -public: - CClientConnection() : wxConnection() {} - ~CClientConnection() {} - - bool OnAdvise (const wxString &topic, const wxString &item, const void *data, size_t size, wxIPCFormat format); -}; - -#endif /* _IPC_H */ - diff --git a/makefile b/makefile index efdfc7ae..0dd62210 100644 --- a/makefile +++ b/makefile @@ -67,12 +67,10 @@ obj/irc.o: irc.cpp $(HEADERS) obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp windres $(WXDEFS) $(INCLUDEPATHS) -o $@ -i $< -obj/ipc.o: ipc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o obj/ipc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o bitcoin.exe: headers.h.gch $(OBJS) -kill /f bitcoin.exe diff --git a/makefile.unix.wx2.8 b/makefile.unix.wx2.8 index 899ce295..b9826d6c 100644 --- a/makefile.unix.wx2.8 +++ b/makefile.unix.wx2.8 @@ -75,12 +75,11 @@ obj/sha.o: sha.cpp sha.h obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/ipc.o: ipc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< + OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ipc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/makefile.unix.wx2.9 b/makefile.unix.wx2.9 index d2c4eb5e..81dcbd70 100644 --- a/makefile.unix.wx2.9 +++ b/makefile.unix.wx2.9 @@ -31,7 +31,7 @@ LIBS= \ -l wx_gtk2u$(D)-2.9 \ -Wl,-Bdynamic \ -l crypto \ - -l gtk-x11-2.0 -l gthread-2.0 -l SM \ + -l gtk-x11-2.0 -l gthread-2.0 -l SM WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) @@ -75,12 +75,11 @@ obj/sha.o: sha.cpp sha.h obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/ipc.o: ipc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< + OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ipc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/ui.cpp b/ui.cpp index 1f221703..5d93ad22 100644 --- a/ui.cpp +++ b/ui.cpp @@ -21,7 +21,6 @@ DEFINE_EVENT_TYPE(wxEVT_REPLY3) CMainFrame* pframeMain = NULL; CMyTaskBarIcon* ptaskbaricon = NULL; -CServer* pserver = NULL; map mapAddressBook; bool fRandSendTest = false; void RandSend(); @@ -385,8 +384,6 @@ CMainFrame::~CMainFrame() pframeMain = NULL; delete ptaskbaricon; ptaskbaricon = NULL; - delete pserver; - pserver = NULL; } void ExitTimeout(void* parg) @@ -1690,8 +1687,8 @@ CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) #if !wxUSE_UNICODE // Workaround until upgrade to wxWidgets supporting UTF-8 wxString str = m_staticTextMain->GetLabel(); - if (str.Find('�') != wxNOT_FOUND) - str.Remove(str.Find('�'), 1); + if (str.Find('Â') != wxNOT_FOUND) + str.Remove(str.Find('Â'), 1); m_staticTextMain->SetLabel(str); #endif #ifndef __WXMSW__ @@ -3551,26 +3548,6 @@ bool CMyApp::OnInit2() return false; } - if (mapArgs.count("-blockamount")) { - CClient client; - wxString hostname = "localhost"; - wxString server = GetDataDir() + "service"; - CClientConnection * pconnection = (CClientConnection *)client.MakeConnection(hostname, server, "ipc test"); - string output = ""; - if (pconnection) { - char * pbuffer = (char *)pconnection->Request("blockamount"); - while (*pbuffer != '\n') { - output += *pbuffer; - pbuffer++; - } - } - else { - output = "Cannot access Bitcoin. Are you sure the program is running?\n"; - } - fprintf(stderr, "%s", output.c_str()); - return false; - } - if (mapArgs.count("-datadir")) strlcpy(pszSetDataDir, mapArgs["-datadir"].c_str(), sizeof(pszSetDataDir)); @@ -3778,8 +3755,6 @@ bool CMyApp::OnInit2() if (fFirstRun) SetStartOnSystemStartup(true); - pserver = new CServer; - pserver->Create(GetDataDir() + "service"); // // Tests From 75990a46a7e4999729ff67c4268c2241c98e486f Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 10 Feb 2010 19:41:22 +0000 Subject: [PATCH 046/133] JSON Spirit library from http://www.codeproject.com/KB/recipes/JSON_Spirit.aspx, MIT license git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@58 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- json/LICENSE.txt | 24 ++ json/json_spirit.h | 18 + json/json_spirit_error_position.h | 54 +++ json/json_spirit_reader.cpp | 137 +++++++ json/json_spirit_reader.h | 62 +++ json/json_spirit_reader_template.h | 612 +++++++++++++++++++++++++++++ json/json_spirit_stream_reader.h | 70 ++++ json/json_spirit_utils.h | 61 +++ json/json_spirit_value.cpp | 8 + json/json_spirit_value.h | 532 +++++++++++++++++++++++++ json/json_spirit_writer.cpp | 95 +++++ json/json_spirit_writer.h | 50 +++ json/json_spirit_writer_template.h | 245 ++++++++++++ 13 files changed, 1968 insertions(+) create mode 100644 json/LICENSE.txt create mode 100644 json/json_spirit.h create mode 100644 json/json_spirit_error_position.h create mode 100644 json/json_spirit_reader.cpp create mode 100644 json/json_spirit_reader.h create mode 100644 json/json_spirit_reader_template.h create mode 100644 json/json_spirit_stream_reader.h create mode 100644 json/json_spirit_utils.h create mode 100644 json/json_spirit_value.cpp create mode 100644 json/json_spirit_value.h create mode 100644 json/json_spirit_writer.cpp create mode 100644 json/json_spirit_writer.h create mode 100644 json/json_spirit_writer_template.h diff --git a/json/LICENSE.txt b/json/LICENSE.txt new file mode 100644 index 00000000..fa193fe7 --- /dev/null +++ b/json/LICENSE.txt @@ -0,0 +1,24 @@ +The MIT License + +Copyright (c) 2007 - 2009 John W. Wilkinson + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/json/json_spirit.h b/json/json_spirit.h new file mode 100644 index 00000000..7dac05c3 --- /dev/null +++ b/json/json_spirit.h @@ -0,0 +1,18 @@ +#ifndef JSON_SPIRIT +#define JSON_SPIRIT + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#if defined(_MSC_VER) && (_MSC_VER >= 1020) +# pragma once +#endif + +#include "json_spirit_value.h" +#include "json_spirit_reader.h" +#include "json_spirit_writer.h" +#include "json_spirit_utils.h" + +#endif diff --git a/json/json_spirit_error_position.h b/json/json_spirit_error_position.h new file mode 100644 index 00000000..4a535ff5 --- /dev/null +++ b/json/json_spirit_error_position.h @@ -0,0 +1,54 @@ +#ifndef JSON_SPIRIT_ERROR_POSITION +#define JSON_SPIRIT_ERROR_POSITION + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#if defined(_MSC_VER) && (_MSC_VER >= 1020) +# pragma once +#endif + +#include + +namespace json_spirit +{ + // An Error_position exception is thrown by the "read_or_throw" functions below on finding an error. + // Note the "read_or_throw" functions are around 3 times slower than the standard functions "read" + // functions that return a bool. + // + struct Error_position + { + Error_position(); + Error_position( unsigned int line, unsigned int column, const std::string& reason ); + bool operator==( const Error_position& lhs ) const; + unsigned int line_; + unsigned int column_; + std::string reason_; + }; + + inline Error_position::Error_position() + : line_( 0 ) + , column_( 0 ) + { + } + + inline Error_position::Error_position( unsigned int line, unsigned int column, const std::string& reason ) + : line_( line ) + , column_( column ) + , reason_( reason ) + { + } + + inline bool Error_position::operator==( const Error_position& lhs ) const + { + if( this == &lhs ) return true; + + return ( reason_ == lhs.reason_ ) && + ( line_ == lhs.line_ ) && + ( column_ == lhs.column_ ); +} +} + +#endif diff --git a/json/json_spirit_reader.cpp b/json/json_spirit_reader.cpp new file mode 100644 index 00000000..8e2fb5e2 --- /dev/null +++ b/json/json_spirit_reader.cpp @@ -0,0 +1,137 @@ +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#include "json_spirit_reader.h" +#include "json_spirit_reader_template.h" + +using namespace json_spirit; + +bool json_spirit::read( const std::string& s, Value& value ) +{ + return read_string( s, value ); +} + +void json_spirit::read_or_throw( const std::string& s, Value& value ) +{ + read_string_or_throw( s, value ); +} + +bool json_spirit::read( std::istream& is, Value& value ) +{ + return read_stream( is, value ); +} + +void json_spirit::read_or_throw( std::istream& is, Value& value ) +{ + read_stream_or_throw( is, value ); +} + +bool json_spirit::read( std::string::const_iterator& begin, std::string::const_iterator end, Value& value ) +{ + return read_range( begin, end, value ); +} + +void json_spirit::read_or_throw( std::string::const_iterator& begin, std::string::const_iterator end, Value& value ) +{ + begin = read_range_or_throw( begin, end, value ); +} + +#ifndef BOOST_NO_STD_WSTRING + +bool json_spirit::read( const std::wstring& s, wValue& value ) +{ + return read_string( s, value ); +} + +void json_spirit::read_or_throw( const std::wstring& s, wValue& value ) +{ + read_string_or_throw( s, value ); +} + +bool json_spirit::read( std::wistream& is, wValue& value ) +{ + return read_stream( is, value ); +} + +void json_spirit::read_or_throw( std::wistream& is, wValue& value ) +{ + read_stream_or_throw( is, value ); +} + +bool json_spirit::read( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wValue& value ) +{ + return read_range( begin, end, value ); +} + +void json_spirit::read_or_throw( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wValue& value ) +{ + begin = read_range_or_throw( begin, end, value ); +} + +#endif + +bool json_spirit::read( const std::string& s, mValue& value ) +{ + return read_string( s, value ); +} + +void json_spirit::read_or_throw( const std::string& s, mValue& value ) +{ + read_string_or_throw( s, value ); +} + +bool json_spirit::read( std::istream& is, mValue& value ) +{ + return read_stream( is, value ); +} + +void json_spirit::read_or_throw( std::istream& is, mValue& value ) +{ + read_stream_or_throw( is, value ); +} + +bool json_spirit::read( std::string::const_iterator& begin, std::string::const_iterator end, mValue& value ) +{ + return read_range( begin, end, value ); +} + +void json_spirit::read_or_throw( std::string::const_iterator& begin, std::string::const_iterator end, mValue& value ) +{ + begin = read_range_or_throw( begin, end, value ); +} + +#ifndef BOOST_NO_STD_WSTRING + +bool json_spirit::read( const std::wstring& s, wmValue& value ) +{ + return read_string( s, value ); +} + +void json_spirit::read_or_throw( const std::wstring& s, wmValue& value ) +{ + read_string_or_throw( s, value ); +} + +bool json_spirit::read( std::wistream& is, wmValue& value ) +{ + return read_stream( is, value ); +} + +void json_spirit::read_or_throw( std::wistream& is, wmValue& value ) +{ + read_stream_or_throw( is, value ); +} + +bool json_spirit::read( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wmValue& value ) +{ + return read_range( begin, end, value ); +} + +void json_spirit::read_or_throw( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wmValue& value ) +{ + begin = read_range_or_throw( begin, end, value ); +} + +#endif diff --git a/json/json_spirit_reader.h b/json/json_spirit_reader.h new file mode 100644 index 00000000..a58bfc10 --- /dev/null +++ b/json/json_spirit_reader.h @@ -0,0 +1,62 @@ +#ifndef JSON_SPIRIT_READER +#define JSON_SPIRIT_READER + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#if defined(_MSC_VER) && (_MSC_VER >= 1020) +# pragma once +#endif + +#include "json_spirit_value.h" +#include "json_spirit_error_position.h" +#include + +namespace json_spirit +{ + // functions to reads a JSON values + + bool read( const std::string& s, Value& value ); + bool read( std::istream& is, Value& value ); + bool read( std::string::const_iterator& begin, std::string::const_iterator end, Value& value ); + + void read_or_throw( const std::string& s, Value& value ); + void read_or_throw( std::istream& is, Value& value ); + void read_or_throw( std::string::const_iterator& begin, std::string::const_iterator end, Value& value ); + +#ifndef BOOST_NO_STD_WSTRING + + bool read( const std::wstring& s, wValue& value ); + bool read( std::wistream& is, wValue& value ); + bool read( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wValue& value ); + + void read_or_throw( const std::wstring& s, wValue& value ); + void read_or_throw( std::wistream& is, wValue& value ); + void read_or_throw( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wValue& value ); + +#endif + + bool read( const std::string& s, mValue& value ); + bool read( std::istream& is, mValue& value ); + bool read( std::string::const_iterator& begin, std::string::const_iterator end, mValue& value ); + + void read_or_throw( const std::string& s, mValue& value ); + void read_or_throw( std::istream& is, mValue& value ); + void read_or_throw( std::string::const_iterator& begin, std::string::const_iterator end, mValue& value ); + +#ifndef BOOST_NO_STD_WSTRING + + bool read( const std::wstring& s, wmValue& value ); + bool read( std::wistream& is, wmValue& value ); + bool read( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wmValue& value ); + + void read_or_throw( const std::wstring& s, wmValue& value ); + void read_or_throw( std::wistream& is, wmValue& value ); + void read_or_throw( std::wstring::const_iterator& begin, std::wstring::const_iterator end, wmValue& value ); + +#endif +} + +#endif diff --git a/json/json_spirit_reader_template.h b/json/json_spirit_reader_template.h new file mode 100644 index 00000000..81cded43 --- /dev/null +++ b/json/json_spirit_reader_template.h @@ -0,0 +1,612 @@ +#ifndef JSON_SPIRIT_READER_TEMPLATE +#define JSON_SPIRIT_READER_TEMPLATE + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#include "json_spirit_value.h" +#include "json_spirit_error_position.h" + +//#define BOOST_SPIRIT_THREADSAFE // uncomment for multithreaded use, requires linking to boost.thread + +#include +#include +#include + +#if BOOST_VERSION >= 103800 + #include + #include + #include + #include + #include + #define spirit_namespace boost::spirit::classic +#else + #include + #include + #include + #include + #include + #define spirit_namespace boost::spirit +#endif + +namespace json_spirit +{ + const spirit_namespace::int_parser < boost::int64_t > int64_p = spirit_namespace::int_parser < boost::int64_t >(); + const spirit_namespace::uint_parser< boost::uint64_t > uint64_p = spirit_namespace::uint_parser< boost::uint64_t >(); + + template< class Iter_type > + bool is_eq( Iter_type first, Iter_type last, const char* c_str ) + { + for( Iter_type i = first; i != last; ++i, ++c_str ) + { + if( *c_str == 0 ) return false; + + if( *i != *c_str ) return false; + } + + return true; + } + + template< class Char_type > + Char_type hex_to_num( const Char_type c ) + { + if( ( c >= '0' ) && ( c <= '9' ) ) return c - '0'; + if( ( c >= 'a' ) && ( c <= 'f' ) ) return c - 'a' + 10; + if( ( c >= 'A' ) && ( c <= 'F' ) ) return c - 'A' + 10; + return 0; + } + + template< class Char_type, class Iter_type > + Char_type hex_str_to_char( Iter_type& begin ) + { + const Char_type c1( *( ++begin ) ); + const Char_type c2( *( ++begin ) ); + + return ( hex_to_num( c1 ) << 4 ) + hex_to_num( c2 ); + } + + template< class Char_type, class Iter_type > + Char_type unicode_str_to_char( Iter_type& begin ) + { + const Char_type c1( *( ++begin ) ); + const Char_type c2( *( ++begin ) ); + const Char_type c3( *( ++begin ) ); + const Char_type c4( *( ++begin ) ); + + return ( hex_to_num( c1 ) << 12 ) + + ( hex_to_num( c2 ) << 8 ) + + ( hex_to_num( c3 ) << 4 ) + + hex_to_num( c4 ); + } + + template< class String_type > + void append_esc_char_and_incr_iter( String_type& s, + typename String_type::const_iterator& begin, + typename String_type::const_iterator end ) + { + typedef typename String_type::value_type Char_type; + + const Char_type c2( *begin ); + + switch( c2 ) + { + case 't': s += '\t'; break; + case 'b': s += '\b'; break; + case 'f': s += '\f'; break; + case 'n': s += '\n'; break; + case 'r': s += '\r'; break; + case '\\': s += '\\'; break; + case '/': s += '/'; break; + case '"': s += '"'; break; + case 'x': + { + if( end - begin >= 3 ) // expecting "xHH..." + { + s += hex_str_to_char< Char_type >( begin ); + } + break; + } + case 'u': + { + if( end - begin >= 5 ) // expecting "uHHHH..." + { + s += unicode_str_to_char< Char_type >( begin ); + } + break; + } + } + } + + template< class String_type > + String_type substitute_esc_chars( typename String_type::const_iterator begin, + typename String_type::const_iterator end ) + { + typedef typename String_type::const_iterator Iter_type; + + if( end - begin < 2 ) return String_type( begin, end ); + + String_type result; + + result.reserve( end - begin ); + + const Iter_type end_minus_1( end - 1 ); + + Iter_type substr_start = begin; + Iter_type i = begin; + + for( ; i < end_minus_1; ++i ) + { + if( *i == '\\' ) + { + result.append( substr_start, i ); + + ++i; // skip the '\' + + append_esc_char_and_incr_iter( result, i, end ); + + substr_start = i + 1; + } + } + + result.append( substr_start, end ); + + return result; + } + + template< class String_type > + String_type get_str_( typename String_type::const_iterator begin, + typename String_type::const_iterator end ) + { + assert( end - begin >= 2 ); + + typedef typename String_type::const_iterator Iter_type; + + Iter_type str_without_quotes( ++begin ); + Iter_type end_without_quotes( --end ); + + return substitute_esc_chars< String_type >( str_without_quotes, end_without_quotes ); + } + + inline std::string get_str( std::string::const_iterator begin, std::string::const_iterator end ) + { + return get_str_< std::string >( begin, end ); + } + + inline std::wstring get_str( std::wstring::const_iterator begin, std::wstring::const_iterator end ) + { + return get_str_< std::wstring >( begin, end ); + } + + template< class String_type, class Iter_type > + String_type get_str( Iter_type begin, Iter_type end ) + { + const String_type tmp( begin, end ); // convert multipass iterators to string iterators + + return get_str( tmp.begin(), tmp.end() ); + } + + // this class's methods get called by the spirit parse resulting + // in the creation of a JSON object or array + // + // NB Iter_type could be a std::string iterator, wstring iterator, a position iterator or a multipass iterator + // + template< class Value_type, class Iter_type > + class Semantic_actions + { + public: + + typedef typename Value_type::Config_type Config_type; + typedef typename Config_type::String_type String_type; + typedef typename Config_type::Object_type Object_type; + typedef typename Config_type::Array_type Array_type; + typedef typename String_type::value_type Char_type; + + Semantic_actions( Value_type& value ) + : value_( value ) + , current_p_( 0 ) + { + } + + void begin_obj( Char_type c ) + { + assert( c == '{' ); + + begin_compound< Object_type >(); + } + + void end_obj( Char_type c ) + { + assert( c == '}' ); + + end_compound(); + } + + void begin_array( Char_type c ) + { + assert( c == '[' ); + + begin_compound< Array_type >(); + } + + void end_array( Char_type c ) + { + assert( c == ']' ); + + end_compound(); + } + + void new_name( Iter_type begin, Iter_type end ) + { + assert( current_p_->type() == obj_type ); + + name_ = get_str< String_type >( begin, end ); + } + + void new_str( Iter_type begin, Iter_type end ) + { + add_to_current( get_str< String_type >( begin, end ) ); + } + + void new_true( Iter_type begin, Iter_type end ) + { + assert( is_eq( begin, end, "true" ) ); + + add_to_current( true ); + } + + void new_false( Iter_type begin, Iter_type end ) + { + assert( is_eq( begin, end, "false" ) ); + + add_to_current( false ); + } + + void new_null( Iter_type begin, Iter_type end ) + { + assert( is_eq( begin, end, "null" ) ); + + add_to_current( Value_type() ); + } + + void new_int( boost::int64_t i ) + { + add_to_current( i ); + } + + void new_uint64( boost::uint64_t ui ) + { + add_to_current( ui ); + } + + void new_real( double d ) + { + add_to_current( d ); + } + + private: + + Semantic_actions& operator=( const Semantic_actions& ); + // to prevent "assignment operator could not be generated" warning + + Value_type* add_first( const Value_type& value ) + { + assert( current_p_ == 0 ); + + value_ = value; + current_p_ = &value_; + return current_p_; + } + + template< class Array_or_obj > + void begin_compound() + { + if( current_p_ == 0 ) + { + add_first( Array_or_obj() ); + } + else + { + stack_.push_back( current_p_ ); + + Array_or_obj new_array_or_obj; // avoid copy by building new array or object in place + + current_p_ = add_to_current( new_array_or_obj ); + } + } + + void end_compound() + { + if( current_p_ != &value_ ) + { + current_p_ = stack_.back(); + + stack_.pop_back(); + } + } + + Value_type* add_to_current( const Value_type& value ) + { + if( current_p_ == 0 ) + { + return add_first( value ); + } + else if( current_p_->type() == array_type ) + { + current_p_->get_array().push_back( value ); + + return ¤t_p_->get_array().back(); + } + + assert( current_p_->type() == obj_type ); + + return &Config_type::add( current_p_->get_obj(), name_, value ); + } + + Value_type& value_; // this is the object or array that is being created + Value_type* current_p_; // the child object or array that is currently being constructed + + std::vector< Value_type* > stack_; // previous child objects and arrays + + String_type name_; // of current name/value pair + }; + + template< typename Iter_type > + void throw_error( spirit_namespace::position_iterator< Iter_type > i, const std::string& reason ) + { + throw Error_position( i.get_position().line, i.get_position().column, reason ); + } + + template< typename Iter_type > + void throw_error( Iter_type i, const std::string& reason ) + { + throw reason; + } + + // the spirit grammer + // + template< class Value_type, class Iter_type > + class Json_grammer : public spirit_namespace::grammar< Json_grammer< Value_type, Iter_type > > + { + public: + + typedef Semantic_actions< Value_type, Iter_type > Semantic_actions_t; + + Json_grammer( Semantic_actions_t& semantic_actions ) + : actions_( semantic_actions ) + { + } + + static void throw_not_value( Iter_type begin, Iter_type end ) + { + throw_error( begin, "not a value" ); + } + + static void throw_not_array( Iter_type begin, Iter_type end ) + { + throw_error( begin, "not an array" ); + } + + static void throw_not_object( Iter_type begin, Iter_type end ) + { + throw_error( begin, "not an object" ); + } + + static void throw_not_pair( Iter_type begin, Iter_type end ) + { + throw_error( begin, "not a pair" ); + } + + static void throw_not_colon( Iter_type begin, Iter_type end ) + { + throw_error( begin, "no colon in pair" ); + } + + static void throw_not_string( Iter_type begin, Iter_type end ) + { + throw_error( begin, "not a string" ); + } + + template< typename ScannerT > + class definition + { + public: + + definition( const Json_grammer& self ) + { + using namespace spirit_namespace; + + typedef typename Value_type::String_type::value_type Char_type; + + // first we convert the semantic action class methods to functors with the + // parameter signature expected by spirit + + typedef boost::function< void( Char_type ) > Char_action; + typedef boost::function< void( Iter_type, Iter_type ) > Str_action; + typedef boost::function< void( double ) > Real_action; + typedef boost::function< void( boost::int64_t ) > Int_action; + typedef boost::function< void( boost::uint64_t ) > Uint64_action; + + Char_action begin_obj ( boost::bind( &Semantic_actions_t::begin_obj, &self.actions_, _1 ) ); + Char_action end_obj ( boost::bind( &Semantic_actions_t::end_obj, &self.actions_, _1 ) ); + Char_action begin_array( boost::bind( &Semantic_actions_t::begin_array, &self.actions_, _1 ) ); + Char_action end_array ( boost::bind( &Semantic_actions_t::end_array, &self.actions_, _1 ) ); + Str_action new_name ( boost::bind( &Semantic_actions_t::new_name, &self.actions_, _1, _2 ) ); + Str_action new_str ( boost::bind( &Semantic_actions_t::new_str, &self.actions_, _1, _2 ) ); + Str_action new_true ( boost::bind( &Semantic_actions_t::new_true, &self.actions_, _1, _2 ) ); + Str_action new_false ( boost::bind( &Semantic_actions_t::new_false, &self.actions_, _1, _2 ) ); + Str_action new_null ( boost::bind( &Semantic_actions_t::new_null, &self.actions_, _1, _2 ) ); + Real_action new_real ( boost::bind( &Semantic_actions_t::new_real, &self.actions_, _1 ) ); + Int_action new_int ( boost::bind( &Semantic_actions_t::new_int, &self.actions_, _1 ) ); + Uint64_action new_uint64 ( boost::bind( &Semantic_actions_t::new_uint64, &self.actions_, _1 ) ); + + // actual grammer + + json_ + = value_ | eps_p[ &throw_not_value ] + ; + + value_ + = string_[ new_str ] + | number_ + | object_ + | array_ + | str_p( "true" ) [ new_true ] + | str_p( "false" )[ new_false ] + | str_p( "null" ) [ new_null ] + ; + + object_ + = ch_p('{')[ begin_obj ] + >> !members_ + >> ( ch_p('}')[ end_obj ] | eps_p[ &throw_not_object ] ) + ; + + members_ + = pair_ >> *( ',' >> pair_ ) + ; + + pair_ + = string_[ new_name ] + >> ( ':' | eps_p[ &throw_not_colon ] ) + >> ( value_ | eps_p[ &throw_not_value ] ) + ; + + array_ + = ch_p('[')[ begin_array ] + >> !elements_ + >> ( ch_p(']')[ end_array ] | eps_p[ &throw_not_array ] ) + ; + + elements_ + = value_ >> *( ',' >> value_ ) + ; + + string_ + = lexeme_d // this causes white space inside a string to be retained + [ + confix_p + ( + '"', + *lex_escape_ch_p, + '"' + ) + ] + ; + + number_ + = strict_real_p[ new_real ] + | int64_p [ new_int ] + | uint64_p [ new_uint64 ] + ; + } + + spirit_namespace::rule< ScannerT > json_, object_, members_, pair_, array_, elements_, value_, string_, number_; + + const spirit_namespace::rule< ScannerT >& start() const { return json_; } + }; + + private: + + Json_grammer& operator=( const Json_grammer& ); // to prevent "assignment operator could not be generated" warning + + Semantic_actions_t& actions_; + }; + + template< class Iter_type, class Value_type > + Iter_type read_range_or_throw( Iter_type begin, Iter_type end, Value_type& value ) + { + Semantic_actions< Value_type, Iter_type > semantic_actions( value ); + + const spirit_namespace::parse_info< Iter_type > info = + spirit_namespace::parse( begin, end, + Json_grammer< Value_type, Iter_type >( semantic_actions ), + spirit_namespace::space_p ); + + if( !info.hit ) + { + assert( false ); // in theory exception should already have been thrown + throw_error( info.stop, "error" ); + } + + return info.stop; + } + + template< class Iter_type, class Value_type > + void add_posn_iter_and_read_range_or_throw( Iter_type begin, Iter_type end, Value_type& value ) + { + typedef spirit_namespace::position_iterator< Iter_type > Posn_iter_t; + + const Posn_iter_t posn_begin( begin, end ); + const Posn_iter_t posn_end( end, end ); + + read_range_or_throw( posn_begin, posn_end, value ); + } + + template< class Iter_type, class Value_type > + bool read_range( Iter_type& begin, Iter_type end, Value_type& value ) + { + try + { + begin = read_range_or_throw( begin, end, value ); + + return true; + } + catch( ... ) + { + return false; + } + } + + template< class String_type, class Value_type > + void read_string_or_throw( const String_type& s, Value_type& value ) + { + add_posn_iter_and_read_range_or_throw( s.begin(), s.end(), value ); + } + + template< class String_type, class Value_type > + bool read_string( const String_type& s, Value_type& value ) + { + typename String_type::const_iterator begin = s.begin(); + + return read_range( begin, s.end(), value ); + } + + template< class Istream_type > + struct Multi_pass_iters + { + typedef typename Istream_type::char_type Char_type; + typedef std::istream_iterator< Char_type, Char_type > istream_iter; + typedef spirit_namespace::multi_pass< istream_iter > Mp_iter; + + Multi_pass_iters( Istream_type& is ) + { + is.unsetf( std::ios::skipws ); + + begin_ = spirit_namespace::make_multi_pass( istream_iter( is ) ); + end_ = spirit_namespace::make_multi_pass( istream_iter() ); + } + + Mp_iter begin_; + Mp_iter end_; + }; + + template< class Istream_type, class Value_type > + bool read_stream( Istream_type& is, Value_type& value ) + { + Multi_pass_iters< Istream_type > mp_iters( is ); + + return read_range( mp_iters.begin_, mp_iters.end_, value ); + } + + template< class Istream_type, class Value_type > + void read_stream_or_throw( Istream_type& is, Value_type& value ) + { + const Multi_pass_iters< Istream_type > mp_iters( is ); + + add_posn_iter_and_read_range_or_throw( mp_iters.begin_, mp_iters.end_, value ); + } +} + +#endif diff --git a/json/json_spirit_stream_reader.h b/json/json_spirit_stream_reader.h new file mode 100644 index 00000000..a9ceeacf --- /dev/null +++ b/json/json_spirit_stream_reader.h @@ -0,0 +1,70 @@ +#ifndef JSON_SPIRIT_READ_STREAM +#define JSON_SPIRIT_READ_STREAM + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#if defined(_MSC_VER) && (_MSC_VER >= 1020) +# pragma once +#endif + +#include "json_spirit_reader_template.h" + +namespace json_spirit +{ + // these classes allows you to read multiple top level contiguous values from a stream, + // the normal stream read functions have a bug that prevent multiple top level values + // from being read unless they are separated by spaces + + template< class Istream_type, class Value_type > + class Stream_reader + { + public: + + Stream_reader( Istream_type& is ) + : iters_( is ) + { + } + + bool read_next( Value_type& value ) + { + return read_range( iters_.begin_, iters_.end_, value ); + } + + private: + + typedef Multi_pass_iters< Istream_type > Mp_iters; + + Mp_iters iters_; + }; + + template< class Istream_type, class Value_type > + class Stream_reader_thrower + { + public: + + Stream_reader_thrower( Istream_type& is ) + : iters_( is ) + , posn_begin_( iters_.begin_, iters_.end_ ) + , posn_end_( iters_.end_, iters_.end_ ) + { + } + + void read_next( Value_type& value ) + { + posn_begin_ = read_range_or_throw( posn_begin_, posn_end_, value ); + } + + private: + + typedef Multi_pass_iters< Istream_type > Mp_iters; + typedef spirit_namespace::position_iterator< typename Mp_iters::Mp_iter > Posn_iter_t; + + Mp_iters iters_; + Posn_iter_t posn_begin_, posn_end_; + }; +} + +#endif diff --git a/json/json_spirit_utils.h b/json/json_spirit_utils.h new file mode 100644 index 00000000..7eb338e7 --- /dev/null +++ b/json/json_spirit_utils.h @@ -0,0 +1,61 @@ +#ifndef JSON_SPIRIT_UTILS +#define JSON_SPIRIT_UTILS + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#if defined(_MSC_VER) && (_MSC_VER >= 1020) +# pragma once +#endif + +#include "json_spirit_value.h" +#include + +namespace json_spirit +{ + template< class Obj_t, class Map_t > + void obj_to_map( const Obj_t& obj, Map_t& mp_obj ) + { + mp_obj.clear(); + + for( typename Obj_t::const_iterator i = obj.begin(); i != obj.end(); ++i ) + { + mp_obj[ i->name_ ] = i->value_; + } + } + + template< class Obj_t, class Map_t > + void map_to_obj( const Map_t& mp_obj, Obj_t& obj ) + { + obj.clear(); + + for( typename Map_t::const_iterator i = mp_obj.begin(); i != mp_obj.end(); ++i ) + { + obj.push_back( typename Obj_t::value_type( i->first, i->second ) ); + } + } + + typedef std::map< std::string, Value > Mapped_obj; + +#ifndef BOOST_NO_STD_WSTRING + typedef std::map< std::wstring, wValue > wMapped_obj; +#endif + + template< class Object_type, class String_type > + const typename Object_type::value_type::Value_type& find_value( const Object_type& obj, const String_type& name ) + { + for( typename Object_type::const_iterator i = obj.begin(); i != obj.end(); ++i ) + { + if( i->name_ == name ) + { + return i->value_; + } + } + + return Object_type::value_type::Value_type::null; + } +} + +#endif diff --git a/json/json_spirit_value.cpp b/json/json_spirit_value.cpp new file mode 100644 index 00000000..dd5b50e5 --- /dev/null +++ b/json/json_spirit_value.cpp @@ -0,0 +1,8 @@ +/* Copyright (c) 2007 John W Wilkinson + + This source code can be used for any purpose as long as + this comment is retained. */ + +// json spirit version 2.00 + +#include "json_spirit_value.h" diff --git a/json/json_spirit_value.h b/json/json_spirit_value.h new file mode 100644 index 00000000..e8be3553 --- /dev/null +++ b/json/json_spirit_value.h @@ -0,0 +1,532 @@ +#ifndef JSON_SPIRIT_VALUE +#define JSON_SPIRIT_VALUE + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#if defined(_MSC_VER) && (_MSC_VER >= 1020) +# pragma once +#endif + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +namespace json_spirit +{ + enum Value_type{ obj_type, array_type, str_type, bool_type, int_type, real_type, null_type }; + + template< class Config > // Config determines whether the value uses std::string or std::wstring and + // whether JSON Objects are represented as vectors or maps + class Value_impl + { + public: + + typedef Config Config_type; + typedef typename Config::String_type String_type; + typedef typename Config::Object_type Object; + typedef typename Config::Array_type Array; + typedef typename String_type::const_pointer Const_str_ptr; // eg const char* + + Value_impl(); // creates null value + Value_impl( Const_str_ptr value ); + Value_impl( const String_type& value ); + Value_impl( const Object& value ); + Value_impl( const Array& value ); + Value_impl( bool value ); + Value_impl( int value ); + Value_impl( boost::int64_t value ); + Value_impl( boost::uint64_t value ); + Value_impl( double value ); + + Value_impl( const Value_impl& other ); + + bool operator==( const Value_impl& lhs ) const; + + Value_impl& operator=( const Value_impl& lhs ); + + Value_type type() const; + + bool is_uint64() const; + bool is_null() const; + + const String_type& get_str() const; + const Object& get_obj() const; + const Array& get_array() const; + bool get_bool() const; + int get_int() const; + boost::int64_t get_int64() const; + boost::uint64_t get_uint64() const; + double get_real() const; + + Object& get_obj(); + Array& get_array(); + + template< typename T > T get_value() const; // example usage: int i = value.get_value< int >(); + // or double d = value.get_value< double >(); + + static const Value_impl null; + + private: + + void check_type( const Value_type vtype ) const; + + typedef boost::variant< String_type, + boost::recursive_wrapper< Object >, boost::recursive_wrapper< Array >, + bool, boost::int64_t, double > Variant; + + Value_type type_; + Variant v_; + bool is_uint64_; + }; + + // vector objects + + template< class Config > + struct Pair_impl + { + typedef typename Config::String_type String_type; + typedef typename Config::Value_type Value_type; + + Pair_impl( const String_type& name, const Value_type& value ); + + bool operator==( const Pair_impl& lhs ) const; + + String_type name_; + Value_type value_; + }; + + template< class String > + struct Config_vector + { + typedef String String_type; + typedef Value_impl< Config_vector > Value_type; + typedef Pair_impl < Config_vector > Pair_type; + typedef std::vector< Value_type > Array_type; + typedef std::vector< Pair_type > Object_type; + + static Value_type& add( Object_type& obj, const String_type& name, const Value_type& value ) + { + obj.push_back( Pair_type( name , value ) ); + + return obj.back().value_; + } + + static String_type get_name( const Pair_type& pair ) + { + return pair.name_; + } + + static Value_type get_value( const Pair_type& pair ) + { + return pair.value_; + } + }; + + // typedefs for ASCII + + typedef Config_vector< std::string > Config; + + typedef Config::Value_type Value; + typedef Config::Pair_type Pair; + typedef Config::Object_type Object; + typedef Config::Array_type Array; + + // typedefs for Unicode + +#ifndef BOOST_NO_STD_WSTRING + + typedef Config_vector< std::wstring > wConfig; + + typedef wConfig::Value_type wValue; + typedef wConfig::Pair_type wPair; + typedef wConfig::Object_type wObject; + typedef wConfig::Array_type wArray; +#endif + + // map objects + + template< class String > + struct Config_map + { + typedef String String_type; + typedef Value_impl< Config_map > Value_type; + typedef std::vector< Value_type > Array_type; + typedef std::map< String_type, Value_type > Object_type; + typedef typename Object_type::value_type Pair_type; + + static Value_type& add( Object_type& obj, const String_type& name, const Value_type& value ) + { + return obj[ name ] = value; + } + + static String_type get_name( const Pair_type& pair ) + { + return pair.first; + } + + static Value_type get_value( const Pair_type& pair ) + { + return pair.second; + } + }; + + // typedefs for ASCII + + typedef Config_map< std::string > mConfig; + + typedef mConfig::Value_type mValue; + typedef mConfig::Object_type mObject; + typedef mConfig::Array_type mArray; + + // typedefs for Unicode + +#ifndef BOOST_NO_STD_WSTRING + + typedef Config_map< std::wstring > wmConfig; + + typedef wmConfig::Value_type wmValue; + typedef wmConfig::Object_type wmObject; + typedef wmConfig::Array_type wmArray; + +#endif + + /////////////////////////////////////////////////////////////////////////////////////////////// + // + // implementation + + template< class Config > + const Value_impl< Config > Value_impl< Config >::null; + + template< class Config > + Value_impl< Config >::Value_impl() + : type_( null_type ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( const Const_str_ptr value ) + : type_( str_type ) + , v_( String_type( value ) ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( const String_type& value ) + : type_( str_type ) + , v_( value ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( const Object& value ) + : type_( obj_type ) + , v_( value ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( const Array& value ) + : type_( array_type ) + , v_( value ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( bool value ) + : type_( bool_type ) + , v_( value ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( int value ) + : type_( int_type ) + , v_( static_cast< boost::int64_t >( value ) ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( boost::int64_t value ) + : type_( int_type ) + , v_( value ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( boost::uint64_t value ) + : type_( int_type ) + , v_( static_cast< boost::int64_t >( value ) ) + , is_uint64_( true ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( double value ) + : type_( real_type ) + , v_( value ) + , is_uint64_( false ) + { + } + + template< class Config > + Value_impl< Config >::Value_impl( const Value_impl< Config >& other ) + : type_( other.type() ) + , v_( other.v_ ) + , is_uint64_( other.is_uint64_ ) + { + } + + template< class Config > + Value_impl< Config >& Value_impl< Config >::operator=( const Value_impl& lhs ) + { + Value_impl tmp( lhs ); + + std::swap( type_, tmp.type_ ); + std::swap( v_, tmp.v_ ); + std::swap( is_uint64_, tmp.is_uint64_ ); + + return *this; + } + + template< class Config > + bool Value_impl< Config >::operator==( const Value_impl& lhs ) const + { + if( this == &lhs ) return true; + + if( type() != lhs.type() ) return false; + + return v_ == lhs.v_; + } + + template< class Config > + Value_type Value_impl< Config >::type() const + { + return type_; + } + + template< class Config > + bool Value_impl< Config >::is_uint64() const + { + return is_uint64_; + } + + template< class Config > + bool Value_impl< Config >::is_null() const + { + return type() == null_type; + } + + template< class Config > + void Value_impl< Config >::check_type( const Value_type vtype ) const + { + if( type() != vtype ) + { + std::ostringstream os; + + os << "value type is " << type() << " not " << vtype; + + throw std::runtime_error( os.str() ); + } + } + + template< class Config > + const typename Config::String_type& Value_impl< Config >::get_str() const + { + check_type( str_type ); + + return *boost::get< String_type >( &v_ ); + } + + template< class Config > + const typename Value_impl< Config >::Object& Value_impl< Config >::get_obj() const + { + check_type( obj_type ); + + return *boost::get< Object >( &v_ ); + } + + template< class Config > + const typename Value_impl< Config >::Array& Value_impl< Config >::get_array() const + { + check_type( array_type ); + + return *boost::get< Array >( &v_ ); + } + + template< class Config > + bool Value_impl< Config >::get_bool() const + { + check_type( bool_type ); + + return boost::get< bool >( v_ ); + } + + template< class Config > + int Value_impl< Config >::get_int() const + { + check_type( int_type ); + + return static_cast< int >( get_int64() ); + } + + template< class Config > + boost::int64_t Value_impl< Config >::get_int64() const + { + check_type( int_type ); + + return boost::get< boost::int64_t >( v_ ); + } + + template< class Config > + boost::uint64_t Value_impl< Config >::get_uint64() const + { + check_type( int_type ); + + return static_cast< boost::uint64_t >( get_int64() ); + } + + template< class Config > + double Value_impl< Config >::get_real() const + { + if( type() == int_type ) + { + return is_uint64() ? static_cast< double >( get_uint64() ) + : static_cast< double >( get_int64() ); + } + + check_type( real_type ); + + return boost::get< double >( v_ ); + } + + template< class Config > + typename Value_impl< Config >::Object& Value_impl< Config >::get_obj() + { + check_type( obj_type ); + + return *boost::get< Object >( &v_ ); + } + + template< class Config > + typename Value_impl< Config >::Array& Value_impl< Config >::get_array() + { + check_type( array_type ); + + return *boost::get< Array >( &v_ ); + } + + template< class Config > + Pair_impl< Config >::Pair_impl( const String_type& name, const Value_type& value ) + : name_( name ) + , value_( value ) + { + } + + template< class Config > + bool Pair_impl< Config >::operator==( const Pair_impl< Config >& lhs ) const + { + if( this == &lhs ) return true; + + return ( name_ == lhs.name_ ) && ( value_ == lhs.value_ ); + } + + // converts a C string, ie. 8 bit char array, to a string object + // + template < class String_type > + String_type to_str( const char* c_str ) + { + String_type result; + + for( const char* p = c_str; *p != 0; ++p ) + { + result += *p; + } + + return result; + } + + // + + namespace internal_ + { + template< typename T > + struct Type_to_type + { + }; + + template< class Value > + int get_value( const Value& value, Type_to_type< int > ) + { + return value.get_int(); + } + + template< class Value > + boost::int64_t get_value( const Value& value, Type_to_type< boost::int64_t > ) + { + return value.get_int64(); + } + + template< class Value > + boost::uint64_t get_value( const Value& value, Type_to_type< boost::uint64_t > ) + { + return value.get_uint64(); + } + + template< class Value > + double get_value( const Value& value, Type_to_type< double > ) + { + return value.get_real(); + } + + template< class Value > + typename Value::String_type get_value( const Value& value, Type_to_type< typename Value::String_type > ) + { + return value.get_str(); + } + + template< class Value > + typename Value::Array get_value( const Value& value, Type_to_type< typename Value::Array > ) + { + return value.get_array(); + } + + template< class Value > + typename Value::Object get_value( const Value& value, Type_to_type< typename Value::Object > ) + { + return value.get_obj(); + } + + template< class Value > + bool get_value( const Value& value, Type_to_type< bool > ) + { + return value.get_bool(); + } + } + + template< class Config > + template< typename T > + T Value_impl< Config >::get_value() const + { + return internal_::get_value( *this, internal_::Type_to_type< T >() ); + } +} + +#endif diff --git a/json/json_spirit_writer.cpp b/json/json_spirit_writer.cpp new file mode 100644 index 00000000..f3367b68 --- /dev/null +++ b/json/json_spirit_writer.cpp @@ -0,0 +1,95 @@ +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#include "json_spirit_writer.h" +#include "json_spirit_writer_template.h" + +void json_spirit::write( const Value& value, std::ostream& os ) +{ + write_stream( value, os, false ); +} + +void json_spirit::write_formatted( const Value& value, std::ostream& os ) +{ + write_stream( value, os, true ); +} + +std::string json_spirit::write( const Value& value ) +{ + return write_string( value, false ); +} + +std::string json_spirit::write_formatted( const Value& value ) +{ + return write_string( value, true ); +} + +#ifndef BOOST_NO_STD_WSTRING + +void json_spirit::write( const wValue& value, std::wostream& os ) +{ + write_stream( value, os, false ); +} + +void json_spirit::write_formatted( const wValue& value, std::wostream& os ) +{ + write_stream( value, os, true ); +} + +std::wstring json_spirit::write( const wValue& value ) +{ + return write_string( value, false ); +} + +std::wstring json_spirit::write_formatted( const wValue& value ) +{ + return write_string( value, true ); +} + +#endif + +void json_spirit::write( const mValue& value, std::ostream& os ) +{ + write_stream( value, os, false ); +} + +void json_spirit::write_formatted( const mValue& value, std::ostream& os ) +{ + write_stream( value, os, true ); +} + +std::string json_spirit::write( const mValue& value ) +{ + return write_string( value, false ); +} + +std::string json_spirit::write_formatted( const mValue& value ) +{ + return write_string( value, true ); +} + +#ifndef BOOST_NO_STD_WSTRING + +void json_spirit::write( const wmValue& value, std::wostream& os ) +{ + write_stream( value, os, false ); +} + +void json_spirit::write_formatted( const wmValue& value, std::wostream& os ) +{ + write_stream( value, os, true ); +} + +std::wstring json_spirit::write( const wmValue& value ) +{ + return write_string( value, false ); +} + +std::wstring json_spirit::write_formatted( const wmValue& value ) +{ + return write_string( value, true ); +} + +#endif diff --git a/json/json_spirit_writer.h b/json/json_spirit_writer.h new file mode 100644 index 00000000..1b67b512 --- /dev/null +++ b/json/json_spirit_writer.h @@ -0,0 +1,50 @@ +#ifndef JSON_SPIRIT_WRITER +#define JSON_SPIRIT_WRITER + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#if defined(_MSC_VER) && (_MSC_VER >= 1020) +# pragma once +#endif + +#include "json_spirit_value.h" +#include + +namespace json_spirit +{ + // functions to convert JSON Values to text, + // the "formatted" versions add whitespace to format the output nicely + + void write ( const Value& value, std::ostream& os ); + void write_formatted( const Value& value, std::ostream& os ); + std::string write ( const Value& value ); + std::string write_formatted( const Value& value ); + +#ifndef BOOST_NO_STD_WSTRING + + void write ( const wValue& value, std::wostream& os ); + void write_formatted( const wValue& value, std::wostream& os ); + std::wstring write ( const wValue& value ); + std::wstring write_formatted( const wValue& value ); + +#endif + + void write ( const mValue& value, std::ostream& os ); + void write_formatted( const mValue& value, std::ostream& os ); + std::string write ( const mValue& value ); + std::string write_formatted( const mValue& value ); + +#ifndef BOOST_NO_STD_WSTRING + + void write ( const wmValue& value, std::wostream& os ); + void write_formatted( const wmValue& value, std::wostream& os ); + std::wstring write ( const wmValue& value ); + std::wstring write_formatted( const wmValue& value ); + +#endif +} + +#endif diff --git a/json/json_spirit_writer_template.h b/json/json_spirit_writer_template.h new file mode 100644 index 00000000..c9937565 --- /dev/null +++ b/json/json_spirit_writer_template.h @@ -0,0 +1,245 @@ +#ifndef JSON_SPIRIT_WRITER_TEMPLATE +#define JSON_SPIRIT_WRITER_TEMPLATE + +// Copyright John W. Wilkinson 2007 - 2009. +// Distributed under the MIT License, see accompanying file LICENSE.txt + +// json spirit version 4.03 + +#include "json_spirit_value.h" + +#include +#include +#include + +namespace json_spirit +{ + inline char to_hex_char( unsigned int c ) + { + assert( c <= 0xF ); + + const char ch = static_cast< char >( c ); + + if( ch < 10 ) return '0' + ch; + + return 'A' - 10 + ch; + } + + template< class String_type > + String_type non_printable_to_string( unsigned int c ) + { + typedef typename String_type::value_type Char_type; + + String_type result( 6, '\\' ); + + result[1] = 'u'; + + result[ 5 ] = to_hex_char( c & 0x000F ); c >>= 4; + result[ 4 ] = to_hex_char( c & 0x000F ); c >>= 4; + result[ 3 ] = to_hex_char( c & 0x000F ); c >>= 4; + result[ 2 ] = to_hex_char( c & 0x000F ); + + return result; + } + + template< typename Char_type, class String_type > + bool add_esc_char( Char_type c, String_type& s ) + { + switch( c ) + { + case '"': s += to_str< String_type >( "\\\"" ); return true; + case '\\': s += to_str< String_type >( "\\\\" ); return true; + case '\b': s += to_str< String_type >( "\\b" ); return true; + case '\f': s += to_str< String_type >( "\\f" ); return true; + case '\n': s += to_str< String_type >( "\\n" ); return true; + case '\r': s += to_str< String_type >( "\\r" ); return true; + case '\t': s += to_str< String_type >( "\\t" ); return true; + } + + return false; + } + + template< class String_type > + String_type add_esc_chars( const String_type& s ) + { + typedef typename String_type::const_iterator Iter_type; + typedef typename String_type::value_type Char_type; + + String_type result; + + const Iter_type end( s.end() ); + + for( Iter_type i = s.begin(); i != end; ++i ) + { + const Char_type c( *i ); + + if( add_esc_char( c, result ) ) continue; + + const wint_t unsigned_c( ( c >= 0 ) ? c : 256 + c ); + + if( iswprint( unsigned_c ) ) + { + result += c; + } + else + { + result += non_printable_to_string< String_type >( unsigned_c ); + } + } + + return result; + } + + // this class generates the JSON text, + // it keeps track of the indentation level etc. + // + template< class Value_type, class Ostream_type > + class Generator + { + typedef typename Value_type::Config_type Config_type; + typedef typename Config_type::String_type String_type; + typedef typename Config_type::Object_type Object_type; + typedef typename Config_type::Array_type Array_type; + typedef typename String_type::value_type Char_type; + typedef typename Object_type::value_type Obj_member_type; + + public: + + Generator( const Value_type& value, Ostream_type& os, bool pretty ) + : os_( os ) + , indentation_level_( 0 ) + , pretty_( pretty ) + { + output( value ); + } + + private: + + void output( const Value_type& value ) + { + switch( value.type() ) + { + case obj_type: output( value.get_obj() ); break; + case array_type: output( value.get_array() ); break; + case str_type: output( value.get_str() ); break; + case bool_type: output( value.get_bool() ); break; + case int_type: output_int( value ); break; + case real_type: os_ << std::showpoint << std::setprecision( 16 ) + << value.get_real(); break; + case null_type: os_ << "null"; break; + default: assert( false ); + } + } + + void output( const Object_type& obj ) + { + output_array_or_obj( obj, '{', '}' ); + } + + void output( const Array_type& arr ) + { + output_array_or_obj( arr, '[', ']' ); + } + + void output( const Obj_member_type& member ) + { + output( Config_type::get_name( member ) ); space(); + os_ << ':'; space(); + output( Config_type::get_value( member ) ); + } + + void output_int( const Value_type& value ) + { + if( value.is_uint64() ) + { + os_ << value.get_uint64(); + } + else + { + os_ << value.get_int64(); + } + } + + void output( const String_type& s ) + { + os_ << '"' << add_esc_chars( s ) << '"'; + } + + void output( bool b ) + { + os_ << to_str< String_type >( b ? "true" : "false" ); + } + + template< class T > + void output_array_or_obj( const T& t, Char_type start_char, Char_type end_char ) + { + os_ << start_char; new_line(); + + ++indentation_level_; + + for( typename T::const_iterator i = t.begin(); i != t.end(); ++i ) + { + indent(); output( *i ); + + typename T::const_iterator next = i; + + if( ++next != t.end()) + { + os_ << ','; + } + + new_line(); + } + + --indentation_level_; + + indent(); os_ << end_char; + } + + void indent() + { + if( !pretty_ ) return; + + for( int i = 0; i < indentation_level_; ++i ) + { + os_ << " "; + } + } + + void space() + { + if( pretty_ ) os_ << ' '; + } + + void new_line() + { + if( pretty_ ) os_ << '\n'; + } + + Generator& operator=( const Generator& ); // to prevent "assignment operator could not be generated" warning + + Ostream_type& os_; + int indentation_level_; + bool pretty_; + }; + + template< class Value_type, class Ostream_type > + void write_stream( const Value_type& value, Ostream_type& os, bool pretty ) + { + Generator< Value_type, Ostream_type >( value, os, pretty ); + } + + template< class Value_type > + typename Value_type::String_type write_string( const Value_type& value, bool pretty ) + { + typedef typename Value_type::String_type::value_type Char_type; + + std::basic_ostringstream< Char_type > os; + + write_stream( value, os, pretty ); + + return os.str(); + } +} + +#endif From fa9dbd6b62ab161c29c9b8cf97b9c8da8ef4346e Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 10 Feb 2010 19:46:04 +0000 Subject: [PATCH 047/133] better error message in check_type(), tell the types by name instead of by number git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@59 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- json/json_spirit_value.h | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/json/json_spirit_value.h b/json/json_spirit_value.h index e8be3553..e9bcd36b 100644 --- a/json/json_spirit_value.h +++ b/json/json_spirit_value.h @@ -24,6 +24,7 @@ namespace json_spirit { enum Value_type{ obj_type, array_type, str_type, bool_type, int_type, real_type, null_type }; + static const char* Value_type_name[]={"obj", "array", "str", "bool", "int", "real", "null"}; template< class Config > // Config determines whether the value uses std::string or std::wstring and // whether JSON Objects are represented as vectors or maps @@ -341,7 +342,8 @@ namespace json_spirit { std::ostringstream os; - os << "value type is " << type() << " not " << vtype; + /// satoshi: tell the types by name instead of by number + os << "value is type " << Value_type_name[type()] << ", expected " << Value_type_name[vtype]; throw std::runtime_error( os.str() ); } From 98500d70a8cf25af4bab80526fd128ccdc36ceeb Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 12 Feb 2010 20:38:44 +0000 Subject: [PATCH 048/133] command line and JSON-RPC first draft, requires Boost 1.35 or higher for boost::asio, added SetBitcoinAddress and GetBitcoinAddress methods on CScript, critsect interlocks around mapAddressBook, added some random delays in tx broadcast to improve privacy, now compiles with MSVC 8.0 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@60 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- bignum.h | 33 ++- build-msw.txt | 27 +- db.h | 10 +- headers.h | 9 +- irc.cpp | 2 +- irc.h | 6 +- license.txt | 2 +- main.cpp | 151 +++++++---- main.h | 9 +- makefile | 25 +- makefile.unix.wx2.8 | 20 +- makefile.unix.wx2.9 | 20 +- makefile.vc | 25 +- net.cpp | 15 +- net.h | 3 +- rpc.cpp | 641 ++++++++++++++++++++++++++++++++++++++++++++ rpc.h | 6 + script.cpp | 9 +- script.h | 44 ++- serialize.h | 4 +- ui.cpp | 436 ++++++++++++++++-------------- util.cpp | 3 +- util.h | 18 +- 23 files changed, 1186 insertions(+), 332 deletions(-) create mode 100644 rpc.cpp create mode 100644 rpc.h diff --git a/bignum.h b/bignum.h index 8aa4e9c4..e1ab165b 100644 --- a/bignum.h +++ b/bignum.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -309,6 +309,37 @@ public: *this = 0 - *this; } + std::string ToString(int nBase=10) const + { + CAutoBN_CTX pctx; + CBigNum bnBase = nBase; + CBigNum bn0 = 0; + string str; + CBigNum bn = *this; + BN_set_negative(&bn, false); + CBigNum dv; + CBigNum rem; + if (BN_cmp(&bn, &bn0) == 0) + return "0"; + while (BN_cmp(&bn, &bn0) > 0) + { + if (!BN_div(&dv, &rem, &bn, &bnBase, pctx)) + throw bignum_error("CBigNum::ToString() : BN_div failed"); + bn = dv; + unsigned int c = rem.getulong(); + str += "0123456789abcdef"[c]; + } + if (BN_is_negative(this)) + str += "-"; + reverse(str.begin(), str.end()); + return str; + } + + std::string GetHex() const + { + return ToString(16); + } + unsigned int GetSerializeSize(int nType=0, int nVersion=VERSION) const { return ::GetSerializeSize(getvch(), nType, nVersion); diff --git a/build-msw.txt b/build-msw.txt index 56a38b02..9786255e 100644 --- a/build-msw.txt +++ b/build-msw.txt @@ -1,6 +1,6 @@ Bitcoin v0.2.0 BETA -Copyright (c) 2009 Satoshi Nakamoto +Copyright (c) 2009-2010 Satoshi Nakamoto Distributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php. This product includes software developed by the OpenSSL Project for use in @@ -13,8 +13,16 @@ WINDOWS BUILD NOTES Compilers Supported ------------------- -MinGW GCC -Microsoft Visual C++ 6.0 SP6 +MinGW GCC (recommended) + +MSVC 6.0 SP6: You'll need Boost version 1.34 because they dropped support +for MSVC 6.0 after that. However, they didn't add Asio until 1.35. +You should still be able to build with MSVC 6.0 by adding Asio to 1.34 by +unpacking boost_asio_*.zip into the boost directory: +http://sourceforge.net/projects/asio/files/asio + +MSVC 8.0 (2005) SP1 has been tested. Note: MSVC 7.0 and up have a habit of +linking to runtime DLLs that are not installed on XP by default. Dependencies @@ -22,8 +30,7 @@ Dependencies Libraries you need to download separately and build: default path download -wxWidgets \wxwidgets http://www.wxwidgets.org/downloads/ - or prebuilt: http://wxpack.sourceforge.net +wxWidgets \wxwidgets prebuilt: http://wxpack.sourceforge.net OpenSSL \openssl http://www.openssl.org/source/ Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html Boost \boost http://www.boost.org/users/download/ @@ -89,3 +96,13 @@ Using MinGW and MSYS: cd \db\build_unix sh ../dist/configure --enable-mingw --enable-cxx make + + +Boost +----- +download bjam.exe from +http://sourceforge.net/project/showfiles.php?group_id=7586&package_id=72941 +cd \boost +bjam toolset=gcc --build-type=complete stage +or +bjam toolset=msvc --build-type=complete stage diff --git a/db.h b/db.h index d32d2d5d..538076b5 100644 --- a/db.h +++ b/db.h @@ -1,8 +1,7 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. -#include class CTransaction; class CTxIndex; class CDiskBlockIndex; @@ -14,6 +13,7 @@ class CAddress; class CWalletTx; extern map mapAddressBook; +extern CCriticalSection cs_mapAddressBook; extern bool fClient; @@ -359,15 +359,17 @@ public: bool WriteName(const string& strAddress, const string& strName) { + CRITICAL_BLOCK(cs_mapAddressBook) + mapAddressBook[strAddress] = strName; nWalletDBUpdated++; - mapAddressBook[strAddress] = strName; return Write(make_pair(string("name"), strAddress), strName); } bool EraseName(const string& strAddress) { + CRITICAL_BLOCK(cs_mapAddressBook) + mapAddressBook.erase(strAddress); nWalletDBUpdated++; - mapAddressBook.erase(strAddress); return Erase(make_pair(string("name"), strAddress)); } diff --git a/headers.h b/headers.h index 45be4b65..dda4f9ca 100644 --- a/headers.h +++ b/headers.h @@ -29,6 +29,7 @@ #include #include #include +#include #include #include #include @@ -36,6 +37,7 @@ #include #include #include +#include #include #include #include @@ -53,6 +55,8 @@ #include #include #include +#include +#include #ifdef __WXMSW__ #include @@ -73,8 +77,6 @@ #include #include #include -#include -#include #endif #ifdef __BSD__ #include @@ -85,8 +87,6 @@ using namespace std; using namespace boost; - - #include "strlcpy.h" #include "serialize.h" #include "uint256.h" @@ -100,6 +100,7 @@ using namespace boost; #include "irc.h" #include "main.h" #include "market.h" +#include "rpc.h" #include "uibase.h" #include "ui.h" diff --git a/irc.cpp b/irc.cpp index f38db6bb..9d563cce 100644 --- a/irc.cpp +++ b/irc.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. diff --git a/irc.h b/irc.h index c69fd9ee..13de570b 100644 --- a/irc.h +++ b/irc.h @@ -1,8 +1,8 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. -extern bool RecvLine(SOCKET hSocket, string& strLine); -extern void ThreadIRCSeed(void* parg); +bool RecvLine(SOCKET hSocket, string& strLine); +void ThreadIRCSeed(void* parg); extern int nGotIRCAddresses; diff --git a/license.txt b/license.txt index 3844dfb4..f96fa98f 100644 --- a/license.txt +++ b/license.txt @@ -1,4 +1,4 @@ -Copyright (c) 2009 Satoshi Nakamoto +Copyright (c) 2009-2010 Satoshi Nakamoto Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/main.cpp b/main.cpp index 53acf8a7..90d239fa 100644 --- a/main.cpp +++ b/main.cpp @@ -26,6 +26,7 @@ CBlockIndex* pindexGenesisBlock = NULL; int nBestHeight = -1; uint256 hashBestChain = 0; CBlockIndex* pindexBest = NULL; +int64 nTimeBestReceived = 0; map mapOrphanBlocks; multimap mapOrphanBlocksByPrev; @@ -45,6 +46,9 @@ CKey keyUser; map mapRequestCount; CCriticalSection cs_mapRequestCount; +map mapAddressBook; +CCriticalSection cs_mapAddressBook; + // Settings int fGenerateBitcoins = false; int64 nTransactionFee = 0; @@ -573,7 +577,7 @@ bool CTransaction::RemoveFromMemoryPool() -int CMerkleTx::GetDepthInMainChain() const +int CMerkleTx::GetDepthInMainChain(int& nHeightRet) const { if (hashBlock == 0 || nIndex == -1) return 0; @@ -594,6 +598,7 @@ int CMerkleTx::GetDepthInMainChain() const fMerkleVerified = true; } + nHeightRet = pindex->nHeight; return pindexBest->nHeight - pindex->nHeight + 1; } @@ -708,15 +713,20 @@ void CWalletTx::RelayWalletTransaction(CTxDB& txdb) } } -void RelayWalletTransactions() +void ResendWalletTransactions() { - static int64 nLastTime; - if (GetTime() - nLastTime < 10 * 60) + // Do this infrequently and randomly to avoid giving away + // that these are our transactions. + static int64 nNextTime; + if (GetTime() < nNextTime) + return; + bool fFirst = (nNextTime == 0); + nNextTime = GetTime() + GetRand(120 * 60); + if (fFirst) return; - nLastTime = GetTime(); // Rebroadcast any of our txes that aren't in a block yet - printf("RelayWalletTransactions()\n"); + printf("ResendWalletTransactions()\n"); CTxDB txdb("r"); CRITICAL_BLOCK(cs_mapWallet) { @@ -725,7 +735,10 @@ void RelayWalletTransactions() foreach(PAIRTYPE(const uint256, CWalletTx)& item, mapWallet) { CWalletTx& wtx = item.second; - mapSorted.insert(make_pair(wtx.nTimeReceived, &wtx)); + // Don't rebroadcast until it's had plenty of time that + // it should have gotten in already by now. + if (nTimeBestReceived - wtx.nTimeReceived > 60 * 60) + mapSorted.insert(make_pair(wtx.nTimeReceived, &wtx)); } foreach(PAIRTYPE(const unsigned int, CWalletTx*)& item, mapSorted) { @@ -1219,10 +1232,11 @@ bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) } } - // New best link + // New best block hashBestChain = hash; pindexBest = pindexNew; nBestHeight = pindexBest->nHeight; + nTimeBestReceived = GetTime(); nTransactionsUpdated++; printf("AddToBlockIndex: new best=%s height=%d\n", hashBestChain.ToString().substr(0,16).c_str(), nBestHeight); } @@ -1232,9 +1246,6 @@ bool CBlock::AddToBlockIndex(unsigned int nFile, unsigned int nBlockPos) if (pindexNew == pindexBest) { - // Relay wallet transactions that haven't gotten in yet - RelayWalletTransactions(); - // Notify UI to display prev block's coinbase if it was ours static uint256 hashPrevBestCoinBase; CRITICAL_BLOCK(cs_mapWallet) @@ -2248,7 +2259,7 @@ bool SendMessages(CNode* pto) return true; // Keep-alive ping - if (pto->nLastSend && GetTime() - pto->nLastSend > 12 * 60 && pto->vSend.empty()) + if (pto->nLastSend && GetTime() - pto->nLastSend > 30 * 60 && pto->vSend.empty()) pto->PushMessage("ping"); // Address refresh broadcast @@ -2270,60 +2281,81 @@ bool SendMessages(CNode* pto) } } + // Delay tx inv messages to protect privacy, + // trickle them out to a few nodes at a time. + bool fSendTxInv = false; + if (GetTimeMillis() - pto->nLastSentTxInv > 1800 + GetRand(200)) + { + pto->nLastSentTxInv = GetTimeMillis(); + fSendTxInv = true; + } + + // Resend wallet transactions that haven't gotten in a block yet + ResendWalletTransactions(); + // // Message: addr // - vector vAddrToSend; - vAddrToSend.reserve(pto->vAddrToSend.size()); + vector vAddr; + vAddr.reserve(pto->vAddrToSend.size()); foreach(const CAddress& addr, pto->vAddrToSend) { // returns true if wasn't already contained in the set if (pto->setAddrKnown.insert(addr).second) { - vAddrToSend.push_back(addr); - if (vAddrToSend.size() >= 1000) + vAddr.push_back(addr); + if (vAddr.size() >= 1000) { - pto->PushMessage("addr", vAddrToSend); - vAddrToSend.clear(); + pto->PushMessage("addr", vAddr); + vAddr.clear(); } } } pto->vAddrToSend.clear(); - if (!vAddrToSend.empty()) - pto->PushMessage("addr", vAddrToSend); + if (!vAddr.empty()) + pto->PushMessage("addr", vAddr); // // Message: inventory // - vector vInventoryToSend; + vector vInv; + vector vInvWait; CRITICAL_BLOCK(pto->cs_inventory) { - vInventoryToSend.reserve(pto->vInventoryToSend.size()); + vInv.reserve(pto->vInventoryToSend.size()); + vInvWait.reserve(pto->vInventoryToSend.size()); foreach(const CInv& inv, pto->vInventoryToSend) { + // delay txes + if (!fSendTxInv && inv.type == MSG_TX) + { + vInvWait.push_back(inv); + continue; + } + // returns true if wasn't already contained in the set if (pto->setInventoryKnown.insert(inv).second) { - vInventoryToSend.push_back(inv); - if (vInventoryToSend.size() >= 1000) + vInv.push_back(inv); + if (vInv.size() >= 1000) { - pto->PushMessage("inv", vInventoryToSend); - vInventoryToSend.clear(); + pto->PushMessage("inv", vInv); + vInv.clear(); } } } - pto->vInventoryToSend.clear(); + pto->vInventoryToSend = vInvWait; } - if (!vInventoryToSend.empty()) - pto->PushMessage("inv", vInventoryToSend); + if (!vInv.empty()) + pto->PushMessage("inv", vInv); // // Message: getdata // - vector vAskFor; + vector vGetData; int64 nNow = GetTime() * 1000000; CTxDB txdb("r"); while (!pto->mapAskFor.empty() && (*pto->mapAskFor.begin()).first <= nNow) @@ -2332,17 +2364,17 @@ bool SendMessages(CNode* pto) if (!AlreadyHave(txdb, inv)) { printf("sending getdata: %s\n", inv.ToString().c_str()); - vAskFor.push_back(inv); - if (vAskFor.size() >= 1000) + vGetData.push_back(inv); + if (vGetData.size() >= 1000) { - pto->PushMessage("getdata", vAskFor); - vAskFor.clear(); + pto->PushMessage("getdata", vGetData); + vGetData.clear(); } } pto->mapAskFor.erase(pto->mapAskFor.begin()); } - if (!vAskFor.empty()) - pto->PushMessage("getdata", vAskFor); + if (!vGetData.empty()) + pto->PushMessage("getdata", vGetData); } return true; @@ -2405,7 +2437,6 @@ void ThreadBitcoinMiner(void* parg) vnThreadsRunning[3]--; PrintException(NULL, "ThreadBitcoinMiner()"); } - printf("ThreadBitcoinMiner exiting, %d threads remaining\n", vnThreadsRunning[3]); } @@ -2842,6 +2873,13 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CK // Fill a vout back to self with any change if (nValueIn > nTotalValue) { + // Note: We use a new key here to keep it from being obvious which side is the change. + // The drawback is that by not reusing a previous key, the change may be lost if a + // backup is restored, if the backup doesn't have the new private key for the change. + // If we reused the old key, it would be possible to add code to look for and + // rediscover unknown transactions that were written with keys of ours to recover + // post-backup change. + // New private key if (keyRet.IsNull()) keyRet.MakeNewKey(); @@ -2899,7 +2937,7 @@ bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key) //// update: This matters even less now that fSpent can get corrected //// when transactions are seen in VerifySignature. The remote chance of //// unmarked fSpent will be handled by that. Don't need to make this - //// transactional. + //// transactional. Pls delete this comment block later. // This is only to keep the database open to defeat the auto-flush for the // duration of this scope. This is the only place where this optimization @@ -2932,7 +2970,7 @@ bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key) -bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) +string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) { CRITICAL_BLOCK(cs_main) { @@ -2945,13 +2983,13 @@ bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) strError = strprintf("Error: This is an oversized transaction that requires a transaction fee of %s ", FormatMoney(nFeeRequired).c_str()); else strError = "Error: Transaction creation failed "; - wxMessageBox(strError, "Sending..."); - return error("SendMoney() : %s", strError.c_str()); + printf("SendMoney() : %s", strError.c_str()); + return strError; } if (!CommitTransactionSpent(wtxNew, key)) { - wxMessageBox("Error finalizing transaction ", "Sending..."); - return error("SendMoney() : Error finalizing transaction"); + printf("SendMoney() : Error finalizing transaction"); + return "Error finalizing transaction"; } // Track how many getdata requests our transaction gets @@ -2964,11 +3002,32 @@ bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) if (!wtxNew.AcceptTransaction()) { // This must not fail. The transaction has already been signed and recorded. - wxMessageBox("Error: The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here.", "Sending..."); - return error("SendMoney() : Error: Transaction not valid"); + printf("SendMoney() : Error: Transaction not valid"); + return "Error: The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here."; } wtxNew.RelayWalletTransaction(); } MainFrameRepaint(); - return true; + return ""; +} + + + +string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtxNew) +{ + // Check amount + if (nValue <= 0) + return "Invalid amount"; + if (nValue + nTransactionFee > GetBalance()) + return "You don't have enough money"; + + // Parse bitcoin address + uint160 hash160; + if (!AddressToHash160(strAddress, hash160)) + return "Invalid bitcoin address"; + + // Send to bitcoin address + CScript scriptPubKey; + scriptPubKey.SetBitcoinAddress(hash160); + return SendMoney(scriptPubKey, nValue, wtxNew); } diff --git a/main.h b/main.h index 14c445c7..bcb6ec7d 100644 --- a/main.h +++ b/main.h @@ -36,6 +36,8 @@ extern CBlockIndex* pindexBest; extern unsigned int nTransactionsUpdated; extern map mapRequestCount; extern CCriticalSection cs_mapRequestCount; +extern map mapAddressBook; +extern CCriticalSection cs_mapAddressBook; // Settings extern int fGenerateBitcoins; @@ -58,7 +60,6 @@ vector GenerateNewKey(); bool AddToWallet(const CWalletTx& wtxIn); void WalletUpdateSpent(const COutPoint& prevout); void ReacceptWalletTransactions(); -void RelayWalletTransactions(); bool LoadBlockIndex(bool fAllowNew=true); void PrintBlockTree(); bool ProcessMessages(CNode* pfrom); @@ -67,7 +68,8 @@ bool SendMessages(CNode* pto); int64 GetBalance(); bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CKey& keyRet, int64& nFeeRequiredRet); bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key); -bool SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew); +string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew); +string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtxNew); void GenerateBitcoins(bool fGenerate); void ThreadBitcoinMiner(void* parg); void BitcoinMiner(); @@ -680,7 +682,8 @@ public: int SetMerkleBranch(const CBlock* pblock=NULL); - int GetDepthInMainChain() const; + int GetDepthInMainChain(int& nHeightRet) const; + int GetDepthInMainChain() const { int nHeight; return GetDepthInMainChain(nHeight); } bool IsInMainChain() const { return GetDepthInMainChain() > 0; } int GetBlocksToMaturity() const; bool AcceptTransaction(CTxDB& txdb, bool fCheckInputs=true); diff --git a/makefile b/makefile index 0dd62210..affbe517 100644 --- a/makefile +++ b/makefile @@ -1,4 +1,4 @@ -# Copyright (c) 2009 Satoshi Nakamoto +# Copyright (c) 2009-2010 Satoshi Nakamoto # Distributed under the MIT/X11 software license, see the accompanying # file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -16,22 +16,23 @@ endif INCLUDEPATHS=-I"/boost" -I"/db/build_unix" -I"/openssl/include" -I"/wxwidgets/lib/vc_lib/mswd" -I"/wxwidgets/include" -LIBPATHS=-L"/db/build_unix" -L"/openssl/out" -L"/wxwidgets/lib/gcc_lib" +LIBPATHS=-L"/boost/stage/lib" -L"/db/build_unix" -L"/openssl/out" -L"/wxwidgets/lib/gcc_lib" LIBS= \ + -l libboost_system-mgw34-mt-d -l libboost_filesystem-mgw34-mt-d \ -l db_cxx \ -l eay32 \ -l wxmsw28$(D)_richtext -l wxmsw28$(D)_html -l wxmsw28$(D)_core -l wxmsw28$(D)_adv -l wxbase28$(D) -l wxtiff$(D) -l wxjpeg$(D) -l wxpng$(D) -l wxzlib$(D) -l wxregex$(D) -l wxexpat$(D) \ -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 -l shlwapi WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h all: bitcoin.exe -headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h +headers.h.gch: headers.h $(HEADERS) g++ -c $(CFLAGS) -o $@ $< obj/util.o: util.cpp $(HEADERS) @@ -40,19 +41,19 @@ obj/util.o: util.cpp $(HEADERS) obj/script.o: script.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/db.o: db.cpp $(HEADERS) market.h +obj/db.o: db.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/net.o: net.cpp $(HEADERS) net.h +obj/net.o: net.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h +obj/main.o: main.cpp $(HEADERS) sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) market.h +obj/market.o: market.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h +obj/ui.o: ui.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< obj/uibase.o: uibase.cpp uibase.h @@ -64,13 +65,17 @@ obj/sha.o: sha.cpp sha.h obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< +obj/rpc.o: rpc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp windres $(WXDEFS) $(INCLUDEPATHS) -o $@ -i $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/ui_res.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o \ + obj/ui_res.o bitcoin.exe: headers.h.gch $(OBJS) -kill /f bitcoin.exe diff --git a/makefile.unix.wx2.8 b/makefile.unix.wx2.8 index b9826d6c..d3358ab2 100644 --- a/makefile.unix.wx2.8 +++ b/makefile.unix.wx2.8 @@ -1,4 +1,4 @@ -# Copyright (c) 2009 Satoshi Nakamoto +# Copyright (c) 2009-2010 Satoshi Nakamoto # Distributed under the MIT/X11 software license, see the accompanying # file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -35,14 +35,14 @@ LIBS= \ WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h all: bitcoin -headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h +headers.h.gch: headers.h $(HEADERS) g++ -c $(CFLAGS) -o $@ $< obj/util.o: util.cpp $(HEADERS) @@ -51,19 +51,19 @@ obj/util.o: util.cpp $(HEADERS) obj/script.o: script.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/db.o: db.cpp $(HEADERS) market.h +obj/db.o: db.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/net.o: net.cpp $(HEADERS) net.h +obj/net.o: net.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h +obj/main.o: main.cpp $(HEADERS) sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) market.h +obj/market.o: market.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h +obj/ui.o: ui.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< obj/uibase.o: uibase.cpp uibase.h @@ -75,11 +75,13 @@ obj/sha.o: sha.cpp sha.h obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< +obj/rpc.o: rpc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/makefile.unix.wx2.9 b/makefile.unix.wx2.9 index 81dcbd70..8defcbb6 100644 --- a/makefile.unix.wx2.9 +++ b/makefile.unix.wx2.9 @@ -1,4 +1,4 @@ -# Copyright (c) 2009 Satoshi Nakamoto +# Copyright (c) 2009-2010 Satoshi Nakamoto # Distributed under the MIT/X11 software license, see the accompanying # file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -35,14 +35,14 @@ LIBS= \ WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h all: bitcoin -headers.h.gch: headers.h $(HEADERS) net.h irc.h market.h uibase.h ui.h +headers.h.gch: headers.h $(HEADERS) g++ -c $(CFLAGS) -o $@ $< obj/util.o: util.cpp $(HEADERS) @@ -51,19 +51,19 @@ obj/util.o: util.cpp $(HEADERS) obj/script.o: script.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/db.o: db.cpp $(HEADERS) market.h +obj/db.o: db.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/net.o: net.cpp $(HEADERS) net.h +obj/net.o: net.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/main.o: main.cpp $(HEADERS) net.h market.h sha.h +obj/main.o: main.cpp $(HEADERS) sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) market.h +obj/market.o: market.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/ui.o: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h +obj/ui.o: ui.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< obj/uibase.o: uibase.cpp uibase.h @@ -75,11 +75,13 @@ obj/sha.o: sha.cpp sha.h obj/irc.o: irc.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< +obj/rpc.o: rpc.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o + obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o bitcoin: headers.h.gch $(OBJS) g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) diff --git a/makefile.vc b/makefile.vc index fb7f3086..73fd1cfd 100644 --- a/makefile.vc +++ b/makefile.vc @@ -1,4 +1,4 @@ -# Copyright (c) 2009 Satoshi Nakamoto +# Copyright (c) 2009-2010 Satoshi Nakamoto # Distributed under the MIT/X11 software license, see the accompanying # file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -14,15 +14,16 @@ DEBUGFLAGS=/Zi /Od /D__WXDEBUG__ INCLUDEPATHS=/I"/boost" /I"/db/build_windows" /I"/openssl/include" /I"/wxwidgets/lib/vc_lib/mswd" /I"/wxwidgets/include" -LIBPATHS=/LIBPATH:"/db/build_windows/$(BUILD)" /LIBPATH:"/openssl/out" /LIBPATH:"/wxwidgets/lib/vc_lib" +LIBPATHS=/LIBPATH:"/boost/stage/lib" /LIBPATH:"/db/build_windows/$(BUILD)" /LIBPATH:"/openssl/out" /LIBPATH:"/wxwidgets/lib/vc_lib" LIBS= \ + libboost_system-vc80-mt-gd.lib libboost_filesystem-vc80-mt-gd.lib \ libdb47s$(D).lib \ libeay32.lib \ wxmsw28$(D)_richtext.lib wxmsw28$(D)_html.lib wxmsw28$(D)_core.lib wxmsw28$(D)_adv.lib wxbase28$(D).lib wxtiff$(D).lib wxjpeg$(D).lib wxpng$(D).lib wxzlib$(D).lib wxregex$(D).lib wxexpat$(D).lib \ kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib shlwapi.lib WXDEFS=/DWIN32 /D__WXMSW__ /D_WINDOWS /DNOPCH CFLAGS=/c /nologo /Ob0 /MD$(D) /EHsc /GR /Zm300 /YX /Fpobj/headers.pch $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h util.h main.h serialize.h uint256.h key.h bignum.h script.h db.h base58.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h @@ -35,37 +36,41 @@ obj\util.obj: util.cpp $(HEADERS) obj\script.obj: script.cpp $(HEADERS) cl $(CFLAGS) /Fo$@ %s -obj\db.obj: db.cpp $(HEADERS) market.h +obj\db.obj: db.cpp $(HEADERS) cl $(CFLAGS) /Fo$@ %s -obj\net.obj: net.cpp $(HEADERS) net.h +obj\net.obj: net.cpp $(HEADERS) cl $(CFLAGS) /Fo$@ %s -obj\main.obj: main.cpp $(HEADERS) net.h market.h +obj\main.obj: main.cpp $(HEADERS) sha.h cl $(CFLAGS) /Fo$@ %s -obj\market.obj: market.cpp $(HEADERS) market.h +obj\market.obj: market.cpp $(HEADERS) cl $(CFLAGS) /Fo$@ %s -obj\ui.obj: ui.cpp $(HEADERS) net.h uibase.h ui.h market.h +obj\ui.obj: ui.cpp $(HEADERS) cl $(CFLAGS) /Fo$@ %s obj\uibase.obj: uibase.cpp uibase.h cl $(CFLAGS) /Fo$@ %s -obj\sha.obj: sha.cpp sha.h +obj\sha.obj: sha.cpp sha.h cl $(CFLAGS) /O2 /Fo$@ %s obj\irc.obj: irc.cpp $(HEADERS) cl $(CFLAGS) /Fo$@ %s +obj\rpc.obj: rpc.cpp $(HEADERS) + cl $(CFLAGS) /Fo$@ %s + obj\ui.res: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp rc $(INCLUDEPATHS) $(WXDEFS) /Fo$@ %s OBJS=obj\util.obj obj\script.obj obj\db.obj obj\net.obj obj\main.obj obj\market.obj \ - obj\ui.obj obj\uibase.obj obj\sha.obj obj\irc.obj obj\ui.res + obj\ui.obj obj\uibase.obj obj\sha.obj obj\irc.obj obj\rpc.obj \ + obj\ui.res bitcoin.exe: $(OBJS) -kill /f bitcoin.exe & sleep 1 diff --git a/net.cpp b/net.cpp index 38d05de1..ada78eb3 100644 --- a/net.cpp +++ b/net.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -522,7 +522,6 @@ void CNode::Cleanup() void ThreadSocketHandler(void* parg) { IMPLEMENT_RANDOMIZE_STACK(ThreadSocketHandler(parg)); - try { vnThreadsRunning[0]++; @@ -536,7 +535,6 @@ void ThreadSocketHandler(void* parg) vnThreadsRunning[0]--; throw; // support pthread_cancel() } - printf("ThreadSocketHandler exiting\n"); } @@ -816,7 +814,6 @@ void ThreadSocketHandler2(void* parg) void ThreadOpenConnections(void* parg) { IMPLEMENT_RANDOMIZE_STACK(ThreadOpenConnections(parg)); - try { vnThreadsRunning[1]++; @@ -830,7 +827,6 @@ void ThreadOpenConnections(void* parg) vnThreadsRunning[1]--; PrintException(NULL, "ThreadOpenConnections()"); } - printf("ThreadOpenConnections exiting\n"); } @@ -928,7 +924,7 @@ void ThreadOpenConnections2(void* parg) // 30 days 27 hours // 90 days 46 hours // 365 days 93 hours - int64 nDelay = (int64)(3600.0 * sqrt(fabs(nSinceLastSeen) / 3600.0) + nRandomizer); + int64 nDelay = (int64)(3600.0 * sqrt(fabs((double)nSinceLastSeen) / 3600.0) + nRandomizer); // Fast reconnect for one hour after last seen if (nSinceLastSeen < 60 * 60) @@ -1016,7 +1012,6 @@ bool OpenNetworkConnection(const CAddress& addrConnect) void ThreadMessageHandler(void* parg) { IMPLEMENT_RANDOMIZE_STACK(ThreadMessageHandler(parg)); - try { vnThreadsRunning[2]++; @@ -1030,7 +1025,6 @@ void ThreadMessageHandler(void* parg) vnThreadsRunning[2]--; PrintException(NULL, "ThreadMessageHandler()"); } - printf("ThreadMessageHandler exiting\n"); } @@ -1329,7 +1323,7 @@ bool StopNode() fShutdown = true; nTransactionsUpdated++; int64 nStart = GetTime(); - while (vnThreadsRunning[0] > 0 || vnThreadsRunning[2] > 0 || vnThreadsRunning[3] > 0) + while (vnThreadsRunning[0] > 0 || vnThreadsRunning[2] > 0 || vnThreadsRunning[3] > 0 || vnThreadsRunning[4] > 0) { if (GetTime() - nStart > 20) break; @@ -1339,7 +1333,8 @@ bool StopNode() if (vnThreadsRunning[1] > 0) printf("ThreadOpenConnections still running\n"); if (vnThreadsRunning[2] > 0) printf("ThreadMessageHandler still running\n"); if (vnThreadsRunning[3] > 0) printf("ThreadBitcoinMiner still running\n"); - while (vnThreadsRunning[2] > 0) + if (vnThreadsRunning[4] > 0) printf("ThreadRPCServer still running\n"); + while (vnThreadsRunning[2] > 0 || vnThreadsRunning[4] > 0) Sleep(20); Sleep(50); diff --git a/net.h b/net.h index ba4607a5..ce6f772f 100644 --- a/net.h +++ b/net.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -518,6 +518,7 @@ public: vector vInventoryToSend; CCriticalSection cs_inventory; multimap mapAskFor; + int64 nLastSentTxInv; // publish and subscription vector vfSubscribe; diff --git a/rpc.cpp b/rpc.cpp new file mode 100644 index 00000000..9f28e7ec --- /dev/null +++ b/rpc.cpp @@ -0,0 +1,641 @@ +// Copyright (c) 2010 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" +#undef printf +#include +#include "json/json_spirit_reader_template.h" +#include "json/json_spirit_writer_template.h" +#include "json/json_spirit_utils.h" +#define printf OutputDebugStringF +// MinGW 3.4.5 gets "fatal error: had to relocate PCH" if the json headers are +// precompiled in headers.h. The problem might be when the pch file goes over +// a certain size around 145MB. If we need access to json_spirit outside this +// file, we could use the compiled json_spirit option. + +using boost::asio::ip::tcp; +using namespace json_spirit; + +void ThreadRPCServer2(void* parg); + + + + + + + +/// +/// Note: I'm not finished designing this interface, it's still subject to change. +/// + + + +Value stop(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "stop (no parameters)\n" + "Stop bitcoin server."); + + // Shutdown will take long enough that the response should get back + CreateThread(Shutdown, NULL); + return "bitcoin server stopping"; +} + + +Value getblockcount(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getblockcount (no parameters)\n" + "Returns the number of blocks in the longest block chain."); + + return nBestHeight + 1; +} + + +Value getblocknumber(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getblocknumber (no parameters)\n" + "Returns the block number of the latest block in the longest block chain."); + + return nBestHeight; +} + + +Value getdifficulty(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getdifficulty (no parameters)\n" + "Returns the proof-of-work difficulty as a multiple of the minimum difficulty."); + + if (pindexBest == NULL) + throw runtime_error("block chain not loaded"); + + // Floating point number that is a multiple of the minimum difficulty, + // minimum difficulty = 1.0. + int nShift = 256 - 32 - 31; // to fit in a uint + double dMinimum = (CBigNum().SetCompact(bnProofOfWorkLimit.GetCompact()) >> nShift).getuint(); + double dCurrently = (CBigNum().SetCompact(pindexBest->nBits) >> nShift).getuint(); + return dMinimum / dCurrently; +} + + +Value getnewaddress(const Array& params) +{ + if (params.size() > 1) + throw runtime_error( + "getnewaddress [label]\n" + "Returns a new bitcoin address for receiving payments. " + "If [label] is specified (recommended), it is added to the address book " + "so payments received with the address will be labeled."); + + // Parse the label first so we don't generate a key if there's an error + string strLabel; + if (params.size() > 0) + strLabel = params[0].get_str(); + + // Generate a new key that is added to wallet + string strAddress = PubKeyToAddress(GenerateNewKey()); + + if (params.size() > 0) + SetAddressBookName(strAddress, strLabel); + return strAddress; +} + + +Value sendtoaddress(const Array& params) +{ + if (params.size() < 2 || params.size() > 4) + throw runtime_error( + "sendtoaddress [comment] [comment-to]\n" + " is a real and is rounded to the nearest 0.01"); + + string strAddress = params[0].get_str(); + + // Amount + if (params[1].get_real() <= 0.0 || params[1].get_real() > 21000000.0) + throw runtime_error("Invalid amount"); + int64 nAmount = roundint64(params[1].get_real() * 100.00) * CENT; + + // Wallet comments + CWalletTx wtx; + if (params.size() > 2 && params[2].type() != null_type && !params[2].get_str().empty()) + wtx.mapValue["message"] = params[2].get_str(); + if (params.size() > 3 && params[3].type() != null_type && !params[3].get_str().empty()) + wtx.mapValue["to"] = params[3].get_str(); + + string strError = SendMoneyToBitcoinAddress(strAddress, nAmount, wtx); + if (strError != "") + throw runtime_error(strError); + return "sent"; +} + + +Value listtransactions(const Array& params) +{ + if (params.size() > 2) + throw runtime_error( + "listtransactions [count=10] [includegenerated=false]\n" + "Returns up to [count] most recent transactions."); + + int64 nCount = 10; + if (params.size() > 0) + nCount = params[0].get_int64(); + bool fGenerated = false; + if (params.size() > 1) + fGenerated = params[1].get_bool(); + + Array ret; + //// not finished + ret.push_back("not implemented yet"); + return ret; +} + + +Value getamountpaid(const Array& params) +{ + if (params.size() < 1 || params.size() > 2) + throw runtime_error( + "getamountpaid [minconf=1]\n" + "Returns the total amount paid to in transactions with at least [minconf] confirmations."); + + // Bitcoin address + string strAddress = params[0].get_str(); + CScript scriptPubKey; + if (!scriptPubKey.SetBitcoinAddress(strAddress)) + throw runtime_error("Invalid bitcoin address"); + + // Minimum confirmations + int nMinDepth = 1; + if (params.size() > 1) + nMinDepth = params[1].get_int(); + + // Tally + int64 nAmount = 0; + CRITICAL_BLOCK(cs_mapWallet) + { + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + const CWalletTx& wtx = (*it).second; + if (wtx.IsCoinBase() || !wtx.IsFinal()) + continue; + + foreach(const CTxOut& txout, wtx.vout) + if (txout.scriptPubKey == scriptPubKey) + if (wtx.GetDepthInMainChain() >= nMinDepth) + nAmount += txout.nValue; + } + } + + return (double)nAmount / (double)COIN; +} + + +struct tallyitem +{ + int64 nAmount; + int nConf; + tallyitem() + { + nAmount = 0; + nConf = INT_MAX; + } +}; + +Value getallpayments(const Array& params) +{ + if (params.size() > 1) + throw runtime_error( + "getallpayments [minconf=1]\n" + "[minconf] is the minimum number of confirmations before payments are included.\n" + "Returns an array of objects containing:\n" + " \"address\" : bitcoin address\n" + " \"amount\" : total amount paid to the address\n" + " \"conf\" : number of confirmations\n" + " \"label\" : the label set for this address when it was created by getnewaddress"); + + // Minimum confirmations + int nMinDepth = 1; + if (params.size() > 0) + nMinDepth = params[0].get_int(); + + // Tally + map mapTally; + CRITICAL_BLOCK(cs_mapWallet) + { + for (map::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it) + { + const CWalletTx& wtx = (*it).second; + if (wtx.IsCoinBase() || !wtx.IsFinal()) + continue; + + int nDepth = wtx.GetDepthInMainChain(); + if (nDepth >= nMinDepth) + { + foreach(const CTxOut& txout, wtx.vout) + { + uint160 hash160 = txout.scriptPubKey.GetBitcoinAddressHash160(); + if (hash160 == 0 || !mapPubKeys.count(hash160)) + continue; + + tallyitem& item = mapTally[hash160]; + item.nAmount += txout.nValue; + item.nConf = min(item.nConf, nDepth); + } + } + } + } + + // Reply + Array ret; + CRITICAL_BLOCK(cs_mapAddressBook) + { + for (map::iterator it = mapTally.begin(); it != mapTally.end(); ++it) + { + string strAddress = Hash160ToAddress((*it).first); + string strLabel; + map::iterator mi = mapAddressBook.find(strAddress); + if (mi != mapAddressBook.end()) + strLabel = (*mi).second; + + Object obj; + obj.push_back(Pair("address", strAddress)); + obj.push_back(Pair("amount", (double)(*it).second.nAmount / (double)COIN)); + obj.push_back(Pair("conf", (*it).second.nConf)); + obj.push_back(Pair("label", strLabel)); + ret.push_back(obj); + } + } + return ret; +} + + + + + + + +// +// Call Table +// + +typedef Value(*rpcfn_type)(const Array& params); +pair pCallTable[] = +{ + make_pair("stop", &stop), + make_pair("getblockcount", &getblockcount), + make_pair("getblocknumber", &getblocknumber), + make_pair("getdifficulty", &getdifficulty), + make_pair("getnewaddress", &getnewaddress), + make_pair("sendtoaddress", &sendtoaddress), + make_pair("listtransactions", &listtransactions), + make_pair("getamountpaid", &getamountpaid), + make_pair("getallpayments", &getallpayments), +}; +map mapCallTable(pCallTable, pCallTable + sizeof(pCallTable)/sizeof(pCallTable[0])); + + + + +// +// HTTP protocol +// +// This ain't Apache. We're just using HTTP header for the length field +// and to be compatible with other JSON-RPC implementations. +// + +string HTTPPost(const string& strMsg) +{ + return strprintf( + "POST / HTTP/1.1\r\n" + "User-Agent: json-rpc/1.0\r\n" + "Host: 127.0.0.1\r\n" + "Content-Type: application/json\r\n" + "Content-Length: %d\r\n" + "Accept: application/json\r\n" + "\r\n" + "%s", + strMsg.size(), + strMsg.c_str()); +} + +string HTTPReply(const string& strMsg, int nStatus=200) +{ + string strStatus; + if (nStatus == 200) strStatus = "OK"; + if (nStatus == 500) strStatus = "Internal Server Error"; + return strprintf( + "HTTP/1.1 %d %s\r\n" + "Connection: close\r\n" + "Content-Length: %d\r\n" + "Content-Type: application/json\r\n" + "Date: Sat, 08 Jul 2006 12:04:08 GMT\r\n" + "Server: json-rpc/1.0\r\n" + "\r\n" + "%s", + nStatus, + strStatus.c_str(), + strMsg.size(), + strMsg.c_str()); +} + +int ReadHTTPHeader(tcp::iostream& stream) +{ + int nLen = 0; + loop + { + string str; + std::getline(stream, str); + if (str.empty() || str == "\r") + break; + if (str.substr(0,15) == "Content-Length:") + nLen = atoi(str.substr(15)); + } + return nLen; +} + +inline string ReadHTTP(tcp::iostream& stream) +{ + // Read header + int nLen = ReadHTTPHeader(stream); + if (nLen <= 0) + return string(); + + // Read message + vector vch(nLen); + stream.read(&vch[0], nLen); + return string(vch.begin(), vch.end()); +} + + + +// +// JSON-RPC protocol +// +// http://json-rpc.org/wiki/specification +// http://www.codeproject.com/KB/recipes/JSON_Spirit.aspx +// + +string JSONRPCRequest(const string& strMethod, const Array& params, const Value& id) +{ + Object request; + request.push_back(Pair("method", strMethod)); + request.push_back(Pair("params", params)); + request.push_back(Pair("id", id)); + return write_string(Value(request), false) + "\n"; +} + +string JSONRPCReply(const Value& result, const Value& error, const Value& id) +{ + Object reply; + if (error.type() != null_type) + reply.push_back(Pair("result", Value::null)); + else + reply.push_back(Pair("result", result)); + reply.push_back(Pair("error", error)); + reply.push_back(Pair("id", id)); + return write_string(Value(reply), false) + "\n"; +} + + + + +void ThreadRPCServer(void* parg) +{ + IMPLEMENT_RANDOMIZE_STACK(ThreadRPCServer(parg)); + try + { + vnThreadsRunning[4]++; + ThreadRPCServer2(parg); + vnThreadsRunning[4]--; + } + catch (std::exception& e) { + vnThreadsRunning[4]--; + PrintException(&e, "ThreadRPCServer()"); + } catch (...) { + vnThreadsRunning[4]--; + PrintException(NULL, "ThreadRPCServer()"); + } + printf("ThreadRPCServer exiting\n"); +} + +void ThreadRPCServer2(void* parg) +{ + printf("ThreadRPCServer started\n"); + + // Bind to loopback 127.0.0.1 so the socket can only be accessed locally + boost::asio::io_service io_service; + tcp::endpoint endpoint(boost::asio::ip::address_v4::loopback(), 8332); + tcp::acceptor acceptor(io_service, endpoint); + + loop + { + // Accept connection + tcp::iostream stream; + tcp::endpoint peer; + vnThreadsRunning[4]--; + acceptor.accept(*stream.rdbuf(), peer); + vnThreadsRunning[4]++; + if (fShutdown) + return; + + // Shouldn't be possible for anyone else to connect, but just in case + if (peer.address().to_string() != "127.0.0.1") + continue; + + // Receive request + string strRequest = ReadHTTP(stream); + printf("ThreadRPCServer request=%s", strRequest.c_str()); + + // Handle multiple invocations per request + string::iterator begin = strRequest.begin(); + while (skipspaces(begin), begin != strRequest.end()) + { + string::iterator prev = begin; + Value id; + try + { + // Parse request + Value valRequest; + if (!read_range(begin, strRequest.end(), valRequest)) + throw runtime_error("Parse error."); + const Object& request = valRequest.get_obj(); + if (find_value(request, "method").type() != str_type || + find_value(request, "params").type() != array_type) + throw runtime_error("Invalid request."); + + string strMethod = find_value(request, "method").get_str(); + const Array& params = find_value(request, "params").get_array(); + id = find_value(request, "id"); + + // Execute + map::iterator mi = mapCallTable.find(strMethod); + if (mi == mapCallTable.end()) + throw runtime_error("Method not found."); + Value result = (*(*mi).second)(params); + + // Send reply + string strReply = JSONRPCReply(result, Value::null, id); + stream << HTTPReply(strReply, 200) << std::flush; + } + catch (std::exception& e) + { + // Send error reply + string strReply = JSONRPCReply(Value::null, e.what(), id); + stream << HTTPReply(strReply, 500) << std::flush; + } + if (begin == prev) + break; + } + } +} + + + + +Value CallRPC(const string& strMethod, const Array& params) +{ + // Connect to localhost + tcp::iostream stream("127.0.0.1", "8332"); + if (stream.fail()) + throw runtime_error("unable to connect to server"); + + // Send request + string strRequest = JSONRPCRequest(strMethod, params, 1); + stream << HTTPPost(strRequest) << std::flush; + + // Receive reply + string strReply = ReadHTTP(stream); + if (strReply.empty()) + throw runtime_error("no response from server"); + + // Parse reply + Value valReply; + if (!read_string(strReply, valReply)) + throw runtime_error("couldn't parse reply from server"); + const Object& reply = valReply.get_obj(); + if (reply.empty()) + throw runtime_error("expected reply to have result, error and id properties"); + + const Value& result = find_value(reply, "result"); + const Value& error = find_value(reply, "error"); + const Value& id = find_value(reply, "id"); + + if (error.type() == str_type) + throw runtime_error(error.get_str()); + else if (error.type() != null_type) + throw runtime_error(write_string(error, false)); + return result; +} + + + + +template +void ConvertTo(Value& value) +{ + if (value.type() == str_type) + { + // reinterpret string as unquoted json value + Value value2; + if (!read_string(value.get_str(), value2)) + throw runtime_error("type mismatch"); + value = value2.get_value(); + } + else + { + value = value.get_value(); + } +} + +int CommandLineRPC(int argc, char *argv[]) +{ + try + { + // Check that method exists + if (argc < 2) + throw runtime_error("too few parameters"); + string strMethod = argv[1]; + if (!mapCallTable.count(strMethod)) + throw runtime_error(strprintf("unknown command: %s", strMethod.c_str())); + + // Parameters default to strings + Array params; + for (int i = 2; i < argc; i++) + params.push_back(argv[i]); + + // Special case other types + int n = params.size(); + if (strMethod == "sendtoaddress" && n > 1) ConvertTo(params[1]); + if (strMethod == "listtransactions" && n > 0) ConvertTo(params[0]); + if (strMethod == "listtransactions" && n > 1) ConvertTo(params[1]); + if (strMethod == "getamountpaid" && n > 1) ConvertTo(params[1]); + if (strMethod == "getallpayments" && n > 0) ConvertTo(params[0]); + + // Execute + Value result = CallRPC(strMethod, params); + + // Print result + string strResult = (result.type() == str_type ? result.get_str() : write_string(result, true)); + if (result.type() != null_type) + { + if (fWindows) + // Windows GUI apps can't print to command line, + // so for now settle for a message box yuck + wxMessageBox(strResult.c_str(), "Bitcoin", wxOK); + else + fprintf(stdout, "%s\n", strResult.c_str()); + } + return 0; + } + catch (std::exception& e) { + if (fWindows) + wxMessageBox(strprintf("error: %s\n", e.what()).c_str(), "Bitcoin", wxOK); + else + fprintf(stderr, "error: %s\n", e.what()); + } catch (...) { + PrintException(NULL, "CommandLineRPC()"); + } + return 1; +} + + + + +#ifdef TEST +int main(int argc, char *argv[]) +{ +#ifdef _MSC_VER + // Turn off microsoft heap dump noise + _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE); + _CrtSetReportFile(_CRT_WARN, CreateFile("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); +#endif + setbuf(stdin, NULL); + setbuf(stdout, NULL); + setbuf(stderr, NULL); + + try + { + if (argc >= 2 && string(argv[1]) == "-server") + { + printf("server ready\n"); + ThreadRPCServer(NULL); + } + else + { + return CommandLineRPC(argc, argv); + } + } + catch (std::exception& e) { + PrintException(&e, "main()"); + } catch (...) { + PrintException(NULL, "main()"); + } + return 0; +} +#endif diff --git a/rpc.h b/rpc.h new file mode 100644 index 00000000..81ad840c --- /dev/null +++ b/rpc.h @@ -0,0 +1,6 @@ +// Copyright (c) 2010 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +void ThreadRPCServer(void* parg); +int CommandLineRPC(int argc, char *argv[]); diff --git a/script.cpp b/script.cpp index a41de2aa..98f5afd5 100644 --- a/script.cpp +++ b/script.cpp @@ -660,7 +660,7 @@ bool EvalScript(const CScript& script, const CTransaction& txTo, unsigned int nI if (stack.size() < 1) return false; valtype& vch = stacktop(-1); - valtype vchHash(opcode == OP_RIPEMD160 || opcode == OP_SHA1 || opcode == OP_HASH160 ? 20 : 32); + valtype vchHash((opcode == OP_RIPEMD160 || opcode == OP_SHA1 || opcode == OP_HASH160) ? 20 : 32); if (opcode == OP_RIPEMD160) RIPEMD160(&vch[0], vch.size(), &vchHash[0]); else if (opcode == OP_SHA1) @@ -753,9 +753,9 @@ bool EvalScript(const CScript& script, const CTransaction& txTo, unsigned int nI CScript scriptCode(pbegincodehash, pend); // Drop the signatures, since there's no way for a signature to sign itself - for (int i = 0; i < nSigsCount; i++) + for (int k = 0; k < nSigsCount; k++) { - valtype& vchSig = stacktop(-isig-i); + valtype& vchSig = stacktop(-isig-k); scriptCode.FindAndDelete(CScript(vchSig)); } @@ -909,7 +909,6 @@ bool CheckSig(vector vchSig, vector vchPubKey, CSc - bool Solver(const CScript& scriptPubKey, vector >& vSolutionRet) { // Templates @@ -919,7 +918,7 @@ bool Solver(const CScript& scriptPubKey, vector >& vSo // Standard tx, sender provides pubkey, receiver adds signature vTemplates.push_back(CScript() << OP_PUBKEY << OP_CHECKSIG); - // Short account number tx, sender provides hash of pubkey, receiver provides signature and pubkey + // Bitcoin address tx, sender provides hash of pubkey, receiver provides signature and pubkey vTemplates.push_back(CScript() << OP_DUP << OP_HASH160 << OP_PUBKEYHASH << OP_EQUALVERIFY << OP_CHECKSIG); } diff --git a/script.h b/script.h index 0d977734..9e418891 100644 --- a/script.h +++ b/script.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -472,7 +472,7 @@ public: bool GetOp(iterator& pc, opcodetype& opcodeRet, vector& vchRet) { - // This is why people hate C++ + // Wrapper so it can be called with either iterator or const_iterator const_iterator pc2 = pc; bool fRet = GetOp(pc2, opcodeRet, vchRet); pc = begin() + (pc2 - begin()); @@ -551,6 +551,46 @@ public: } + uint160 GetBitcoinAddressHash160() const + { + opcodetype opcode; + vector vch; + CScript::const_iterator pc = begin(); + if (!GetOp(pc, opcode, vch) || opcode != OP_DUP) return 0; + if (!GetOp(pc, opcode, vch) || opcode != OP_HASH160) return 0; + if (!GetOp(pc, opcode, vch) || vch.size() != sizeof(uint160)) return 0; + uint160 hash160 = uint160(vch); + if (!GetOp(pc, opcode, vch) || opcode != OP_EQUALVERIFY) return 0; + if (!GetOp(pc, opcode, vch) || opcode != OP_CHECKSIG) return 0; + if (pc != end()) return 0; + return hash160; + } + + string GetBitcoinAddress() const + { + uint160 hash160 = GetBitcoinAddressHash160(); + if (hash160 == 0) + return ""; + return Hash160ToAddress(hash160); + } + + void SetBitcoinAddress(const uint160& hash160) + { + this->clear(); + *this << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; + } + + bool SetBitcoinAddress(const string& strAddress) + { + this->clear(); + uint160 hash160; + if (!AddressToHash160(strAddress, hash160)) + return false; + SetBitcoinAddress(hash160); + return true; + } + + void PrintHex() const { printf("CScript(%s)\n", HexStr(begin(), end()).c_str()); diff --git a/serialize.h b/serialize.h index 439ef641..eb090ae2 100644 --- a/serialize.h +++ b/serialize.h @@ -19,8 +19,8 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 200; -static const char* pszSubVer = " test2"; +static const int VERSION = 201; +static const char* pszSubVer = ".0"; diff --git a/ui.cpp b/ui.cpp index 5d93ad22..586200d6 100644 --- a/ui.cpp +++ b/ui.cpp @@ -21,7 +21,6 @@ DEFINE_EVENT_TYPE(wxEVT_REPLY3) CMainFrame* pframeMain = NULL; CMyTaskBarIcon* ptaskbaricon = NULL; -map mapAddressBook; bool fRandSendTest = false; void RandSend(); extern int g_isPainting; @@ -177,8 +176,11 @@ void CalledMessageBox(const string& message, const string& caption, int style, w int ThreadSafeMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) { - if (mapArgs.count("-noui")) + if (fDaemon) + { + printf("wxMessageBox %s: %s\n", caption.c_str(), message.c_str()); return wxOK; + } #ifdef __WXMSW__ return wxMessageBox(message, caption, style, parent, x, y); @@ -413,7 +415,7 @@ void Shutdown(void* parg) StopNode(); DBFlush(true); CreateThread(ExitTimeout, NULL); - Sleep(10); + Sleep(50); printf("Bitcoin exiting\n\n"); fExit = true; exit(0); @@ -697,19 +699,22 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) vector vchPubKey; if (ExtractPubKey(txout.scriptPubKey, true, vchPubKey)) { - string strAddress = PubKeyToAddress(vchPubKey); - if (mapAddressBook.count(strAddress)) + CRITICAL_BLOCK(cs_mapAddressBook) { - //strDescription += "Received payment to "; - //strDescription += "Received with address "; - strDescription += "From: unknown, To: "; - strDescription += strAddress; - /// The labeling feature is just too confusing, so I hid it - /// by putting it at the end where it runs off the screen. - /// It can still be seen by widening the column, or in the - /// details dialog. - if (!mapAddressBook[strAddress].empty()) - strDescription += " (" + mapAddressBook[strAddress] + ")"; + string strAddress = PubKeyToAddress(vchPubKey); + if (mapAddressBook.count(strAddress)) + { + //strDescription += "Received payment to "; + //strDescription += "Received with address "; + strDescription += "From: unknown, To: "; + strDescription += strAddress; + /// The labeling feature is just too confusing, so I hid it + /// by putting it at the end where it runs off the screen. + /// It can still be seen by widening the column, or in the + /// details dialog. + if (!mapAddressBook[strAddress].empty()) + strDescription += " (" + mapAddressBook[strAddress] + ")"; + } } } break; @@ -776,8 +781,9 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) } string strDescription = "To: "; - if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) - strDescription += mapAddressBook[strAddress] + " "; + CRITICAL_BLOCK(cs_mapAddressBook) + if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) + strDescription += mapAddressBook[strAddress] + " "; strDescription += strAddress; if (!mapValue["message"].empty()) { @@ -1273,238 +1279,241 @@ void CMainFrame::OnListItemActivatedOrdersReceived(wxListEvent& event) CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetailsDialogBase(parent) { - string strHTML; - strHTML.reserve(4000); - strHTML += ""; + CRITICAL_BLOCK(cs_mapAddressBook) + { + string strHTML; + strHTML.reserve(4000); + strHTML += ""; - int64 nTime = wtx.GetTxTime(); - int64 nCredit = wtx.GetCredit(); - int64 nDebit = wtx.GetDebit(); - int64 nNet = nCredit - nDebit; + int64 nTime = wtx.GetTxTime(); + int64 nCredit = wtx.GetCredit(); + int64 nDebit = wtx.GetDebit(); + int64 nNet = nCredit - nDebit; - strHTML += "Status: " + FormatTxStatus(wtx); - int nRequests = wtx.GetRequestCount(); - if (nRequests != -1) - { - if (nRequests == 0) - strHTML += ", has not been successfully broadcast yet"; - else if (nRequests == 1) - strHTML += strprintf(", broadcast through %d node", nRequests); - else - strHTML += strprintf(", broadcast through %d nodes", nRequests); - } - strHTML += "
"; + strHTML += "Status: " + FormatTxStatus(wtx); + int nRequests = wtx.GetRequestCount(); + if (nRequests != -1) + { + if (nRequests == 0) + strHTML += ", has not been successfully broadcast yet"; + else if (nRequests == 1) + strHTML += strprintf(", broadcast through %d node", nRequests); + else + strHTML += strprintf(", broadcast through %d nodes", nRequests); + } + strHTML += "
"; - strHTML += "Date: " + (nTime ? DateTimeStr(nTime) : "") + "
"; + strHTML += "Date: " + (nTime ? DateTimeStr(nTime) : "") + "
"; - // - // From - // - if (wtx.IsCoinBase()) - { - strHTML += "Source: Generated
"; - } - else if (!wtx.mapValue["from"].empty()) - { - // Online transaction - if (!wtx.mapValue["from"].empty()) - strHTML += "From: " + HtmlEscape(wtx.mapValue["from"]) + "
"; - } - else - { - // Offline transaction - if (nNet > 0) + // + // From + // + if (wtx.IsCoinBase()) { - // Credit - foreach(const CTxOut& txout, wtx.vout) + strHTML += "Source: Generated
"; + } + else if (!wtx.mapValue["from"].empty()) + { + // Online transaction + if (!wtx.mapValue["from"].empty()) + strHTML += "From: " + HtmlEscape(wtx.mapValue["from"]) + "
"; + } + else + { + // Offline transaction + if (nNet > 0) { - if (txout.IsMine()) + // Credit + foreach(const CTxOut& txout, wtx.vout) { - vector vchPubKey; - if (ExtractPubKey(txout.scriptPubKey, true, vchPubKey)) + if (txout.IsMine()) { - string strAddress = PubKeyToAddress(vchPubKey); - if (mapAddressBook.count(strAddress)) + vector vchPubKey; + if (ExtractPubKey(txout.scriptPubKey, true, vchPubKey)) { - strHTML += "From: unknown
"; - strHTML += "To: "; - strHTML += HtmlEscape(strAddress); - if (!mapAddressBook[strAddress].empty()) - strHTML += " (yours, label: " + mapAddressBook[strAddress] + ")"; - else - strHTML += " (yours)"; - strHTML += "
"; + string strAddress = PubKeyToAddress(vchPubKey); + if (mapAddressBook.count(strAddress)) + { + strHTML += "From: unknown
"; + strHTML += "To: "; + strHTML += HtmlEscape(strAddress); + if (!mapAddressBook[strAddress].empty()) + strHTML += " (yours, label: " + mapAddressBook[strAddress] + ")"; + else + strHTML += " (yours)"; + strHTML += "
"; + } } + break; } - break; } } } - } - // - // To - // - string strAddress; - if (!wtx.mapValue["to"].empty()) - { - // Online transaction - strAddress = wtx.mapValue["to"]; - strHTML += "To: "; - if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) - strHTML += mapAddressBook[strAddress] + " "; - strHTML += HtmlEscape(strAddress) + "
"; - } - - - // - // Amount - // - if (wtx.IsCoinBase() && nCredit == 0) - { - // - // Coinbase - // - int64 nUnmatured = 0; - foreach(const CTxOut& txout, wtx.vout) - nUnmatured += txout.GetCredit(); - if (wtx.IsInMainChain()) - strHTML += strprintf("Credit: (%s matures in %d more blocks)
", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); - else - strHTML += "Credit: (not accepted)
"; - } - else if (nNet > 0) - { // - // Credit + // To // - strHTML += "Credit: " + FormatMoney(nNet) + "
"; - } - else - { - bool fAllFromMe = true; - foreach(const CTxIn& txin, wtx.vin) - fAllFromMe = fAllFromMe && txin.IsMine(); + string strAddress; + if (!wtx.mapValue["to"].empty()) + { + // Online transaction + strAddress = wtx.mapValue["to"]; + strHTML += "To: "; + if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) + strHTML += mapAddressBook[strAddress] + " "; + strHTML += HtmlEscape(strAddress) + "
"; + } - bool fAllToMe = true; - foreach(const CTxOut& txout, wtx.vout) - fAllToMe = fAllToMe && txout.IsMine(); - if (fAllFromMe) + // + // Amount + // + if (wtx.IsCoinBase() && nCredit == 0) { // - // Debit + // Coinbase // + int64 nUnmatured = 0; foreach(const CTxOut& txout, wtx.vout) - { - if (txout.IsMine()) - continue; + nUnmatured += txout.GetCredit(); + if (wtx.IsInMainChain()) + strHTML += strprintf("Credit: (%s matures in %d more blocks)
", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); + else + strHTML += "Credit: (not accepted)
"; + } + else if (nNet > 0) + { + // + // Credit + // + strHTML += "Credit: " + FormatMoney(nNet) + "
"; + } + else + { + bool fAllFromMe = true; + foreach(const CTxIn& txin, wtx.vin) + fAllFromMe = fAllFromMe && txin.IsMine(); + + bool fAllToMe = true; + foreach(const CTxOut& txout, wtx.vout) + fAllToMe = fAllToMe && txout.IsMine(); - if (wtx.mapValue["to"].empty()) + if (fAllFromMe) + { + // + // Debit + // + foreach(const CTxOut& txout, wtx.vout) { - // Offline transaction - uint160 hash160; - if (ExtractHash160(txout.scriptPubKey, hash160)) + if (txout.IsMine()) + continue; + + if (wtx.mapValue["to"].empty()) { - string strAddress = Hash160ToAddress(hash160); - strHTML += "To: "; - if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) - strHTML += mapAddressBook[strAddress] + " "; - strHTML += strAddress; - strHTML += "
"; + // Offline transaction + uint160 hash160; + if (ExtractHash160(txout.scriptPubKey, hash160)) + { + string strAddress = Hash160ToAddress(hash160); + strHTML += "To: "; + if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) + strHTML += mapAddressBook[strAddress] + " "; + strHTML += strAddress; + strHTML += "
"; + } } + + strHTML += "Debit: " + FormatMoney(-txout.nValue) + "
"; } - strHTML += "Debit: " + FormatMoney(-txout.nValue) + "
"; - } + if (fAllToMe) + { + // Payment to self + /// issue: can't tell which is the payment and which is the change anymore + //int64 nValue = wtx.vout[0].nValue; + //strHTML += "Debit: " + FormatMoney(-nValue) + "
"; + //strHTML += "Credit: " + FormatMoney(nValue) + "
"; + } - if (fAllToMe) + int64 nTxFee = nDebit - wtx.GetValueOut(); + if (nTxFee > 0) + strHTML += "Transaction fee: " + FormatMoney(-nTxFee) + "
"; + } + else { - // Payment to self - /// issue: can't tell which is the payment and which is the change anymore - //int64 nValue = wtx.vout[0].nValue; - //strHTML += "Debit: " + FormatMoney(-nValue) + "
"; - //strHTML += "Credit: " + FormatMoney(nValue) + "
"; + // + // Mixed debit transaction + // + foreach(const CTxIn& txin, wtx.vin) + if (txin.IsMine()) + strHTML += "Debit: " + FormatMoney(-txin.GetDebit()) + "
"; + foreach(const CTxOut& txout, wtx.vout) + if (txout.IsMine()) + strHTML += "Credit: " + FormatMoney(txout.GetCredit()) + "
"; } - - int64 nTxFee = nDebit - wtx.GetValueOut(); - if (nTxFee > 0) - strHTML += "Transaction fee: " + FormatMoney(-nTxFee) + "
"; } - else - { - // - // Mixed debit transaction - // - foreach(const CTxIn& txin, wtx.vin) - if (txin.IsMine()) - strHTML += "Debit: " + FormatMoney(-txin.GetDebit()) + "
"; - foreach(const CTxOut& txout, wtx.vout) - if (txout.IsMine()) - strHTML += "Credit: " + FormatMoney(txout.GetCredit()) + "
"; - } - } - strHTML += "Net amount: " + FormatMoney(nNet, true) + "
"; + strHTML += "Net amount: " + FormatMoney(nNet, true) + "
"; - // - // Message - // - if (!wtx.mapValue["message"].empty()) - strHTML += "
Message:
" + HtmlEscape(wtx.mapValue["message"], true) + "
"; - - if (wtx.IsCoinBase()) - strHTML += "
Generated coins must wait 120 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, it will change to \"not accepted\" and not be spendable. This may occasionally happen if another node generates a block within a few seconds of yours.
"; + // + // Message + // + if (!wtx.mapValue["message"].empty()) + strHTML += "
Message:
" + HtmlEscape(wtx.mapValue["message"], true) + "
"; + if (wtx.IsCoinBase()) + strHTML += "
Generated coins must wait 120 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, it will change to \"not accepted\" and not be spendable. This may occasionally happen if another node generates a block within a few seconds of yours.
"; - // - // Debug view - // - if (fDebug) - { - strHTML += "

debug print

"; - foreach(const CTxIn& txin, wtx.vin) - if (txin.IsMine()) - strHTML += "Debit: " + FormatMoney(-txin.GetDebit()) + "
"; - foreach(const CTxOut& txout, wtx.vout) - if (txout.IsMine()) - strHTML += "Credit: " + FormatMoney(txout.GetCredit()) + "
"; - strHTML += "Inputs:
"; - CRITICAL_BLOCK(cs_mapWallet) + // + // Debug view + // + if (fDebug) { + strHTML += "

debug print

"; foreach(const CTxIn& txin, wtx.vin) + if (txin.IsMine()) + strHTML += "Debit: " + FormatMoney(-txin.GetDebit()) + "
"; + foreach(const CTxOut& txout, wtx.vout) + if (txout.IsMine()) + strHTML += "Credit: " + FormatMoney(txout.GetCredit()) + "
"; + + strHTML += "Inputs:
"; + CRITICAL_BLOCK(cs_mapWallet) { - COutPoint prevout = txin.prevout; - map::iterator mi = mapWallet.find(prevout.hash); - if (mi != mapWallet.end()) + foreach(const CTxIn& txin, wtx.vin) { - const CWalletTx& prev = (*mi).second; - if (prevout.n < prev.vout.size()) + COutPoint prevout = txin.prevout; + map::iterator mi = mapWallet.find(prevout.hash); + if (mi != mapWallet.end()) { - strHTML += HtmlEscape(prev.ToString(), true); - strHTML += "    " + FormatTxStatus(prev) + ", "; - strHTML = strHTML + "IsMine=" + (prev.vout[prevout.n].IsMine() ? "true" : "false") + "
"; + const CWalletTx& prev = (*mi).second; + if (prevout.n < prev.vout.size()) + { + strHTML += HtmlEscape(prev.ToString(), true); + strHTML += "    " + FormatTxStatus(prev) + ", "; + strHTML = strHTML + "IsMine=" + (prev.vout[prevout.n].IsMine() ? "true" : "false") + "
"; + } } } } - } - strHTML += "


Transaction:
"; - strHTML += HtmlEscape(wtx.ToString(), true); - } + strHTML += "


Transaction:
"; + strHTML += HtmlEscape(wtx.ToString(), true); + } - strHTML += "
"; - string(strHTML.begin(), strHTML.end()).swap(strHTML); - m_htmlWin->SetPage(strHTML); - m_buttonOK->SetFocus(); + strHTML += "
"; + string(strHTML.begin(), strHTML.end()).swap(strHTML); + m_htmlWin->SetPage(strHTML); + m_buttonOK->SetFocus(); + } } void CTxDetailsDialog::OnButtonOK(wxCommandEvent& event) @@ -1686,9 +1695,10 @@ CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) #if !wxUSE_UNICODE // Workaround until upgrade to wxWidgets supporting UTF-8 + // Hack to change the (c) character from UTF-8 back to ANSI wxString str = m_staticTextMain->GetLabel(); - if (str.Find('Â') != wxNOT_FOUND) - str.Remove(str.Find('Â'), 1); + if (str.Find('\xC2') != wxNOT_FOUND) + str.Remove(str.Find('\xC2'), 1); m_staticTextMain->SetLabel(str); #endif #ifndef __WXMSW__ @@ -1843,10 +1853,11 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event) CScript scriptPubKey; scriptPubKey << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; - if (!SendMoney(scriptPubKey, nValue, wtx)) - return; - - wxMessageBox("Payment sent ", "Sending..."); + string strError = SendMoney(scriptPubKey, nValue, wtx); + if (strError != "") + wxMessageBox(strError + " ", "Sending..."); + else + wxMessageBox("Payment sent ", "Sending..."); } else { @@ -1869,8 +1880,9 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event) return; } - if (!mapAddressBook.count(strAddress)) - SetAddressBookName(strAddress, ""); + CRITICAL_BLOCK(cs_mapAddressBook) + if (!mapAddressBook.count(strAddress)) + SetAddressBookName(strAddress, ""); EndModal(true); } @@ -2227,6 +2239,7 @@ CYourAddressDialog::CYourAddressDialog(wxWindow* parent, const string& strInitSe // Fill listctrl with address book data CRITICAL_BLOCK(cs_mapKeys) + CRITICAL_BLOCK(cs_mapAddressBook) { foreach(const PAIRTYPE(string, string)& item, mapAddressBook) { @@ -2366,6 +2379,7 @@ CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInit // Fill listctrl with address book data CRITICAL_BLOCK(cs_mapKeys) + CRITICAL_BLOCK(cs_mapAddressBook) { foreach(const PAIRTYPE(string, string)& item, mapAddressBook) { @@ -3515,6 +3529,12 @@ bool CMyApp::OnInit2() // // Parameters // + if (argc > 1 && argv[1][0] != '-' && argv[1][0] != '/') + { + int ret = CommandLineRPC(argc, argv); + exit(ret); + } + ParseParameters(argc, argv); if (mapArgs.count("-?") || mapArgs.count("--help")) { @@ -3557,6 +3577,13 @@ bool CMyApp::OnInit2() if (mapArgs.count("-printtodebugger")) fPrintToDebugger = true; + if (mapArgs.count("-daemon") || mapArgs.count("-d")) + { + fDaemon = true; + /// todo: need to fork + /// should it fork after the bind/single instance stuff? + } + if (!fDebug && !pszSetDataDir[0]) ShrinkDebugFile(); printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); @@ -3730,7 +3757,7 @@ bool CMyApp::OnInit2() // // Create the main frame window // - if (!mapArgs.count("-noui")) + if (!fDaemon) { pframeMain = new CMainFrame(NULL); if (mapArgs.count("-min")) @@ -3752,6 +3779,9 @@ bool CMyApp::OnInit2() if (!CreateThread(StartNode, NULL)) wxMessageBox("Error: CreateThread(StartNode) failed", "Bitcoin"); + if (mapArgs.count("-server") || fDaemon) + CreateThread(ThreadRPCServer, NULL); + if (fFirstRun) SetStartOnSystemStartup(true); diff --git a/util.cpp b/util.cpp index 42922467..2f5be222 100644 --- a/util.cpp +++ b/util.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -12,6 +12,7 @@ bool fPrintToConsole = false; bool fPrintToDebugger = false; char pszSetDataDir[MAX_PATH] = ""; bool fShutdown = false; +bool fDaemon = false; diff --git a/util.h b/util.h index d20648bc..6a7fabc5 100644 --- a/util.h +++ b/util.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 Satoshi Nakamoto +// Copyright (c) 2009-2010 Satoshi Nakamoto // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. @@ -55,6 +55,7 @@ inline T& REF(const T& val) } #ifdef __WXMSW__ +static const bool fWindows = true; #define MSG_NOSIGNAL 0 #define MSG_DONTWAIT 0 #ifndef UINT64_MAX @@ -66,7 +67,9 @@ inline T& REF(const T& val) #define S_IRUSR 0400 #define S_IWUSR 0200 #endif +#define unlink _unlink #else +static const bool fWindows = false; #define WSAGetLastError() errno #define WSAEWOULDBLOCK EWOULDBLOCK #define WSAEMSGSIZE EMSGSIZE @@ -116,6 +119,7 @@ extern bool fPrintToConsole; extern bool fPrintToDebugger; extern char pszSetDataDir[MAX_PATH]; extern bool fShutdown; +extern bool fDaemon; void RandAddSeed(); void RandAddSeedPerfmon(); @@ -258,6 +262,11 @@ inline int roundint(double d) return (int)(d > 0 ? d + 0.5 : d - 0.5); } +inline int64 roundint64(double d) +{ + return (int64)(d > 0 ? d + 0.5 : d - 0.5); +} + template string HexStr(const T itbegin, const T itend, bool fSpaces=true) { @@ -323,7 +332,12 @@ inline string DateTimeStrFormat(const char* pszFormat, int64 nTime) return pszTime; } - +template +void skipspaces(T& it) +{ + while (isspace(*it)) + ++it; +} From c85dfb148a1e3d75773bc5f6b7cad14363fd666b Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sat, 13 Feb 2010 02:09:07 +0000 Subject: [PATCH 049/133] updated build-unix.txt git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@61 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-unix.txt | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/build-unix.txt b/build-unix.txt index 5467bbea..f1a72eaa 100644 --- a/build-unix.txt +++ b/build-unix.txt @@ -20,11 +20,15 @@ sudo apt-get install libdb4.7-dev sudo apt-get install libdb4.7++-dev sudo apt-get install libboost-dev +The release was built with wxWidgets 2.8.9 ansi on 32-bit. The current +sourcecode can be built on 64-bit with wxWidgets 2.9.0. + There is currently no libwxgtk2.8-ansi-dev debian package for Karmic. libwxgtk2.8-dev is the "unicode" build, but for wxWidgets 2.8 "unicode" means wchar, not UTF-8. wchar wxString doesn't convert to std::string. +We haven't been able to compile the 2.8 versions on 64-bit. -In wxWidgets 2.9, unicode is UTF-8 and that's the only version. +wxWidgets 2.9 is UTF-8 and compiles on 64-bit. You need to download wxWidgets from http://www.wxwidgets.org/downloads/ and build it yourself. See the build instructions and configure parameters @@ -60,7 +64,7 @@ mkdir buildgtk cd buildgtk ../configure --with-gtk --enable-debug --disable-shared --enable-monolithic make -su +sudo su make install ldconfig From c4319e678f693d5fbc49bd357ded1c8f951476e9 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 14 Feb 2010 00:08:27 +0000 Subject: [PATCH 050/133] Workaround for bug on wxWidgets 2.9.0 Ubuntu 9.10 64-bit where first character of the hidden columns were displayed so status column had three numbers overprinted. Fixed by adding a leading space to the hidden columns. 64-bit compile with wxWidgets 2.9.0 seems to be fully working normally now. git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@62 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- bignum.h | 8 ++------ main.cpp | 11 +++++------ net.cpp | 2 +- rpc.cpp | 2 +- ui.cpp | 40 ++++++++++++---------------------------- uint256.h | 14 +++++++++----- 6 files changed, 30 insertions(+), 47 deletions(-) diff --git a/bignum.h b/bignum.h index e1ab165b..61ba3fa0 100644 --- a/bignum.h +++ b/bignum.h @@ -64,12 +64,6 @@ public: } } - explicit CBigNum(const std::string& str) - { - BN_init(this); - SetHex(str); - } - CBigNum& operator=(const CBigNum& b) { if (!BN_copy(this, &b)) @@ -407,6 +401,7 @@ public: CBigNum& operator>>=(unsigned int shift) { + // Note: BN_rshift segfaults on 64-bit ubuntu 9.10 if 2^shift is greater than the number if (!BN_rshift(this, this, shift)) throw bignum_error("CBigNum:operator>>= : BN_rshift failed"); return *this; @@ -516,6 +511,7 @@ inline const CBigNum operator<<(const CBigNum& a, unsigned int shift) inline const CBigNum operator>>(const CBigNum& a, unsigned int shift) { CBigNum r; + // Note: BN_rshift segfaults on 64-bit ubuntu 9.10 if 2^shift is greater than the number if (!BN_rshift(&r, &a, shift)) throw bignum_error("CBigNum:operator>> : BN_rshift failed"); return r; diff --git a/main.cpp b/main.cpp index 90d239fa..0c7aff84 100644 --- a/main.cpp +++ b/main.cpp @@ -1570,7 +1570,9 @@ bool LoadBlockIndex(bool fAllowNew) txNew.vout.resize(1); txNew.vin[0].scriptSig = CScript() << 486604799 << CBigNum(4) << vector((const unsigned char*)pszTimestamp, (const unsigned char*)pszTimestamp + strlen(pszTimestamp)); txNew.vout[0].nValue = 50 * COIN; - txNew.vout[0].scriptPubKey = CScript() << CBigNum("0x5F1DF16B2B704C8A578D0BBAF74D385CDE12C11EE50455F3C438EF4C3FBCF649B6DE611FEAE06279A60939E028A8D65C10B73071A6F16719274855FEB0FD8A6704") << OP_CHECKSIG; + CBigNum bnPubKey; + bnPubKey.SetHex("0x5F1DF16B2B704C8A578D0BBAF74D385CDE12C11EE50455F3C438EF4C3FBCF649B6DE611FEAE06279A60939E028A8D65C10B73071A6F16719274855FEB0FD8A6704"); + txNew.vout[0].scriptPubKey = CScript() << bnPubKey << OP_CHECKSIG; CBlock block; block.vtx.push_back(txNew); block.hashPrevBlock = 0; @@ -3022,12 +3024,9 @@ string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtx return "You don't have enough money"; // Parse bitcoin address - uint160 hash160; - if (!AddressToHash160(strAddress, hash160)) + CScript scriptPubKey; + if (!scriptPubKey.SetBitcoinAddress(strAddress)) return "Invalid bitcoin address"; - // Send to bitcoin address - CScript scriptPubKey; - scriptPubKey.SetBitcoinAddress(hash160); return SendMoney(scriptPubKey, nValue, wtxNew); } diff --git a/net.cpp b/net.cpp index ada78eb3..d669b8c5 100644 --- a/net.cpp +++ b/net.cpp @@ -1032,7 +1032,7 @@ void ThreadMessageHandler2(void* parg) { printf("ThreadMessageHandler started\n"); SetThreadPriority(THREAD_PRIORITY_BELOW_NORMAL); - loop + while (!fShutdown) { // Poll the connected nodes for messages vector vNodesCopy; diff --git a/rpc.cpp b/rpc.cpp index 9f28e7ec..b176b158 100644 --- a/rpc.cpp +++ b/rpc.cpp @@ -503,7 +503,7 @@ Value CallRPC(const string& strMethod, const Array& params) // Connect to localhost tcp::iostream stream("127.0.0.1", "8332"); if (stream.fail()) - throw runtime_error("unable to connect to server"); + throw runtime_error("couldn't connect to server"); // Send request string strRequest = JSONRPCRequest(strMethod, params, 1); diff --git a/ui.cpp b/ui.cpp index 586200d6..aad599c7 100644 --- a/ui.cpp +++ b/ui.cpp @@ -328,7 +328,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_toolBar->Realize(); // resize to fit ubuntu's huge default font dResize = 1.20; - SetSize(dResize * GetSize().GetWidth(), 1.1 * GetSize().GetHeight()); + SetSize((dResize + 0.02) * GetSize().GetWidth(), 1.09 * GetSize().GetHeight()); #endif m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); @@ -346,24 +346,6 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_listCtrl->InsertColumn(5, "Debit", wxLIST_FORMAT_RIGHT, dResize * 79); m_listCtrl->InsertColumn(6, "Credit", wxLIST_FORMAT_RIGHT, dResize * 79); - //m_listCtrlProductsSent->InsertColumn(0, "Category", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlProductsSent->InsertColumn(1, "Title", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlProductsSent->InsertColumn(2, "Description", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlProductsSent->InsertColumn(3, "Price", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlProductsSent->InsertColumn(4, "", wxLIST_FORMAT_LEFT, 100); - - //m_listCtrlOrdersSent->InsertColumn(0, "Time", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersSent->InsertColumn(1, "Price", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersSent->InsertColumn(2, "", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersSent->InsertColumn(3, "", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersSent->InsertColumn(4, "", wxLIST_FORMAT_LEFT, 100); - - //m_listCtrlOrdersReceived->InsertColumn(0, "Time", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersReceived->InsertColumn(1, "Price", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersReceived->InsertColumn(2, "Payment Status", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersReceived->InsertColumn(3, "", wxLIST_FORMAT_LEFT, 100); - //m_listCtrlOrdersReceived->InsertColumn(4, "", wxLIST_FORMAT_LEFT, 100); - // Init status bar int pnWidths[3] = { -100, 88, 290 }; #ifndef __WXMSW__ @@ -503,33 +485,34 @@ int CMainFrame::GetSortIndex(const string& strSort) void CMainFrame::InsertLine(bool fNew, int nIndex, uint256 hashKey, string strSort, const wxString& str2, const wxString& str3, const wxString& str4, const wxString& str5, const wxString& str6) { - string str0 = strSort; - long nData = *(long*)&hashKey; + strSort = " " + strSort; // leading space to workaround wx2.9.0 ubuntu 9.10 bug + long nData = *(long*)&hashKey; // where first char of hidden column is displayed // Find item if (!fNew && nIndex == -1) { + string strHash = " " + hashKey.ToString(); while ((nIndex = m_listCtrl->FindItem(nIndex, nData)) != -1) - if (GetItemText(m_listCtrl, nIndex, 1) == hashKey.ToString()) + if (GetItemText(m_listCtrl, nIndex, 1) == strHash) break; } // fNew is for blind insert, only use if you're sure it's new if (fNew || nIndex == -1) { - nIndex = m_listCtrl->InsertItem(GetSortIndex(strSort), str0); + nIndex = m_listCtrl->InsertItem(GetSortIndex(strSort), strSort); } else { // If sort key changed, must delete and reinsert to make it relocate - if (GetItemText(m_listCtrl, nIndex, 0) != str0) + if (GetItemText(m_listCtrl, nIndex, 0) != strSort) { m_listCtrl->DeleteItem(nIndex); - nIndex = m_listCtrl->InsertItem(GetSortIndex(strSort), str0); + nIndex = m_listCtrl->InsertItem(GetSortIndex(strSort), strSort); } } - m_listCtrl->SetItem(nIndex, 1, hashKey.ToString()); + m_listCtrl->SetItem(nIndex, 1, " " + hashKey.ToString()); m_listCtrl->SetItem(nIndex, 2, str2); m_listCtrl->SetItem(nIndex, 3, str3); m_listCtrl->SetItem(nIndex, 4, str4); @@ -544,8 +527,9 @@ bool CMainFrame::DeleteLine(uint256 hashKey) // Find item int nIndex = -1; + string strHash = " " + hashKey.ToString(); while ((nIndex = m_listCtrl->FindItem(nIndex, nData)) != -1) - if (GetItemText(m_listCtrl, nIndex, 1) == hashKey.ToString()) + if (GetItemText(m_listCtrl, nIndex, 1) == strHash) break; if (nIndex != -1) @@ -1916,7 +1900,7 @@ CSendingDialog::CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 n fUIDone = false; fWorkDone = false; #ifndef __WXMSW__ - SetSize(1.2 * GetSize().GetWidth(), 1.05 * GetSize().GetHeight()); + SetSize(1.2 * GetSize().GetWidth(), 1.08 * GetSize().GetHeight()); #endif SetTitle(strprintf("Sending %s to %s", FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); diff --git a/uint256.h b/uint256.h index 9a0c7704..6976d9dc 100644 --- a/uint256.h +++ b/uint256.h @@ -299,19 +299,18 @@ public: return string(psz, psz + sizeof(pn)*2); } - void SetHex(const std::string& str) + void SetHex(const char* psz) { for (int i = 0; i < WIDTH; i++) pn[i] = 0; - // skip 0x - const char* psz = str.c_str(); + // skip leading spaces while (isspace(*psz)) psz++; + + // skip 0x if (psz[0] == '0' && tolower(psz[1]) == 'x') psz += 2; - while (isspace(*psz)) - psz++; // hex string to uint static char phexdigit[256] = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,1,2,3,4,5,6,7,8,9,0,0,0,0,0,0, 0,0xa,0xb,0xc,0xd,0xe,0xf,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0xa,0xb,0xc,0xd,0xe,0xf,0,0,0,0,0,0,0,0,0 }; @@ -332,6 +331,11 @@ public: } } + void SetHex(const std::string& str) + { + SetHex(str.c_str()); + } + std::string ToString() const { return (GetHex()); From 64a474a49b20a66453fbcd2382e4b042427b0a0f Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 14 Feb 2010 22:27:13 +0000 Subject: [PATCH 051/133] renamed a few rpc methods git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@63 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- makefile.vc | 2 +- net.cpp | 4 ---- rpc.cpp | 26 +++++++++++++------------- serialize.h | 2 +- util.h | 1 + 5 files changed, 16 insertions(+), 19 deletions(-) diff --git a/makefile.vc b/makefile.vc index 73fd1cfd..527f2467 100644 --- a/makefile.vc +++ b/makefile.vc @@ -22,7 +22,7 @@ LIBS= \ wxmsw28$(D)_richtext.lib wxmsw28$(D)_html.lib wxmsw28$(D)_core.lib wxmsw28$(D)_adv.lib wxbase28$(D).lib wxtiff$(D).lib wxjpeg$(D).lib wxpng$(D).lib wxzlib$(D).lib wxregex$(D).lib wxexpat$(D).lib \ kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib shlwapi.lib WXDEFS=/DWIN32 /D__WXMSW__ /D_WINDOWS /DNOPCH -CFLAGS=/c /nologo /Ob0 /MD$(D) /EHsc /GR /Zm300 /YX /Fpobj/headers.pch $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +CFLAGS=/c /nologo /Ob0 /MD$(D) /EHsc /GR /Zm300 $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h diff --git a/net.cpp b/net.cpp index d669b8c5..061f000a 100644 --- a/net.cpp +++ b/net.cpp @@ -655,11 +655,7 @@ void ThreadSocketHandler2(void* parg) if (FD_ISSET(hListenSocket, &fdsetRecv)) { struct sockaddr_in sockaddr; -#ifdef __WXMSW__ - int len = sizeof(sockaddr); -#else socklen_t len = sizeof(sockaddr); -#endif SOCKET hSocket = accept(hListenSocket, (struct sockaddr*)&sockaddr, &len); CAddress addr(sockaddr); if (hSocket == INVALID_SOCKET) diff --git a/rpc.cpp b/rpc.cpp index b176b158..f4eef37f 100644 --- a/rpc.cpp +++ b/rpc.cpp @@ -157,12 +157,12 @@ Value listtransactions(const Array& params) } -Value getamountpaid(const Array& params) +Value getamountreceived(const Array& params) { if (params.size() < 1 || params.size() > 2) throw runtime_error( - "getamountpaid [minconf=1]\n" - "Returns the total amount paid to in transactions with at least [minconf] confirmations."); + "getamountreceived [minconf=1]\n" + "Returns the total amount received by in transactions with at least [minconf] confirmations."); // Bitcoin address string strAddress = params[0].get_str(); @@ -207,15 +207,15 @@ struct tallyitem } }; -Value getallpayments(const Array& params) +Value getallreceived(const Array& params) { if (params.size() > 1) throw runtime_error( - "getallpayments [minconf=1]\n" + "getallreceived [minconf=1]\n" "[minconf] is the minimum number of confirmations before payments are included.\n" "Returns an array of objects containing:\n" " \"address\" : bitcoin address\n" - " \"amount\" : total amount paid to the address\n" + " \"amount\" : total amount received by the address\n" " \"conf\" : number of confirmations\n" " \"label\" : the label set for this address when it was created by getnewaddress"); @@ -294,8 +294,8 @@ pair pCallTable[] = make_pair("getnewaddress", &getnewaddress), make_pair("sendtoaddress", &sendtoaddress), make_pair("listtransactions", &listtransactions), - make_pair("getamountpaid", &getamountpaid), - make_pair("getallpayments", &getallpayments), + make_pair("getamountreceived", &getamountreceived), + make_pair("getallreceived", &getallreceived), }; map mapCallTable(pCallTable, pCallTable + sizeof(pCallTable)/sizeof(pCallTable[0])); @@ -571,11 +571,11 @@ int CommandLineRPC(int argc, char *argv[]) // Special case other types int n = params.size(); - if (strMethod == "sendtoaddress" && n > 1) ConvertTo(params[1]); - if (strMethod == "listtransactions" && n > 0) ConvertTo(params[0]); - if (strMethod == "listtransactions" && n > 1) ConvertTo(params[1]); - if (strMethod == "getamountpaid" && n > 1) ConvertTo(params[1]); - if (strMethod == "getallpayments" && n > 0) ConvertTo(params[0]); + if (strMethod == "sendtoaddress" && n > 1) ConvertTo(params[1]); + if (strMethod == "listtransactions" && n > 0) ConvertTo(params[0]); + if (strMethod == "listtransactions" && n > 1) ConvertTo(params[1]); + if (strMethod == "getamountreceived" && n > 1) ConvertTo(params[1]); + if (strMethod == "getallreceived" && n > 0) ConvertTo(params[0]); // Execute Value result = CallRPC(strMethod, params); diff --git a/serialize.h b/serialize.h index eb090ae2..f7377314 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 201; -static const char* pszSubVer = ".0"; +static const char* pszSubVer = ".1"; diff --git a/util.h b/util.h index 6a7fabc5..933c71be 100644 --- a/util.h +++ b/util.h @@ -68,6 +68,7 @@ static const bool fWindows = true; #define S_IWUSR 0200 #endif #define unlink _unlink +typedef int socklen_t; #else static const bool fWindows = false; #define WSAGetLastError() errno From 5253d1ab77fab1995ede03fb934edd67f1359ba8 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Mon, 15 Feb 2010 04:03:07 +0000 Subject: [PATCH 052/133] strip out unfinished product, review and market stuff, enable _() instead of wxT() in uiproject.fbp so it uses wxGetTranslation for the wxFormBuilder generated part of the UI git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@64 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.cpp | 21 - db.h | 39 - headers.h | 1 - main.cpp | 24 +- makefile | 7 +- makefile.unix.wx2.8 | 7 +- makefile.unix.wx2.9 | 7 +- makefile.vc | 7 +- market.cpp | 264 -- market.h | 182 -- net.cpp | 9 - net.h | 6 - serialize.h | 2 +- ui.cpp | 930 +------ ui.h | 142 +- uibase.cpp | 851 +------ uibase.h | 287 +-- uiproject.fbp | 5805 +------------------------------------------ 18 files changed, 127 insertions(+), 8464 deletions(-) delete mode 100644 market.cpp delete mode 100644 market.h diff --git a/db.cpp b/db.cpp index ff9ece52..bbf24a0e 100644 --- a/db.cpp +++ b/db.cpp @@ -507,27 +507,6 @@ bool LoadAddresses() -// -// CReviewDB -// - -bool CReviewDB::ReadReviews(uint256 hash, vector& vReviews) -{ - vReviews.size(); // msvc workaround, just need to do anything with vReviews - return Read(make_pair(string("reviews"), hash), vReviews); -} - -bool CReviewDB::WriteReviews(uint256 hash, const vector& vReviews) -{ - return Write(make_pair(string("reviews"), hash), vReviews); -} - - - - - - - // // CWalletDB // diff --git a/db.h b/db.h index 538076b5..fb4e9268 100644 --- a/db.h +++ b/db.h @@ -285,45 +285,6 @@ public: -class CReviewDB : public CDB -{ -public: - CReviewDB(const char* pszMode="r+") : CDB("reviews.dat", pszMode) { } -private: - CReviewDB(const CReviewDB&); - void operator=(const CReviewDB&); -public: - bool ReadUser(uint256 hash, CUser& user) - { - return Read(make_pair(string("user"), hash), user); - } - - bool WriteUser(uint256 hash, const CUser& user) - { - return Write(make_pair(string("user"), hash), user); - } - - bool ReadReviews(uint256 hash, vector& vReviews); - bool WriteReviews(uint256 hash, const vector& vReviews); -}; - - - - - -class CMarketDB : public CDB -{ -public: - CMarketDB(const char* pszMode="r+") : CDB("market.dat", pszMode) { } -private: - CMarketDB(const CMarketDB&); - void operator=(const CMarketDB&); -}; - - - - - class CAddrDB : public CDB { public: diff --git a/headers.h b/headers.h index dda4f9ca..7fb227ec 100644 --- a/headers.h +++ b/headers.h @@ -99,7 +99,6 @@ using namespace boost; #include "net.h" #include "irc.h" #include "main.h" -#include "market.h" #include "rpc.h" #include "uibase.h" #include "ui.h" diff --git a/main.cpp b/main.cpp index 0c7aff84..3391a299 100644 --- a/main.cpp +++ b/main.cpp @@ -1706,10 +1706,8 @@ bool AlreadyHave(CTxDB& txdb, const CInv& inv) { switch (inv.type) { - case MSG_TX: return mapTransactions.count(inv.hash) || txdb.ContainsTx(inv.hash); - case MSG_BLOCK: return mapBlockIndex.count(inv.hash) || mapOrphanBlocks.count(inv.hash); - case MSG_REVIEW: return true; - case MSG_PRODUCT: return mapProducts.count(inv.hash); + case MSG_TX: return mapTransactions.count(inv.hash) || txdb.ContainsTx(inv.hash); + case MSG_BLOCK: return mapBlockIndex.count(inv.hash) || mapOrphanBlocks.count(inv.hash); } // Don't know what it is, just say we already got one return true; @@ -2108,24 +2106,6 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) } - else if (strCommand == "review") - { - CDataStream vMsg(vRecv); - CReview review; - vRecv >> review; - - CInv inv(MSG_REVIEW, review.GetHash()); - pfrom->AddInventoryKnown(inv); - - if (review.AcceptReview()) - { - // Relay the original message as-is in case it's a higher version than we know how to parse - RelayMessage(inv, vMsg); - mapAlreadyAskedFor.erase(inv); - } - } - - else if (strCommand == "block") { auto_ptr pblock(new CBlock); diff --git a/makefile b/makefile index affbe517..fb8dd274 100644 --- a/makefile +++ b/makefile @@ -25,7 +25,7 @@ LIBS= \ -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 -l shlwapi WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h @@ -50,9 +50,6 @@ obj/net.o: net.cpp $(HEADERS) obj/main.o: main.cpp $(HEADERS) sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - obj/ui.o: ui.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< @@ -73,7 +70,7 @@ obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp -OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ +OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o \ obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o \ obj/ui_res.o diff --git a/makefile.unix.wx2.8 b/makefile.unix.wx2.8 index d3358ab2..38d9bc0e 100644 --- a/makefile.unix.wx2.8 +++ b/makefile.unix.wx2.8 @@ -35,7 +35,7 @@ LIBS= \ WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h @@ -60,9 +60,6 @@ obj/net.o: net.cpp $(HEADERS) obj/main.o: main.cpp $(HEADERS) sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - obj/ui.o: ui.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< @@ -80,7 +77,7 @@ obj/rpc.o: rpc.cpp $(HEADERS) -OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ +OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o \ obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o bitcoin: headers.h.gch $(OBJS) diff --git a/makefile.unix.wx2.9 b/makefile.unix.wx2.9 index 8defcbb6..0b3b09b5 100644 --- a/makefile.unix.wx2.9 +++ b/makefile.unix.wx2.9 @@ -35,7 +35,7 @@ LIBS= \ WXDEFS=-D__WXGTK__ -DNOPCH CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h @@ -60,9 +60,6 @@ obj/net.o: net.cpp $(HEADERS) obj/main.o: main.cpp $(HEADERS) sha.h g++ -c $(CFLAGS) -o $@ $< -obj/market.o: market.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - obj/ui.o: ui.cpp $(HEADERS) g++ -c $(CFLAGS) -o $@ $< @@ -80,7 +77,7 @@ obj/rpc.o: rpc.cpp $(HEADERS) -OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o obj/market.o \ +OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o \ obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o bitcoin: headers.h.gch $(OBJS) diff --git a/makefile.vc b/makefile.vc index 527f2467..5ca8de46 100644 --- a/makefile.vc +++ b/makefile.vc @@ -23,7 +23,7 @@ LIBS= \ kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib shlwapi.lib WXDEFS=/DWIN32 /D__WXMSW__ /D_WINDOWS /DNOPCH CFLAGS=/c /nologo /Ob0 /MD$(D) /EHsc /GR /Zm300 $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h market.h rpc.h uibase.h ui.h +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h @@ -45,9 +45,6 @@ obj\net.obj: net.cpp $(HEADERS) obj\main.obj: main.cpp $(HEADERS) sha.h cl $(CFLAGS) /Fo$@ %s -obj\market.obj: market.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s - obj\ui.obj: ui.cpp $(HEADERS) cl $(CFLAGS) /Fo$@ %s @@ -68,7 +65,7 @@ obj\ui.res: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp r -OBJS=obj\util.obj obj\script.obj obj\db.obj obj\net.obj obj\main.obj obj\market.obj \ +OBJS=obj\util.obj obj\script.obj obj\db.obj obj\net.obj obj\main.obj \ obj\ui.obj obj\uibase.obj obj\sha.obj obj\irc.obj obj\rpc.obj \ obj\ui.res diff --git a/market.cpp b/market.cpp deleted file mode 100644 index 22b5365f..00000000 --- a/market.cpp +++ /dev/null @@ -1,264 +0,0 @@ -// Copyright (c) 2009 Satoshi Nakamoto -// Distributed under the MIT/X11 software license, see the accompanying -// file license.txt or http://www.opensource.org/licenses/mit-license.php. - -#include "headers.h" - - - - - - - - - - -// -// Global state variables -// - -//// later figure out how these are persisted -map mapMyProducts; - - - - -map mapProducts; -CCriticalSection cs_mapProducts; - -bool AdvertInsert(const CProduct& product) -{ - uint256 hash = product.GetHash(); - bool fNew = false; - bool fUpdated = false; - - CRITICAL_BLOCK(cs_mapProducts) - { - // Insert or find existing product - pair::iterator, bool> item = mapProducts.insert(make_pair(hash, product)); - CProduct* pproduct = &(*(item.first)).second; - fNew = item.second; - - // Update if newer - if (product.nSequence > pproduct->nSequence) - { - *pproduct = product; - fUpdated = true; - } - } - - //if (fNew) - // NotifyProductAdded(hash); - //else if (fUpdated) - // NotifyProductUpdated(hash); - - return (fNew || fUpdated); -} - -void AdvertErase(const CProduct& product) -{ - uint256 hash = product.GetHash(); - CRITICAL_BLOCK(cs_mapProducts) - mapProducts.erase(hash); - //NotifyProductDeleted(hash); -} - - - - - - - - - - - - - - - - - - - -template -unsigned int Union(T& v1, T& v2) -{ - // v1 = v1 union v2 - // v1 and v2 must be sorted - // returns the number of elements added to v1 - - ///// need to check that this is equivalent, then delete this comment - //vector vUnion(v1.size() + v2.size()); - //vUnion.erase(set_union(v1.begin(), v1.end(), - // v2.begin(), v2.end(), - // vUnion.begin()), - // vUnion.end()); - - T vUnion; - vUnion.reserve(v1.size() + v2.size()); - set_union(v1.begin(), v1.end(), - v2.begin(), v2.end(), - back_inserter(vUnion)); - unsigned int nAdded = vUnion.size() - v1.size(); - if (nAdded > 0) - v1 = vUnion; - return nAdded; -} - -void CUser::AddAtom(unsigned short nAtom, bool fOrigin) -{ - // Ignore duplicates - if (binary_search(vAtomsIn.begin(), vAtomsIn.end(), nAtom) || - find(vAtomsNew.begin(), vAtomsNew.end(), nAtom) != vAtomsNew.end()) - return; - - //// instead of zero atom, should change to free atom that propagates, - //// limited to lower than a certain value like 5 so conflicts quickly - // The zero atom never propagates, - // new atoms always propagate through the user that created them - if (nAtom == 0 || fOrigin) - { - vector vTmp(1, nAtom); - Union(vAtomsIn, vTmp); - if (fOrigin) - vAtomsOut.push_back(nAtom); - return; - } - - vAtomsNew.push_back(nAtom); - - if (vAtomsNew.size() >= nFlowthroughRate || vAtomsOut.empty()) - { - // Select atom to flow through to vAtomsOut - vAtomsOut.push_back(vAtomsNew[GetRand(vAtomsNew.size())]); - - // Merge vAtomsNew into vAtomsIn - sort(vAtomsNew.begin(), vAtomsNew.end()); - Union(vAtomsIn, vAtomsNew); - vAtomsNew.clear(); - } -} - -bool AddAtomsAndPropagate(uint256 hashUserStart, const vector& vAtoms, bool fOrigin) -{ - CReviewDB reviewdb; - map > pmapPropagate[2]; - pmapPropagate[0][hashUserStart] = vAtoms; - - for (int side = 0; !pmapPropagate[side].empty(); side = 1 - side) - { - map >& mapFrom = pmapPropagate[side]; - map >& mapTo = pmapPropagate[1 - side]; - - for (map >::iterator mi = mapFrom.begin(); mi != mapFrom.end(); ++mi) - { - const uint256& hashUser = (*mi).first; - const vector& vReceived = (*mi).second; - - ///// this would be a lot easier on the database if it put the new atom at the beginning of the list, - ///// so the change would be right next to the vector size. - - // Read user - CUser user; - reviewdb.ReadUser(hashUser, user); - unsigned int nIn = user.vAtomsIn.size(); - unsigned int nNew = user.vAtomsNew.size(); - unsigned int nOut = user.vAtomsOut.size(); - - // Add atoms received - foreach(unsigned short nAtom, vReceived) - user.AddAtom(nAtom, fOrigin); - fOrigin = false; - - // Don't bother writing to disk if no changes - if (user.vAtomsIn.size() == nIn && user.vAtomsNew.size() == nNew) - continue; - - // Propagate - if (user.vAtomsOut.size() > nOut) - foreach(const uint256& hash, user.vLinksOut) - mapTo[hash].insert(mapTo[hash].end(), user.vAtomsOut.begin() + nOut, user.vAtomsOut.end()); - - // Write back - if (!reviewdb.WriteUser(hashUser, user)) - return false; - } - mapFrom.clear(); - } - return true; -} - - - - - - -bool CReview::AcceptReview() -{ - // Timestamp - nTime = GetTime(); - - // Check signature - if (!CKey::Verify(vchPubKeyFrom, GetSigHash(), vchSig)) - return false; - - CReviewDB reviewdb; - - // Add review text to recipient - vector vReviews; - reviewdb.ReadReviews(hashTo, vReviews); - vReviews.push_back(*this); - if (!reviewdb.WriteReviews(hashTo, vReviews)) - return false; - - // Add link from sender - CUser user; - uint256 hashFrom = Hash(vchPubKeyFrom.begin(), vchPubKeyFrom.end()); - reviewdb.ReadUser(hashFrom, user); - user.vLinksOut.push_back(hashTo); - if (!reviewdb.WriteUser(hashFrom, user)) - return false; - - reviewdb.Close(); - - // Propagate atoms to recipient - vector vZeroAtom(1, 0); - if (!AddAtomsAndPropagate(hashTo, user.vAtomsOut.size() ? user.vAtomsOut : vZeroAtom, false)) - return false; - - return true; -} - - - - - -bool CProduct::CheckSignature() -{ - return (CKey::Verify(vchPubKeyFrom, GetSigHash(), vchSig)); -} - -bool CProduct::CheckProduct() -{ - if (!CheckSignature()) - return false; - - // Make sure it's a summary product - if (!mapDetails.empty() || !vOrderForm.empty()) - return false; - - // Look up seller's atom count - CReviewDB reviewdb("r"); - CUser user; - reviewdb.ReadUser(GetUserHash(), user); - nAtoms = user.GetAtomCount(); - reviewdb.Close(); - - ////// delme, this is now done by AdvertInsert - //// Store to memory - //CRITICAL_BLOCK(cs_mapProducts) - // mapProducts[GetHash()] = *this; - - return true; -} diff --git a/market.h b/market.h deleted file mode 100644 index 27147873..00000000 --- a/market.h +++ /dev/null @@ -1,182 +0,0 @@ -// Copyright (c) 2009 Satoshi Nakamoto -// Distributed under the MIT/X11 software license, see the accompanying -// file license.txt or http://www.opensource.org/licenses/mit-license.php. - -class CUser; -class CReview; -class CProduct; - -static const unsigned int nFlowthroughRate = 2; - - - - -bool AdvertInsert(const CProduct& product); -void AdvertErase(const CProduct& product); -bool AddAtomsAndPropagate(uint256 hashUserStart, const vector& vAtoms, bool fOrigin); - - - - - - - - -class CUser -{ -public: - vector vAtomsIn; - vector vAtomsNew; - vector vAtomsOut; - vector vLinksOut; - - CUser() - { - } - - IMPLEMENT_SERIALIZE - ( - if (!(nType & SER_GETHASH)) - READWRITE(nVersion); - READWRITE(vAtomsIn); - READWRITE(vAtomsNew); - READWRITE(vAtomsOut); - READWRITE(vLinksOut); - ) - - void SetNull() - { - vAtomsIn.clear(); - vAtomsNew.clear(); - vAtomsOut.clear(); - vLinksOut.clear(); - } - - uint256 GetHash() const { return SerializeHash(*this); } - - - int GetAtomCount() const - { - return (vAtomsIn.size() + vAtomsNew.size()); - } - - void AddAtom(unsigned short nAtom, bool fOrigin); -}; - - - - - - - -class CReview -{ -public: - int nVersion; - uint256 hashTo; - map mapValue; - vector vchPubKeyFrom; - vector vchSig; - - // memory only - unsigned int nTime; - int nAtoms; - - - CReview() - { - nVersion = 1; - hashTo = 0; - nTime = 0; - nAtoms = 0; - } - - IMPLEMENT_SERIALIZE - ( - READWRITE(this->nVersion); - nVersion = this->nVersion; - if (!(nType & SER_DISK)) - READWRITE(hashTo); - READWRITE(mapValue); - READWRITE(vchPubKeyFrom); - if (!(nType & SER_GETHASH)) - READWRITE(vchSig); - ) - - uint256 GetHash() const { return SerializeHash(*this); } - uint256 GetSigHash() const { return SerializeHash(*this, SER_GETHASH|SER_SKIPSIG); } - uint256 GetUserHash() const { return Hash(vchPubKeyFrom.begin(), vchPubKeyFrom.end()); } - - - bool AcceptReview(); -}; - - - - - - - -class CProduct -{ -public: - int nVersion; - CAddress addr; - map mapValue; - map mapDetails; - vector > vOrderForm; - unsigned int nSequence; - vector vchPubKeyFrom; - vector vchSig; - - // disk only - int nAtoms; - - // memory only - set setSources; - - CProduct() - { - nVersion = 1; - nAtoms = 0; - nSequence = 0; - } - - IMPLEMENT_SERIALIZE - ( - READWRITE(this->nVersion); - nVersion = this->nVersion; - READWRITE(addr); - READWRITE(mapValue); - if (!(nType & SER_GETHASH)) - { - READWRITE(mapDetails); - READWRITE(vOrderForm); - READWRITE(nSequence); - } - READWRITE(vchPubKeyFrom); - if (!(nType & SER_GETHASH)) - READWRITE(vchSig); - if (nType & SER_DISK) - READWRITE(nAtoms); - ) - - uint256 GetHash() const { return SerializeHash(*this); } - uint256 GetSigHash() const { return SerializeHash(*this, SER_GETHASH|SER_SKIPSIG); } - uint256 GetUserHash() const { return Hash(vchPubKeyFrom.begin(), vchPubKeyFrom.end()); } - - - bool CheckSignature(); - bool CheckProduct(); -}; - - - - - - - - -extern map mapProducts; -extern CCriticalSection cs_mapProducts; -extern map mapMyProducts; diff --git a/net.cpp b/net.cpp index 061f000a..8e8127d7 100644 --- a/net.cpp +++ b/net.cpp @@ -381,11 +381,6 @@ void CNode::CancelSubscribe(unsigned int nChannel) foreach(CNode* pnode, vNodes) if (pnode != this) pnode->PushMessage("sub-cancel", nChannel); - - // Clear memory, no longer subscribed - if (nChannel == MSG_PRODUCT) - CRITICAL_BLOCK(cs_mapProducts) - mapProducts.clear(); } } @@ -497,10 +492,6 @@ void CNode::Cleanup() // All of a nodes broadcasts and subscriptions are automatically torn down // when it goes down, so a node has to stay up to keep its broadcast going. - CRITICAL_BLOCK(cs_mapProducts) - for (map::iterator mi = mapProducts.begin(); mi != mapProducts.end();) - AdvertRemoveSource(this, MSG_PRODUCT, 0, (*(mi++)).second); - // Cancel subscriptions for (unsigned int nChannel = 0; nChannel < vfSubscribe.size(); nChannel++) if (vfSubscribe[nChannel]) diff --git a/net.h b/net.h index ce6f772f..c7d15460 100644 --- a/net.h +++ b/net.h @@ -341,9 +341,6 @@ enum { MSG_TX = 1, MSG_BLOCK, - MSG_REVIEW, - MSG_PRODUCT, - MSG_TABLE, }; static const char* ppszTypeName[] = @@ -351,9 +348,6 @@ static const char* ppszTypeName[] = "ERROR", "tx", "block", - "review", - "product", - "table", }; class CInv diff --git a/serialize.h b/serialize.h index f7377314..12b58ad1 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 201; -static const char* pszSubVer = ".1"; +static const char* pszSubVer = ".2"; diff --git a/ui.cpp b/ui.cpp index aad599c7..f506fcf3 100644 --- a/ui.cpp +++ b/ui.cpp @@ -7,22 +7,15 @@ #include #endif -void ThreadRequestProductDetails(void* parg); -void ThreadRandSendTest(void* parg); bool GetStartOnSystemStartup(); void SetStartOnSystemStartup(bool fAutoStart); DEFINE_EVENT_TYPE(wxEVT_UITHREADCALL) -DEFINE_EVENT_TYPE(wxEVT_REPLY1) -DEFINE_EVENT_TYPE(wxEVT_REPLY2) -DEFINE_EVENT_TYPE(wxEVT_REPLY3) CMainFrame* pframeMain = NULL; CMyTaskBarIcon* ptaskbaricon = NULL; -bool fRandSendTest = false; -void RandSend(); extern int g_isPainting; bool fClosedToTray = false; @@ -210,93 +203,6 @@ int ThreadSafeMessageBox(const string& message, const string& caption, int style -////////////////////////////////////////////////////////////////////////////// -// -// Custom events -// -// If this code gets used again, it should be replaced with something like UIThreadCall - -set setCallbackAvailable; -CCriticalSection cs_setCallbackAvailable; - -void AddCallbackAvailable(void* p) -{ - CRITICAL_BLOCK(cs_setCallbackAvailable) - setCallbackAvailable.insert(p); -} - -void RemoveCallbackAvailable(void* p) -{ - CRITICAL_BLOCK(cs_setCallbackAvailable) - setCallbackAvailable.erase(p); -} - -bool IsCallbackAvailable(void* p) -{ - CRITICAL_BLOCK(cs_setCallbackAvailable) - return setCallbackAvailable.count(p); - return false; -} - -template -void AddPendingCustomEvent(wxEvtHandler* pevthandler, int nEventID, const T pbeginIn, const T pendIn) -{ - // Need to rewrite with something like UIThreadCall - // I'm tired of maintaining this hack that's only called by unfinished unused code. - assert(("Unimplemented", 0)); - //if (!pevthandler) - // return; - // - //const char* pbegin = (pendIn != pbeginIn) ? &pbeginIn[0] : NULL; - //const char* pend = pbegin + (pendIn - pbeginIn) * sizeof(pbeginIn[0]); - //wxCommandEvent event(nEventID); - //wxString strData(wxChar(0), (pend - pbegin) / sizeof(wxChar) + 1); - //memcpy(&strData[0], pbegin, pend - pbegin); - //event.SetString(strData); - //event.SetInt(pend - pbegin); - // - //pevthandler->AddPendingEvent(event); -} - -template -void AddPendingCustomEvent(wxEvtHandler* pevthandler, int nEventID, const T& obj) -{ - CDataStream ss; - ss << obj; - AddPendingCustomEvent(pevthandler, nEventID, ss.begin(), ss.end()); -} - -void AddPendingReplyEvent1(void* pevthandler, CDataStream& vRecv) -{ - if (IsCallbackAvailable(pevthandler)) - AddPendingCustomEvent((wxEvtHandler*)pevthandler, wxEVT_REPLY1, vRecv.begin(), vRecv.end()); -} - -void AddPendingReplyEvent2(void* pevthandler, CDataStream& vRecv) -{ - if (IsCallbackAvailable(pevthandler)) - AddPendingCustomEvent((wxEvtHandler*)pevthandler, wxEVT_REPLY2, vRecv.begin(), vRecv.end()); -} - -void AddPendingReplyEvent3(void* pevthandler, CDataStream& vRecv) -{ - if (IsCallbackAvailable(pevthandler)) - AddPendingCustomEvent((wxEvtHandler*)pevthandler, wxEVT_REPLY3, vRecv.begin(), vRecv.end()); -} - -CDataStream GetStreamFromEvent(const wxCommandEvent& event) -{ - wxString strData = event.GetString(); - const char* pszBegin = strData.c_str(); - return CDataStream(pszBegin, pszBegin + event.GetInt(), SER_NETWORK); -} - - - - - - - ////////////////////////////////////////////////////////////////////////////// // // CMainFrame @@ -1180,16 +1086,6 @@ void CMainFrame::OnMouseEventsAddress(wxMouseEvent& event) event.Skip(); } -void CMainFrame::OnButtonCopy(wxCommandEvent& event) -{ - // Copy address box to clipboard - if (wxTheClipboard->Open()) - { - wxTheClipboard->SetData(new wxTextDataObject(m_textCtrlAddress->GetValue())); - wxTheClipboard->Close(); - } -} - void CMainFrame::OnButtonChange(wxCommandEvent& event) { CYourAddressDialog dialog(this, string(m_textCtrlAddress->GetValue())); @@ -1208,6 +1104,16 @@ void CMainFrame::OnButtonChange(wxCommandEvent& event) } } +void CMainFrame::OnButtonCopy(wxCommandEvent& event) +{ + // Copy address box to clipboard + if (wxTheClipboard->Open()) + { + wxTheClipboard->SetData(new wxTextDataObject(m_textCtrlAddress->GetValue())); + wxTheClipboard->Close(); + } +} + void CMainFrame::OnListItemActivated(wxListEvent& event) { uint256 hash((string)GetItemText(m_listCtrl, event.GetIndex(), 1)); @@ -1228,28 +1134,6 @@ void CMainFrame::OnListItemActivated(wxListEvent& event) //pdialog->Show(); } -void CMainFrame::OnListItemActivatedProductsSent(wxListEvent& event) -{ - CProduct& product = *(CProduct*)event.GetItem().GetData(); - CEditProductDialog* pdialog = new CEditProductDialog(this); - pdialog->SetProduct(product); - pdialog->Show(); -} - -void CMainFrame::OnListItemActivatedOrdersSent(wxListEvent& event) -{ - CWalletTx& order = *(CWalletTx*)event.GetItem().GetData(); - CViewOrderDialog* pdialog = new CViewOrderDialog(this, order, false); - pdialog->Show(); -} - -void CMainFrame::OnListItemActivatedOrdersReceived(wxListEvent& event) -{ - CWalletTx& order = *(CWalletTx*)event.GetItem().GetData(); - CViewOrderDialog* pdialog = new CViewOrderDialog(this, order, true); - pdialog->Show(); -} - @@ -2525,800 +2409,6 @@ void CAddressBookDialog::OnClose(wxCloseEvent& event) -////////////////////////////////////////////////////////////////////////////// -// -// CProductsDialog -// - -bool CompareIntStringPairBestFirst(const pair& item1, const pair& item2) -{ - return (item1.first > item2.first); -} - -CProductsDialog::CProductsDialog(wxWindow* parent) : CProductsDialogBase(parent) -{ - // Init column headers - m_listCtrl->InsertColumn(0, "Title", wxLIST_FORMAT_LEFT, 200); - m_listCtrl->InsertColumn(1, "Price", wxLIST_FORMAT_LEFT, 80); - m_listCtrl->InsertColumn(2, "Seller", wxLIST_FORMAT_LEFT, 80); - m_listCtrl->InsertColumn(3, "Stars", wxLIST_FORMAT_LEFT, 50); - m_listCtrl->InsertColumn(4, "Power", wxLIST_FORMAT_LEFT, 50); - - // Tally top categories - map mapTopCategories; - CRITICAL_BLOCK(cs_mapProducts) - for (map::iterator mi = mapProducts.begin(); mi != mapProducts.end(); ++mi) - mapTopCategories[(*mi).second.mapValue["category"]]++; - - // Sort top categories - vector > vTopCategories; - for (map::iterator mi = mapTopCategories.begin(); mi != mapTopCategories.end(); ++mi) - vTopCategories.push_back(make_pair((*mi).second, (*mi).first)); - sort(vTopCategories.begin(), vTopCategories.end(), CompareIntStringPairBestFirst); - - // Fill categories combo box - int nLimit = 250; - for (vector >::iterator it = vTopCategories.begin(); it != vTopCategories.end() && nLimit-- > 0; ++it) - m_comboBoxCategory->Append((*it).second); - - // Fill window with initial search - //wxCommandEvent event; - //OnButtonSearch(event); -} - -void CProductsDialog::OnCombobox(wxCommandEvent& event) -{ - OnButtonSearch(event); -} - -bool CompareProductsBestFirst(const CProduct* p1, const CProduct* p2) -{ - return (p1->nAtoms > p2->nAtoms); -} - -void CProductsDialog::OnButtonSearch(wxCommandEvent& event) -{ - string strCategory = (string)m_comboBoxCategory->GetValue(); - string strSearch = (string)m_textCtrlSearch->GetValue(); - - // Search products - vector vProductsFound; - CRITICAL_BLOCK(cs_mapProducts) - { - for (map::iterator mi = mapProducts.begin(); mi != mapProducts.end(); ++mi) - { - CProduct& product = (*mi).second; - if (product.mapValue["category"].find(strCategory) != -1) - { - if (product.mapValue["title"].find(strSearch) != -1 || - product.mapValue["description"].find(strSearch) != -1 || - product.mapValue["seller"].find(strSearch) != -1) - { - vProductsFound.push_back(&product); - } - } - } - } - - // Sort - sort(vProductsFound.begin(), vProductsFound.end(), CompareProductsBestFirst); - - // Display - foreach(CProduct* pproduct, vProductsFound) - { - InsertLine(m_listCtrl, - pproduct->mapValue["title"], - pproduct->mapValue["price"], - pproduct->mapValue["seller"], - pproduct->mapValue["stars"], - itostr(pproduct->nAtoms)); - } -} - -void CProductsDialog::OnListItemActivated(wxListEvent& event) -{ - // Doubleclick opens product - CViewProductDialog* pdialog = new CViewProductDialog(this, m_vProduct[event.GetIndex()]); - pdialog->Show(); -} - - - - - - - -////////////////////////////////////////////////////////////////////////////// -// -// CEditProductDialog -// - -CEditProductDialog::CEditProductDialog(wxWindow* parent) : CEditProductDialogBase(parent) -{ - m_textCtrlLabel[0 ] = m_textCtrlLabel0; - m_textCtrlLabel[1 ] = m_textCtrlLabel1; - m_textCtrlLabel[2 ] = m_textCtrlLabel2; - m_textCtrlLabel[3 ] = m_textCtrlLabel3; - m_textCtrlLabel[4 ] = m_textCtrlLabel4; - m_textCtrlLabel[5 ] = m_textCtrlLabel5; - m_textCtrlLabel[6 ] = m_textCtrlLabel6; - m_textCtrlLabel[7 ] = m_textCtrlLabel7; - m_textCtrlLabel[8 ] = m_textCtrlLabel8; - m_textCtrlLabel[9 ] = m_textCtrlLabel9; - m_textCtrlLabel[10] = m_textCtrlLabel10; - m_textCtrlLabel[11] = m_textCtrlLabel11; - m_textCtrlLabel[12] = m_textCtrlLabel12; - m_textCtrlLabel[13] = m_textCtrlLabel13; - m_textCtrlLabel[14] = m_textCtrlLabel14; - m_textCtrlLabel[15] = m_textCtrlLabel15; - m_textCtrlLabel[16] = m_textCtrlLabel16; - m_textCtrlLabel[17] = m_textCtrlLabel17; - m_textCtrlLabel[18] = m_textCtrlLabel18; - m_textCtrlLabel[19] = m_textCtrlLabel19; - - m_textCtrlField[0 ] = m_textCtrlField0; - m_textCtrlField[1 ] = m_textCtrlField1; - m_textCtrlField[2 ] = m_textCtrlField2; - m_textCtrlField[3 ] = m_textCtrlField3; - m_textCtrlField[4 ] = m_textCtrlField4; - m_textCtrlField[5 ] = m_textCtrlField5; - m_textCtrlField[6 ] = m_textCtrlField6; - m_textCtrlField[7 ] = m_textCtrlField7; - m_textCtrlField[8 ] = m_textCtrlField8; - m_textCtrlField[9 ] = m_textCtrlField9; - m_textCtrlField[10] = m_textCtrlField10; - m_textCtrlField[11] = m_textCtrlField11; - m_textCtrlField[12] = m_textCtrlField12; - m_textCtrlField[13] = m_textCtrlField13; - m_textCtrlField[14] = m_textCtrlField14; - m_textCtrlField[15] = m_textCtrlField15; - m_textCtrlField[16] = m_textCtrlField16; - m_textCtrlField[17] = m_textCtrlField17; - m_textCtrlField[18] = m_textCtrlField18; - m_textCtrlField[19] = m_textCtrlField19; - - m_buttonDel[0 ] = m_buttonDel0; - m_buttonDel[1 ] = m_buttonDel1; - m_buttonDel[2 ] = m_buttonDel2; - m_buttonDel[3 ] = m_buttonDel3; - m_buttonDel[4 ] = m_buttonDel4; - m_buttonDel[5 ] = m_buttonDel5; - m_buttonDel[6 ] = m_buttonDel6; - m_buttonDel[7 ] = m_buttonDel7; - m_buttonDel[8 ] = m_buttonDel8; - m_buttonDel[9 ] = m_buttonDel9; - m_buttonDel[10] = m_buttonDel10; - m_buttonDel[11] = m_buttonDel11; - m_buttonDel[12] = m_buttonDel12; - m_buttonDel[13] = m_buttonDel13; - m_buttonDel[14] = m_buttonDel14; - m_buttonDel[15] = m_buttonDel15; - m_buttonDel[16] = m_buttonDel16; - m_buttonDel[17] = m_buttonDel17; - m_buttonDel[18] = m_buttonDel18; - m_buttonDel[19] = m_buttonDel19; - - for (int i = 1; i < FIELDS_MAX; i++) - ShowLine(i, false); - - LayoutAll(); -} - -void CEditProductDialog::LayoutAll() -{ - m_scrolledWindow->Layout(); - m_scrolledWindow->GetSizer()->Fit(m_scrolledWindow); - this->Layout(); -} - -void CEditProductDialog::ShowLine(int i, bool fShow) -{ - m_textCtrlLabel[i]->Show(fShow); - m_textCtrlField[i]->Show(fShow); - m_buttonDel[i]->Show(fShow); -} - -void CEditProductDialog::OnButtonDel0(wxCommandEvent& event) { OnButtonDel(event, 0); } -void CEditProductDialog::OnButtonDel1(wxCommandEvent& event) { OnButtonDel(event, 1); } -void CEditProductDialog::OnButtonDel2(wxCommandEvent& event) { OnButtonDel(event, 2); } -void CEditProductDialog::OnButtonDel3(wxCommandEvent& event) { OnButtonDel(event, 3); } -void CEditProductDialog::OnButtonDel4(wxCommandEvent& event) { OnButtonDel(event, 4); } -void CEditProductDialog::OnButtonDel5(wxCommandEvent& event) { OnButtonDel(event, 5); } -void CEditProductDialog::OnButtonDel6(wxCommandEvent& event) { OnButtonDel(event, 6); } -void CEditProductDialog::OnButtonDel7(wxCommandEvent& event) { OnButtonDel(event, 7); } -void CEditProductDialog::OnButtonDel8(wxCommandEvent& event) { OnButtonDel(event, 8); } -void CEditProductDialog::OnButtonDel9(wxCommandEvent& event) { OnButtonDel(event, 9); } -void CEditProductDialog::OnButtonDel10(wxCommandEvent& event) { OnButtonDel(event, 10); } -void CEditProductDialog::OnButtonDel11(wxCommandEvent& event) { OnButtonDel(event, 11); } -void CEditProductDialog::OnButtonDel12(wxCommandEvent& event) { OnButtonDel(event, 12); } -void CEditProductDialog::OnButtonDel13(wxCommandEvent& event) { OnButtonDel(event, 13); } -void CEditProductDialog::OnButtonDel14(wxCommandEvent& event) { OnButtonDel(event, 14); } -void CEditProductDialog::OnButtonDel15(wxCommandEvent& event) { OnButtonDel(event, 15); } -void CEditProductDialog::OnButtonDel16(wxCommandEvent& event) { OnButtonDel(event, 16); } -void CEditProductDialog::OnButtonDel17(wxCommandEvent& event) { OnButtonDel(event, 17); } -void CEditProductDialog::OnButtonDel18(wxCommandEvent& event) { OnButtonDel(event, 18); } -void CEditProductDialog::OnButtonDel19(wxCommandEvent& event) { OnButtonDel(event, 19); } - -void CEditProductDialog::OnButtonDel(wxCommandEvent& event, int n) -{ - Freeze(); - int x, y; - m_scrolledWindow->GetViewStart(&x, &y); - int i; - for (i = n; i < FIELDS_MAX-1; i++) - { - m_textCtrlLabel[i]->SetValue(m_textCtrlLabel[i+1]->GetValue()); - m_textCtrlField[i]->SetValue(m_textCtrlField[i+1]->GetValue()); - if (!m_buttonDel[i+1]->IsShown()) - break; - } - m_textCtrlLabel[i]->SetValue(""); - m_textCtrlField[i]->SetValue(""); - ShowLine(i, false); - m_buttonAddField->Enable(true); - LayoutAll(); - m_scrolledWindow->Scroll(0, y); - Thaw(); -} - -void CEditProductDialog::OnButtonAddField(wxCommandEvent& event) -{ - for (int i = 0; i < FIELDS_MAX; i++) - { - if (!m_buttonDel[i]->IsShown()) - { - Freeze(); - ShowLine(i, true); - if (i == FIELDS_MAX-1) - m_buttonAddField->Enable(false); - LayoutAll(); - m_scrolledWindow->Scroll(0, 99999); - Thaw(); - break; - } - } -} - -void AddToMyProducts(CProduct product) -{ - CProduct& productInsert = mapMyProducts[product.GetHash()]; - productInsert = product; - //InsertLine(pframeMain->m_listCtrlProductsSent, &productInsert, - // product.mapValue["category"], - // product.mapValue["title"].substr(0, 100), - // product.mapValue["description"].substr(0, 100), - // product.mapValue["price"], - // ""); -} - -void CEditProductDialog::OnButtonSend(wxCommandEvent& event) -{ - CProduct product; - GetProduct(product); - - // Sign the detailed product - product.vchPubKeyFrom = keyUser.GetPubKey(); - if (!keyUser.Sign(product.GetSigHash(), product.vchSig)) - { - wxMessageBox("Error digitally signing the product "); - return; - } - - // Save detailed product - AddToMyProducts(product); - - // Strip down to summary product - product.mapDetails.clear(); - product.vOrderForm.clear(); - - // Sign the summary product - if (!keyUser.Sign(product.GetSigHash(), product.vchSig)) - { - wxMessageBox("Error digitally signing the product "); - return; - } - - // Verify - if (!product.CheckProduct()) - { - wxMessageBox("Errors found in product "); - return; - } - - // Broadcast - AdvertStartPublish(pnodeLocalHost, MSG_PRODUCT, 0, product); - - Destroy(); -} - -void CEditProductDialog::OnButtonPreview(wxCommandEvent& event) -{ - CProduct product; - GetProduct(product); - CViewProductDialog* pdialog = new CViewProductDialog(this, product); - pdialog->Show(); -} - -void CEditProductDialog::OnButtonCancel(wxCommandEvent& event) -{ - Destroy(); -} - -void CEditProductDialog::SetProduct(const CProduct& productIn) -{ - CProduct product = productIn; - - m_comboBoxCategory->SetValue(product.mapValue["category"]); - m_textCtrlTitle->SetValue(product.mapValue["title"]); - m_textCtrlPrice->SetValue(product.mapValue["price"]); - m_textCtrlDescription->SetValue(product.mapValue["description"]); - m_textCtrlInstructions->SetValue(product.mapValue["instructions"]); - - for (int i = 0; i < FIELDS_MAX; i++) - { - bool fUsed = i < product.vOrderForm.size(); - m_buttonDel[i]->Show(fUsed); - m_textCtrlLabel[i]->Show(fUsed); - m_textCtrlField[i]->Show(fUsed); - if (!fUsed) - continue; - - m_textCtrlLabel[i]->SetValue(product.vOrderForm[i].first); - string strControl = product.vOrderForm[i].second; - if (strControl.substr(0, 5) == "text=") - m_textCtrlField[i]->SetValue(""); - else if (strControl.substr(0, 7) == "choice=") - m_textCtrlField[i]->SetValue(strControl.substr(7)); - else - m_textCtrlField[i]->SetValue(strControl); - } -} - -void CEditProductDialog::GetProduct(CProduct& product) -{ - // map mapValue; - // vector > vOrderForm; - - product.mapValue["category"] = m_comboBoxCategory->GetValue().Trim(); - product.mapValue["title"] = m_textCtrlTitle->GetValue().Trim(); - product.mapValue["price"] = m_textCtrlPrice->GetValue().Trim(); - product.mapValue["description"] = m_textCtrlDescription->GetValue().Trim(); - product.mapValue["instructions"] = m_textCtrlInstructions->GetValue().Trim(); - - for (int i = 0; i < FIELDS_MAX; i++) - { - if (m_buttonDel[i]->IsShown()) - { - string strLabel = (string)m_textCtrlLabel[i]->GetValue().Trim(); - string strControl = (string)m_textCtrlField[i]->GetValue(); - if (strControl.empty()) - strControl = "text="; - else - strControl = "choice=" + strControl; - product.vOrderForm.push_back(make_pair(strLabel, strControl)); - } - } -} - - - - - - - -////////////////////////////////////////////////////////////////////////////// -// -// CViewProductDialog -// - -CViewProductDialog::CViewProductDialog(wxWindow* parent, const CProduct& productIn) : CViewProductDialogBase(parent) -{ - Connect(wxEVT_REPLY1, wxCommandEventHandler(CViewProductDialog::OnReply1), NULL, this); - AddCallbackAvailable(GetEventHandler()); - - // Fill display with product summary while waiting for details - product = productIn; - UpdateProductDisplay(false); - - m_buttonBack->Enable(false); - m_buttonNext->Enable(!product.vOrderForm.empty()); - m_htmlWinReviews->Show(true); - m_scrolledWindow->Show(false); - this->Layout(); - - // Request details from seller - CreateThread(ThreadRequestProductDetails, new pair(product, GetEventHandler())); -} - -CViewProductDialog::~CViewProductDialog() -{ - RemoveCallbackAvailable(GetEventHandler()); -} - -void ThreadRequestProductDetails(void* parg) -{ - // Extract parameters - pair* pitem = (pair*)parg; - CProduct product = pitem->first; - wxEvtHandler* pevthandler = pitem->second; - delete pitem; - - // Connect to seller - CNode* pnode = ConnectNode(product.addr, 5 * 60); - if (!pnode) - { - CDataStream ssEmpty; - AddPendingReplyEvent1(pevthandler, ssEmpty); - return; - } - - // Request detailed product, with response going to OnReply1 via dialog's event handler - pnode->PushRequest("getdetails", product.GetHash(), AddPendingReplyEvent1, (void*)pevthandler); -} - -void CViewProductDialog::OnReply1(wxCommandEvent& event) -{ - CDataStream ss = GetStreamFromEvent(event); - if (ss.empty()) - { - product.mapValue["description"] = "-- CAN'T CONNECT TO SELLER --\n"; - UpdateProductDisplay(true); - return; - } - - int nRet; - CProduct product2; - try - { - ss >> nRet; - if (nRet > 0) - throw false; - ss >> product2; - if (product2.GetHash() != product.GetHash()) - throw false; - if (!product2.CheckSignature()) - throw false; - } - catch (...) - { - product.mapValue["description"] = "-- INVALID RESPONSE --\n"; - UpdateProductDisplay(true); - return; - } - - product = product2; - UpdateProductDisplay(true); -} - -bool CompareReviewsBestFirst(const CReview* p1, const CReview* p2) -{ - return (p1->nAtoms > p2->nAtoms); -} - -void CViewProductDialog::UpdateProductDisplay(bool fDetails) -{ - // Product and reviews - string strHTML; - strHTML.reserve(4000); - strHTML += "\n" - "\n" - "\n" - "\n" - "\n"; - strHTML += "Category: " + HtmlEscape(product.mapValue["category"]) + "
\n"; - strHTML += "Title: " + HtmlEscape(product.mapValue["title"]) + "
\n"; - strHTML += "Price: " + HtmlEscape(product.mapValue["price"]) + "
\n"; - - if (!fDetails) - strHTML += "Loading details...
\n
\n"; - else - strHTML += HtmlEscape(product.mapValue["description"], true) + "
\n
\n"; - - strHTML += "Reviews:
\n
\n"; - - if (!product.vchPubKeyFrom.empty()) - { - CReviewDB reviewdb("r"); - - // Get reviews - vector vReviews; - reviewdb.ReadReviews(product.GetUserHash(), vReviews); - - // Get reviewer's number of atoms - vector vSortedReviews; - vSortedReviews.reserve(vReviews.size()); - for (vector::reverse_iterator it = vReviews.rbegin(); it != vReviews.rend(); ++it) - { - CReview& review = *it; - CUser user; - reviewdb.ReadUser(review.GetUserHash(), user); - review.nAtoms = user.GetAtomCount(); - vSortedReviews.push_back(&review); - } - - reviewdb.Close(); - - // Sort - stable_sort(vSortedReviews.begin(), vSortedReviews.end(), CompareReviewsBestFirst); - - // Format reviews - foreach(CReview* preview, vSortedReviews) - { - CReview& review = *preview; - int nStars = atoi(review.mapValue["stars"].c_str()); - if (nStars < 1 || nStars > 5) - continue; - - strHTML += "" + itostr(nStars) + (nStars == 1 ? " star" : " stars") + ""; - strHTML += "     "; - strHTML += DateStr(atoi64(review.mapValue["date"])) + "
\n"; - strHTML += HtmlEscape(review.mapValue["review"], true); - strHTML += "
\n
\n"; - } - } - - strHTML += "\n\n"; - - // Shrink capacity to fit - string(strHTML.begin(), strHTML.end()).swap(strHTML); - - m_htmlWinReviews->SetPage(strHTML); - - ///// need to find some other indicator to use so can allow empty order form - if (product.vOrderForm.empty()) - return; - - // Order form - m_staticTextInstructions->SetLabel(product.mapValue["instructions"]); - for (int i = 0; i < FIELDS_MAX; i++) - { - m_staticTextLabel[i] = NULL; - m_textCtrlField[i] = NULL; - m_choiceField[i] = NULL; - } - - // Construct flexgridsizer - wxBoxSizer* bSizer21 = (wxBoxSizer*)m_scrolledWindow->GetSizer(); - wxFlexGridSizer* fgSizer; - fgSizer = new wxFlexGridSizer(0, 2, 0, 0); - fgSizer->AddGrowableCol(1); - fgSizer->SetFlexibleDirection(wxBOTH); - fgSizer->SetNonFlexibleGrowMode(wxFLEX_GROWMODE_SPECIFIED); - - // Construct order form fields - wxWindow* windowLast = NULL; - for (int i = 0; i < product.vOrderForm.size(); i++) - { - string strLabel = product.vOrderForm[i].first; - string strControl = product.vOrderForm[i].second; - - if (strLabel.size() < 20) - strLabel.insert(strLabel.begin(), 20 - strLabel.size(), ' '); - - m_staticTextLabel[i] = new wxStaticText(m_scrolledWindow, wxID_ANY, strLabel, wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT); - m_staticTextLabel[i]->Wrap(-1); - fgSizer->Add(m_staticTextLabel[i], 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxALL, 5); - - if (strControl.substr(0, 5) == "text=") - { - m_textCtrlField[i] = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - fgSizer->Add(m_textCtrlField[i], 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5); - windowLast = m_textCtrlField[i]; - } - else if (strControl.substr(0, 7) == "choice=") - { - vector vChoices; - ParseString(strControl.substr(7), ',', vChoices); - - wxArrayString arraystring; - foreach(const string& str, vChoices) - arraystring.Add(str); - - m_choiceField[i] = new wxChoice(m_scrolledWindow, wxID_ANY, wxDefaultPosition, wxDefaultSize, arraystring, 0); - fgSizer->Add(m_choiceField[i], 0, wxALL|wxALIGN_CENTER_VERTICAL, 5); - windowLast = m_choiceField[i]; - } - else - { - m_textCtrlField[i] = new wxTextCtrl(m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0); - fgSizer->Add(m_textCtrlField[i], 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5); - m_staticTextLabel[i]->Show(false); - m_textCtrlField[i]->Show(false); - } - } - - // Insert after instructions and before submit/cancel buttons - bSizer21->Insert(2, fgSizer, 0, wxEXPAND|wxRIGHT|wxLEFT, 5); - m_scrolledWindow->Layout(); - bSizer21->Fit(m_scrolledWindow); - this->Layout(); - - // Fixup the tab order - m_buttonSubmitForm->MoveAfterInTabOrder(windowLast); - m_buttonCancelForm->MoveAfterInTabOrder(m_buttonSubmitForm); - //m_buttonBack->MoveAfterInTabOrder(m_buttonCancelForm); - //m_buttonNext->MoveAfterInTabOrder(m_buttonBack); - //m_buttonCancel->MoveAfterInTabOrder(m_buttonNext); - this->Layout(); -} - -void CViewProductDialog::GetOrder(CWalletTx& wtx) -{ - wtx.SetNull(); - for (int i = 0; i < product.vOrderForm.size(); i++) - { - string strValue; - if (m_textCtrlField[i]) - strValue = m_textCtrlField[i]->GetValue().Trim(); - else - strValue = m_choiceField[i]->GetStringSelection(); - wtx.vOrderForm.push_back(make_pair(m_staticTextLabel[i]->GetLabel(), strValue)); - } -} - -void CViewProductDialog::OnButtonSubmitForm(wxCommandEvent& event) -{ - m_buttonSubmitForm->Enable(false); - m_buttonCancelForm->Enable(false); - - CWalletTx wtx; - GetOrder(wtx); - - CSendingDialog* pdialog = new CSendingDialog(this, product.addr, atoi64(product.mapValue["price"]), wtx); - if (!pdialog->ShowModal()) - { - m_buttonSubmitForm->Enable(true); - m_buttonCancelForm->Enable(true); - return; - } -} - -void CViewProductDialog::OnButtonCancelForm(wxCommandEvent& event) -{ - Destroy(); -} - -void CViewProductDialog::OnButtonBack(wxCommandEvent& event) -{ - Freeze(); - m_htmlWinReviews->Show(true); - m_scrolledWindow->Show(false); - m_buttonBack->Enable(false); - m_buttonNext->Enable(!product.vOrderForm.empty()); - this->Layout(); - Thaw(); -} - -void CViewProductDialog::OnButtonNext(wxCommandEvent& event) -{ - if (!product.vOrderForm.empty()) - { - Freeze(); - m_htmlWinReviews->Show(false); - m_scrolledWindow->Show(true); - m_buttonBack->Enable(true); - m_buttonNext->Enable(false); - this->Layout(); - Thaw(); - } -} - -void CViewProductDialog::OnButtonCancel(wxCommandEvent& event) -{ - Destroy(); -} - - - - - - - -////////////////////////////////////////////////////////////////////////////// -// -// CViewOrderDialog -// - -CViewOrderDialog::CViewOrderDialog(wxWindow* parent, CWalletTx order, bool fReceived) : CViewOrderDialogBase(parent) -{ - int64 nPrice = (fReceived ? order.GetCredit() : order.GetDebit()); - - string strHTML; - strHTML.reserve(4000); - strHTML += "\n" - "\n" - "\n" - "\n" - "\n"; - strHTML += "Time: " + HtmlEscape(DateTimeStr(order.nTimeReceived)) + "
\n"; - strHTML += "Price: " + HtmlEscape(FormatMoney(nPrice)) + "
\n"; - strHTML += "Status: " + HtmlEscape(FormatTxStatus(order)) + "
\n"; - - strHTML += "\n"; - for (int i = 0; i < order.vOrderForm.size(); i++) - { - strHTML += " "; - strHTML += "\n"; - } - strHTML += "
" + HtmlEscape(order.vOrderForm[i].first) + ":" + HtmlEscape(order.vOrderForm[i].second) + "
\n"; - - strHTML += "\n\n"; - - // Shrink capacity to fit - // (strings are ref counted, so it may live on in SetPage) - string(strHTML.begin(), strHTML.end()).swap(strHTML); - - m_htmlWin->SetPage(strHTML); -} - -void CViewOrderDialog::OnButtonOK(wxCommandEvent& event) -{ - Destroy(); -} - - - - - - - -////////////////////////////////////////////////////////////////////////////// -// -// CEditReviewDialog -// - -CEditReviewDialog::CEditReviewDialog(wxWindow* parent) : CEditReviewDialogBase(parent) -{ -} - -void CEditReviewDialog::OnButtonSubmit(wxCommandEvent& event) -{ - if (m_choiceStars->GetSelection() == -1) - { - wxMessageBox("Please select a rating "); - return; - } - - CReview review; - GetReview(review); - - // Sign the review - review.vchPubKeyFrom = keyUser.GetPubKey(); - if (!keyUser.Sign(review.GetSigHash(), review.vchSig)) - { - wxMessageBox("Unable to digitally sign the review "); - return; - } - - // Broadcast - if (!review.AcceptReview()) - { - wxMessageBox("Save failed "); - return; - } - RelayMessage(CInv(MSG_REVIEW, review.GetHash()), review); - - Destroy(); -} - -void CEditReviewDialog::OnButtonCancel(wxCommandEvent& event) -{ - Destroy(); -} - -void CEditReviewDialog::GetReview(CReview& review) -{ - review.mapValue["time"] = i64tostr(GetAdjustedTime()); - review.mapValue["stars"] = itostr(m_choiceStars->GetSelection()+1); - review.mapValue["review"] = m_textCtrlReview->GetValue(); -} - - - - - - - ////////////////////////////////////////////////////////////////////////////// // // CMyTaskBarIcon diff --git a/ui.h b/ui.h index 43e05659..da1ada7d 100644 --- a/ui.h +++ b/ui.h @@ -6,12 +6,6 @@ DECLARE_EVENT_TYPE(wxEVT_UITHREADCALL, -1) -DECLARE_EVENT_TYPE(wxEVT_REPLY1, -1) -DECLARE_EVENT_TYPE(wxEVT_REPLY2, -1) -DECLARE_EVENT_TYPE(wxEVT_REPLY3, -1) - - - extern map mapArgs; @@ -57,8 +51,8 @@ protected: void OnButtonAddressBook(wxCommandEvent& event); void OnSetFocusAddress(wxFocusEvent& event); void OnMouseEventsAddress(wxMouseEvent& event); - void OnButtonCopy(wxCommandEvent& event); void OnButtonChange(wxCommandEvent& event); + void OnButtonCopy(wxCommandEvent& event); void OnListColBeginDrag(wxListEvent& event); void OnListItemActivated(wxListEvent& event); void OnListItemActivatedProductsSent(wxListEvent& event); @@ -263,138 +257,6 @@ public: -class CProductsDialog : public CProductsDialogBase -{ -protected: - // Event handlers - void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } - void OnCombobox(wxCommandEvent& event); - void OnButtonSearch(wxCommandEvent& event); - void OnListItemActivated(wxListEvent& event); - -public: - /** Constructor */ - CProductsDialog(wxWindow* parent); - - // Custom - vector m_vProduct; -}; - - - -class CEditProductDialog : public CEditProductDialogBase -{ -protected: - // Event handlers - void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } - void OnButtonDel0(wxCommandEvent& event); - void OnButtonDel1(wxCommandEvent& event); - void OnButtonDel2(wxCommandEvent& event); - void OnButtonDel3(wxCommandEvent& event); - void OnButtonDel4(wxCommandEvent& event); - void OnButtonDel5(wxCommandEvent& event); - void OnButtonDel6(wxCommandEvent& event); - void OnButtonDel7(wxCommandEvent& event); - void OnButtonDel8(wxCommandEvent& event); - void OnButtonDel9(wxCommandEvent& event); - void OnButtonDel10(wxCommandEvent& event); - void OnButtonDel11(wxCommandEvent& event); - void OnButtonDel12(wxCommandEvent& event); - void OnButtonDel13(wxCommandEvent& event); - void OnButtonDel14(wxCommandEvent& event); - void OnButtonDel15(wxCommandEvent& event); - void OnButtonDel16(wxCommandEvent& event); - void OnButtonDel17(wxCommandEvent& event); - void OnButtonDel18(wxCommandEvent& event); - void OnButtonDel19(wxCommandEvent& event); - void OnButtonAddField(wxCommandEvent& event); - void OnButtonSend(wxCommandEvent& event); - void OnButtonPreview(wxCommandEvent& event); - void OnButtonCancel(wxCommandEvent& event); - -public: - /** Constructor */ - CEditProductDialog(wxWindow* parent); - - // Custom - enum { FIELDS_MAX = 20 }; - wxTextCtrl* m_textCtrlLabel[FIELDS_MAX]; - wxTextCtrl* m_textCtrlField[FIELDS_MAX]; - wxButton* m_buttonDel[FIELDS_MAX]; - - void LayoutAll(); - void ShowLine(int i, bool fShow=true); - void OnButtonDel(wxCommandEvent& event, int n); - void SetProduct(const CProduct& productIn); - void GetProduct(CProduct& product); - -}; - - - -class CViewProductDialog : public CViewProductDialogBase -{ -protected: - // Event handlers - void OnButtonSubmitForm(wxCommandEvent& event); - void OnButtonCancelForm(wxCommandEvent& event); - void OnButtonBack(wxCommandEvent& event); - void OnButtonNext(wxCommandEvent& event); - void OnButtonCancel(wxCommandEvent& event); - -public: - /** Constructor */ - CViewProductDialog(wxWindow* parent, const CProduct& productIn); - ~CViewProductDialog(); - - // Custom - CProduct product; - enum { FIELDS_MAX = 20 }; - wxStaticText* m_staticTextLabel[FIELDS_MAX]; - wxTextCtrl* m_textCtrlField[FIELDS_MAX]; - wxChoice* m_choiceField[FIELDS_MAX]; - - void GetOrder(CWalletTx& order); - void UpdateProductDisplay(bool fDetails); - void OnReply1(wxCommandEvent& event); -}; - - - -class CViewOrderDialog : public CViewOrderDialogBase -{ -protected: - // Event handlers - void OnButtonOK(wxCommandEvent& event); - -public: - /** Constructor */ - CViewOrderDialog(wxWindow* parent, CWalletTx order, bool fReceived); - - // Custom - bool fReceived; -}; - - - -class CEditReviewDialog : public CEditReviewDialogBase -{ -protected: - // Event handlers - void OnKeyDown(wxKeyEvent& event) { HandleCtrlA(event); } - void OnButtonSubmit(wxCommandEvent& event); - void OnButtonCancel(wxCommandEvent& event); - -public: - /** Constructor */ - CEditReviewDialog(wxWindow* parent); - - // Custom - void GetReview(CReview& review); -}; - - - class CGetTextFromUserDialog : public CGetTextFromUserDialogBase { protected: @@ -430,6 +292,8 @@ public: m_textCtrl2->SetValue(strValue2); SetSize(wxDefaultCoord, 180); } + if (!fWindows) + SetSize(1.14 * GetSize().GetWidth(), 1.14 * GetSize().GetHeight()); } // Custom diff --git a/uibase.cpp b/uibase.cpp index 08c025fa..03bd7912 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -27,39 +27,39 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_menuFile = new wxMenu(); wxMenuItem* m_menuFileExit; - m_menuFileExit = new wxMenuItem( m_menuFile, wxID_ANY, wxString( wxT("E&xit") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuFileExit = new wxMenuItem( m_menuFile, wxID_ANY, wxString( _("E&xit") ) , wxEmptyString, wxITEM_NORMAL ); m_menuFile->Append( m_menuFileExit ); - m_menubar->Append( m_menuFile, wxT("&File") ); + m_menubar->Append( m_menuFile, _("&File") ); m_menuView = new wxMenu(); wxMenuItem* m_menuViewShowGenerated; - m_menuViewShowGenerated = new wxMenuItem( m_menuView, wxID_VIEWSHOWGENERATED, wxString( wxT("&Show Generated Coins") ) , wxEmptyString, wxITEM_CHECK ); + m_menuViewShowGenerated = new wxMenuItem( m_menuView, wxID_VIEWSHOWGENERATED, wxString( _("&Show Generated Coins") ) , wxEmptyString, wxITEM_CHECK ); m_menuView->Append( m_menuViewShowGenerated ); - m_menubar->Append( m_menuView, wxT("&View") ); + m_menubar->Append( m_menuView, _("&View") ); m_menuOptions = new wxMenu(); wxMenuItem* m_menuOptionsGenerateBitcoins; - m_menuOptionsGenerateBitcoins = new wxMenuItem( m_menuOptions, wxID_OPTIONSGENERATEBITCOINS, wxString( wxT("&Generate Coins") ) , wxEmptyString, wxITEM_CHECK ); + m_menuOptionsGenerateBitcoins = new wxMenuItem( m_menuOptions, wxID_OPTIONSGENERATEBITCOINS, wxString( _("&Generate Coins") ) , wxEmptyString, wxITEM_CHECK ); m_menuOptions->Append( m_menuOptionsGenerateBitcoins ); wxMenuItem* m_menuOptionsChangeYourAddress; - m_menuOptionsChangeYourAddress = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( wxT("&Change Your Address...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptionsChangeYourAddress = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( _("&Change Your Address...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuOptions->Append( m_menuOptionsChangeYourAddress ); wxMenuItem* m_menuOptionsOptions; - m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_MENUOPTIONSOPTIONS, wxString( wxT("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_MENUOPTIONSOPTIONS, wxString( _("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuOptions->Append( m_menuOptionsOptions ); - m_menubar->Append( m_menuOptions, wxT("&Options") ); + m_menubar->Append( m_menuOptions, _("&Options") ); m_menuHelp = new wxMenu(); wxMenuItem* m_menuHelpAbout; - m_menuHelpAbout = new wxMenuItem( m_menuHelp, wxID_ANY, wxString( wxT("&About...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuHelpAbout = new wxMenuItem( m_menuHelp, wxID_ANY, wxString( _("&About...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuHelp->Append( m_menuHelpAbout ); - m_menubar->Append( m_menuHelp, wxT("&Help") ); + m_menubar->Append( m_menuHelp, _("&Help") ); this->SetMenuBar( m_menubar ); @@ -68,8 +68,8 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_toolBar->SetToolSeparation( 1 ); m_toolBar->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); - m_toolBar->AddTool( wxID_BUTTONSEND, wxT("&Send Coins"), wxBitmap( send20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); - m_toolBar->AddTool( wxID_BUTTONRECEIVE, wxT("&Address Book"), wxBitmap( addressbook20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->AddTool( wxID_BUTTONSEND, _("&Send Coins"), wxBitmap( send20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->AddTool( wxID_BUTTONRECEIVE, _("&Address Book"), wxBitmap( addressbook20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); m_toolBar->Realize(); m_statusBar = this->CreateStatusBar( 1, wxST_SIZEGRIP, wxID_ANY ); @@ -84,7 +84,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& wxBoxSizer* bSizer85; bSizer85 = new wxBoxSizer( wxHORIZONTAL ); - m_staticText32 = new wxStaticText( this, wxID_ANY, wxT("Your Bitcoin Address:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText32 = new wxStaticText( this, wxID_ANY, _("Your Bitcoin Address:"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText32->Wrap( -1 ); bSizer85->Add( m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxLEFT, 5 ); @@ -93,13 +93,13 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& bSizer85->Add( m_textCtrlAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT(" &Copy to Clipboard "), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT ); - bSizer85->Add( m_buttonCopy, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); + m_buttonNew = new wxButton( this, wxID_BUTTONCHANGE, _("&New..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonNew->Hide(); - m_button91 = new wxButton( this, wxID_BUTTONCHANGE, wxT("C&hange..."), wxDefaultPosition, wxDefaultSize, 0 ); - m_button91->Hide(); + bSizer85->Add( m_buttonNew, 0, wxRIGHT, 5 ); - bSizer85->Add( m_button91, 0, wxRIGHT, 5 ); + m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, _(" &Copy to Clipboard "), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT ); + bSizer85->Add( m_buttonCopy, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); bSizer85->Add( 0, 0, 0, wxEXPAND, 5 ); @@ -113,7 +113,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& wxBoxSizer* bSizer66; bSizer66 = new wxBoxSizer( wxHORIZONTAL ); - m_staticText41 = new wxStaticText( m_panel14, wxID_ANY, wxT("Balance:"), wxDefaultPosition, wxSize( -1,15 ), 0 ); + m_staticText41 = new wxStaticText( m_panel14, wxID_ANY, _("Balance:"), wxDefaultPosition, wxSize( -1,15 ), 0 ); m_staticText41->Wrap( -1 ); bSizer66->Add( m_staticText41, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 ); @@ -132,7 +132,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& bSizer3->Add( 0, 0, 0, wxEXPAND, 5 ); - wxString m_choiceFilterChoices[] = { wxT(" All"), wxT(" Sent"), wxT(" Received"), wxT(" In Progress") }; + wxString m_choiceFilterChoices[] = { _(" All"), _(" Sent"), _(" Received"), _(" In Progress") }; int m_choiceFilterNChoices = sizeof( m_choiceFilterChoices ) / sizeof( wxString ); m_choiceFilter = new wxChoice( this, wxID_ANY, wxDefaultPosition, wxSize( 110,-1 ), m_choiceFilterNChoices, m_choiceFilterChoices, 0 ); m_choiceFilter->SetSelection( 0 ); @@ -191,8 +191,8 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_textCtrlAddress->Connect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Connect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); + m_buttonNew->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); m_buttonCopy->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); - m_button91->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); m_listCtrl->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); m_listCtrl->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); @@ -243,8 +243,8 @@ CMainFrameBase::~CMainFrameBase() m_textCtrlAddress->Disconnect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Disconnect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); + m_buttonNew->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); m_buttonCopy->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); - m_button91->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); m_listCtrl->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); @@ -268,7 +268,7 @@ CTxDetailsDialogBase::CTxDetailsDialogBase( wxWindow* parent, wxWindowID id, con wxBoxSizer* bSizer65; bSizer65 = new wxBoxSizer( wxHORIZONTAL ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, _("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer65->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer64->Add( bSizer65, 0, wxALIGN_RIGHT, 5 ); @@ -311,7 +311,7 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer69->Add( 0, 16, 0, wxEXPAND, 5 ); - m_staticText32 = new wxStaticText( m_panelMain, wxID_ANY, wxT("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText32 = new wxStaticText( m_panelMain, wxID_ANY, _("Optional transaction fee you give to the nodes that process your transactions."), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText32->Wrap( -1 ); m_staticText32->Hide(); @@ -320,7 +320,7 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w wxBoxSizer* bSizer56; bSizer56 = new wxBoxSizer( wxHORIZONTAL ); - m_staticText31 = new wxStaticText( m_panelMain, wxID_ANY, wxT("Transaction fee:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText31 = new wxStaticText( m_panelMain, wxID_ANY, _("Transaction fee:"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText31->Wrap( -1 ); m_staticText31->Hide(); @@ -336,35 +336,35 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w wxBoxSizer* bSizer71; bSizer71 = new wxBoxSizer( wxHORIZONTAL ); - m_checkBoxLimitProcessors = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Limit coin generation to"), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkBoxLimitProcessors = new wxCheckBox( m_panelMain, wxID_ANY, _("&Limit coin generation to"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer71->Add( m_checkBoxLimitProcessors, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); m_spinCtrlLimitProcessors = new wxSpinCtrl( m_panelMain, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( 48,-1 ), wxSP_ARROW_KEYS, 1, 999, 1 ); bSizer71->Add( m_spinCtrlLimitProcessors, 0, wxALIGN_CENTER_VERTICAL, 5 ); - m_staticText35 = new wxStaticText( m_panelMain, wxID_ANY, wxT("processors"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText35 = new wxStaticText( m_panelMain, wxID_ANY, _("processors"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText35->Wrap( -1 ); bSizer71->Add( m_staticText35, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); bSizer69->Add( bSizer71, 0, 0, 5 ); - m_checkBoxStartOnSystemStartup = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Start Bitcoin on system startup"), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkBoxStartOnSystemStartup = new wxCheckBox( m_panelMain, wxID_ANY, _("&Start Bitcoin on system startup"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_checkBoxStartOnSystemStartup, 0, wxALL, 5 ); - m_checkBoxMinimizeToTray = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Minimize to the tray instead of the taskbar"), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkBoxMinimizeToTray = new wxCheckBox( m_panelMain, wxID_ANY, _("&Minimize to the tray instead of the taskbar"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_checkBoxMinimizeToTray, 0, wxALL, 5 ); - m_checkBoxMinimizeOnClose = new wxCheckBox( m_panelMain, wxID_ANY, wxT("M&inimize to the tray on close"), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkBoxMinimizeOnClose = new wxCheckBox( m_panelMain, wxID_ANY, _("M&inimize to the tray on close"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_checkBoxMinimizeOnClose, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); wxBoxSizer* bSizer102; bSizer102 = new wxBoxSizer( wxHORIZONTAL ); - m_checkBoxUseProxy = new wxCheckBox( m_panelMain, wxID_ANY, wxT("&Connect through socks4 proxy: "), wxDefaultPosition, wxDefaultSize, 0 ); + m_checkBoxUseProxy = new wxCheckBox( m_panelMain, wxID_ANY, _("&Connect through socks4 proxy: "), wxDefaultPosition, wxDefaultSize, 0 ); bSizer102->Add( m_checkBoxUseProxy, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); @@ -376,7 +376,7 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer103->Add( 18, 0, 0, 0, 5 ); - m_staticTextProxyIP = new wxStaticText( m_panelMain, wxID_ANY, wxT("Proxy &IP:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextProxyIP = new wxStaticText( m_panelMain, wxID_ANY, _("Proxy &IP:"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextProxyIP->Wrap( -1 ); bSizer103->Add( m_staticTextProxyIP, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); @@ -384,7 +384,7 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w m_textCtrlProxyIP->SetMaxLength( 15 ); bSizer103->Add( m_textCtrlProxyIP, 0, wxALIGN_CENTER_VERTICAL, 5 ); - m_staticTextProxyPort = new wxStaticText( m_panelMain, wxID_ANY, wxT(" &Port:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextProxyPort = new wxStaticText( m_panelMain, wxID_ANY, _(" &Port:"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextProxyPort->Wrap( -1 ); bSizer103->Add( m_staticTextProxyPort, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); @@ -406,19 +406,19 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer64->Add( 0, 16, 0, wxEXPAND, 5 ); - m_staticText321 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("Test panel 2 for future expansion"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText321 = new wxStaticText( m_panelTest2, wxID_ANY, _("Test panel 2 for future expansion"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText321->Wrap( -1 ); bSizer64->Add( m_staticText321, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); - m_staticText69 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("Let's not start multiple pages until the first page is filled up"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText69 = new wxStaticText( m_panelTest2, wxID_ANY, _("Let's not start multiple pages until the first page is filled up"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText69->Wrap( -1 ); bSizer64->Add( m_staticText69, 0, wxALL, 5 ); - m_staticText70 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText70 = new wxStaticText( m_panelTest2, wxID_ANY, _("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText70->Wrap( -1 ); bSizer64->Add( m_staticText70, 0, wxALL, 5 ); - m_staticText71 = new wxStaticText( m_panelTest2, wxID_ANY, wxT("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText71 = new wxStaticText( m_panelTest2, wxID_ANY, _("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText71->Wrap( -1 ); bSizer64->Add( m_staticText71, 0, wxALL, 5 ); @@ -437,13 +437,13 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w wxBoxSizer* bSizer58; bSizer58 = new wxBoxSizer( wxHORIZONTAL ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, _("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer58->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel = new wxButton( this, wxID_CANCEL, _("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer58->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonApply = new wxButton( this, wxID_APPLY, wxT("&Apply"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonApply = new wxButton( this, wxID_APPLY, _("&Apply"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer58->Add( m_buttonApply, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer55->Add( bSizer58, 0, wxALIGN_RIGHT, 5 ); @@ -504,13 +504,13 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr wxBoxSizer* bSizer64; bSizer64 = new wxBoxSizer( wxHORIZONTAL ); - m_staticText40 = new wxStaticText( this, wxID_ANY, wxT("Bitcoin "), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText40 = new wxStaticText( this, wxID_ANY, _("Bitcoin "), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText40->Wrap( -1 ); m_staticText40->SetFont( wxFont( 10, 74, 90, 92, false, wxT("Tahoma") ) ); bSizer64->Add( m_staticText40, 0, wxALIGN_BOTTOM|wxTOP|wxBOTTOM|wxLEFT, 5 ); - m_staticTextVersion = new wxStaticText( this, wxID_ANY, wxT("version"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextVersion = new wxStaticText( this, wxID_ANY, _("version"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextVersion->Wrap( -1 ); m_staticTextVersion->SetFont( wxFont( 10, 74, 90, 90, false, wxT("Tahoma") ) ); @@ -521,7 +521,7 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr bSizer631->Add( 0, 4, 0, wxEXPAND, 5 ); - m_staticTextMain = new wxStaticText( this, wxID_ANY, wxT("Copyright © 2009-2010 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file \nlicense.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the \nOpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by \nEric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMain = new wxStaticText( this, wxID_ANY, _("Copyright © 2009-2010 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file \nlicense.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the \nOpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by \nEric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextMain->Wrap( -1 ); bSizer631->Add( m_staticTextMain, 0, wxALL, 5 ); @@ -538,7 +538,7 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr bSizer61->Add( 0, 0, 1, wxEXPAND, 5 ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, _("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer61->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer60->Add( bSizer61, 0, wxALIGN_RIGHT|wxEXPAND|wxRIGHT, 5 ); @@ -577,7 +577,7 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin fgSizer1->Add( 0, 0, 0, wxEXPAND, 5 ); - m_staticTextInstructions = new wxStaticText( this, wxID_ANY, wxT("Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, \nor Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextInstructions = new wxStaticText( this, wxID_ANY, _("Enter the recipient's IP address (e.g. 123.45.6.7) for online transfer with comments and confirmation, \nor Bitcoin address (e.g. 1NS17iag9jJgTHD1VXjvLCEnZuQ3rJED9L) if recipient is not online."), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextInstructions->Wrap( -1 ); fgSizer1->Add( m_staticTextInstructions, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); @@ -591,7 +591,7 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin m_bitmapCheckMark = new wxStaticBitmap( this, wxID_ANY, wxBitmap( check_xpm ), wxDefaultPosition, wxSize( 16,16 ), 0 ); bSizer47->Add( m_bitmapCheckMark, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - m_staticText36 = new wxStaticText( this, wxID_ANY, wxT("Pay &To:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); + m_staticText36 = new wxStaticText( this, wxID_ANY, _("Pay &To:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); m_staticText36->Wrap( -1 ); bSizer47->Add( m_staticText36, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 ); @@ -606,17 +606,17 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin wxBoxSizer* bSizer66; bSizer66 = new wxBoxSizer( wxHORIZONTAL ); - m_buttonPaste = new wxButton( this, wxID_BUTTONPASTE, wxT("&Paste"), wxDefaultPosition, wxSize( -1,-1 ), wxBU_EXACTFIT ); + m_buttonPaste = new wxButton( this, wxID_BUTTONPASTE, _("&Paste"), wxDefaultPosition, wxSize( -1,-1 ), wxBU_EXACTFIT ); bSizer66->Add( m_buttonPaste, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxEXPAND, 5 ); - m_buttonAddress = new wxButton( this, wxID_BUTTONADDRESSBOOK, wxT(" Address &Book..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonAddress = new wxButton( this, wxID_BUTTONADDRESSBOOK, _(" Address &Book..."), wxDefaultPosition, wxDefaultSize, 0 ); bSizer66->Add( m_buttonAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxEXPAND, 5 ); bSizer19->Add( bSizer66, 0, wxALIGN_CENTER_VERTICAL, 5 ); fgSizer1->Add( bSizer19, 1, wxEXPAND|wxRIGHT, 5 ); - m_staticText19 = new wxStaticText( this, wxID_ANY, wxT("&Amount:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); + m_staticText19 = new wxStaticText( this, wxID_ANY, _("&Amount:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); m_staticText19->Wrap( -1 ); fgSizer1->Add( m_staticText19, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT|wxALIGN_RIGHT, 5 ); @@ -626,13 +626,13 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin fgSizer1->Add( m_textCtrlAmount, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); - m_staticText20 = new wxStaticText( this, wxID_ANY, wxT("T&ransfer:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); + m_staticText20 = new wxStaticText( this, wxID_ANY, _("T&ransfer:"), wxDefaultPosition, wxSize( -1,-1 ), wxALIGN_RIGHT ); m_staticText20->Wrap( -1 ); m_staticText20->Hide(); fgSizer1->Add( m_staticText20, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); - wxString m_choiceTransferTypeChoices[] = { wxT(" Standard") }; + wxString m_choiceTransferTypeChoices[] = { _(" Standard") }; int m_choiceTransferTypeNChoices = sizeof( m_choiceTransferTypeChoices ) / sizeof( wxString ); m_choiceTransferType = new wxChoice( this, wxID_CHOICETRANSFERTYPE, wxDefaultPosition, wxDefaultSize, m_choiceTransferTypeNChoices, m_choiceTransferTypeChoices, 0 ); m_choiceTransferType->SetSelection( 0 ); @@ -654,7 +654,7 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin wxBoxSizer* bSizer681; bSizer681 = new wxBoxSizer( wxVERTICAL ); - m_staticTextFrom = new wxStaticText( this, wxID_ANY, wxT("&From:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextFrom = new wxStaticText( this, wxID_ANY, _("&From:"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextFrom->Wrap( -1 ); bSizer681->Add( m_staticTextFrom, 0, wxBOTTOM|wxLEFT, 5 ); @@ -671,7 +671,7 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin wxBoxSizer* bSizer68; bSizer68 = new wxBoxSizer( wxVERTICAL ); - m_staticTextMessage = new wxStaticText( this, wxID_ANY, wxT("&Message:"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMessage = new wxStaticText( this, wxID_ANY, _("&Message:"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextMessage->Wrap( -1 ); bSizer68->Add( m_staticTextMessage, 0, wxTOP|wxBOTTOM|wxLEFT, 5 ); @@ -688,12 +688,12 @@ CSendDialogBase::CSendDialogBase( wxWindow* parent, wxWindowID id, const wxStrin bSizer23->Add( 0, 0, 1, wxEXPAND, 5 ); - m_buttonSend = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonSend = new wxButton( this, wxID_BUTTONSEND, _("&Send"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); m_buttonSend->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); bSizer23->Add( m_buttonSend, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel = new wxButton( this, wxID_CANCEL, _("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer23->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer21->Add( bSizer23, 0, wxEXPAND, 5 ); @@ -740,7 +740,7 @@ CSendingDialogBase::CSendingDialogBase( wxWindow* parent, wxWindowID id, const w m_staticTextSending->Wrap( -1 ); bSizer68->Add( m_staticTextSending, 0, wxALIGN_CENTER_VERTICAL|wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 8 ); - m_textCtrlStatus = new wxTextCtrl( this, wxID_ANY, wxT("\n\nConnecting..."), wxDefaultPosition, wxDefaultSize, wxTE_CENTRE|wxTE_MULTILINE|wxTE_NO_VSCROLL|wxTE_READONLY|wxNO_BORDER ); + m_textCtrlStatus = new wxTextCtrl( this, wxID_ANY, _("\n\nConnecting..."), wxDefaultPosition, wxDefaultSize, wxTE_CENTRE|wxTE_MULTILINE|wxTE_NO_VSCROLL|wxTE_READONLY|wxNO_BORDER ); m_textCtrlStatus->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE ) ); bSizer68->Add( m_textCtrlStatus, 1, wxEXPAND|wxRIGHT|wxLEFT, 10 ); @@ -751,12 +751,12 @@ CSendingDialogBase::CSendingDialogBase( wxWindow* parent, wxWindowID id, const w bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); - m_buttonOK = new wxButton( this, wxID_ANY, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonOK = new wxButton( this, wxID_ANY, _("OK"), wxDefaultPosition, wxDefaultSize, 0 ); m_buttonOK->Enable( false ); bSizer69->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel = new wxButton( this, wxID_CANCEL, _("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer69->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); @@ -790,7 +790,7 @@ CYourAddressDialogBase::CYourAddressDialogBase( wxWindow* parent, wxWindowID id, bSizer68->Add( 0, 5, 0, wxEXPAND, 5 ); - m_staticText45 = new wxStaticText( this, wxID_ANY, wxT("These are your Bitcoin addresses for receiving payments. You may want to give a different one to each sender so you can keep track of who is paying you. The highlighted address is displayed in the main window."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText45 = new wxStaticText( this, wxID_ANY, _("These are your Bitcoin addresses for receiving payments. You may want to give a different one to each sender so you can keep track of who is paying you. The highlighted address is displayed in the main window."), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText45->Wrap( 590 ); bSizer68->Add( m_staticText45, 0, wxALL, 5 ); @@ -803,19 +803,19 @@ CYourAddressDialogBase::CYourAddressDialogBase( wxWindow* parent, wxWindowID id, bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); - m_buttonRename = new wxButton( this, wxID_BUTTONRENAME, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonRename = new wxButton( this, wxID_BUTTONRENAME, _("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_buttonRename, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT(" &New Address... "), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonNew = new wxButton( this, wxID_BUTTONNEW, _(" &New Address... "), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer69->Add( m_buttonNew, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, wxT(" &Copy to Clipboard "), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, _(" &Copy to Clipboard "), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer69->Add( m_buttonCopy, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonOK = new wxButton( this, wxID_OK, _("OK"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel = new wxButton( this, wxID_CANCEL, _("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); m_buttonCancel->Hide(); bSizer69->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); @@ -861,7 +861,7 @@ CAddressBookDialogBase::CAddressBookDialogBase( wxWindow* parent, wxWindowID id, bSizer68->Add( 0, 5, 0, wxEXPAND, 5 ); - m_staticText55 = new wxStaticText( this, wxID_ANY, wxT("Bitcoin Address"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText55 = new wxStaticText( this, wxID_ANY, _("Bitcoin Address"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText55->Wrap( -1 ); m_staticText55->Hide(); @@ -876,19 +876,19 @@ CAddressBookDialogBase::CAddressBookDialogBase( wxWindow* parent, wxWindowID id, bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); - m_buttonEdit = new wxButton( this, wxID_BUTTONEDIT, wxT("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonEdit = new wxButton( this, wxID_BUTTONEDIT, _("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_buttonEdit, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonNew = new wxButton( this, wxID_BUTTONNEW, wxT(" &New Address... "), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonNew = new wxButton( this, wxID_BUTTONNEW, _(" &New Address... "), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_buttonNew, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonDelete = new wxButton( this, wxID_BUTTONDELETE, wxT("&Delete"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonDelete = new wxButton( this, wxID_BUTTONDELETE, _("&Delete"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_buttonDelete, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, _("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer69->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonCancel = new wxButton( this, wxID_CANCEL, _("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer69->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); @@ -922,713 +922,6 @@ CAddressBookDialogBase::~CAddressBookDialogBase() m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonCancel ), NULL, this ); } -CProductsDialogBase::CProductsDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) -{ - this->SetSizeHints( wxDefaultSize, wxDefaultSize ); - - wxBoxSizer* bSizer22; - bSizer22 = new wxBoxSizer( wxVERTICAL ); - - wxBoxSizer* bSizer23; - bSizer23 = new wxBoxSizer( wxHORIZONTAL ); - - m_comboBoxCategory = new wxComboBox( this, wxID_ANY, wxT("(Any Category)"), wxDefaultPosition, wxSize( 150,-1 ), 0, NULL, 0 ); - m_comboBoxCategory->Append( wxT("(Any Category)") ); - bSizer23->Add( m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlSearch = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - bSizer23->Add( m_textCtrlSearch, 1, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonSearch = new wxButton( this, wxID_ANY, wxT("&Search"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer23->Add( m_buttonSearch, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - bSizer22->Add( bSizer23, 0, wxEXPAND|wxTOP|wxBOTTOM|wxRIGHT, 5 ); - - m_listCtrl = new wxListCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT ); - bSizer22->Add( m_listCtrl, 1, wxALL|wxEXPAND, 5 ); - - this->SetSizer( bSizer22 ); - this->Layout(); - - // Connect Events - m_comboBoxCategory->Connect( wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler( CProductsDialogBase::OnCombobox ), NULL, this ); - m_textCtrlSearch->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CProductsDialogBase::OnKeyDown ), NULL, this ); - m_buttonSearch->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CProductsDialogBase::OnButtonSearch ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CProductsDialogBase::OnListItemActivated ), NULL, this ); -} - -CProductsDialogBase::~CProductsDialogBase() -{ - // Disconnect Events - m_comboBoxCategory->Disconnect( wxEVT_COMMAND_COMBOBOX_SELECTED, wxCommandEventHandler( CProductsDialogBase::OnCombobox ), NULL, this ); - m_textCtrlSearch->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CProductsDialogBase::OnKeyDown ), NULL, this ); - m_buttonSearch->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CProductsDialogBase::OnButtonSearch ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CProductsDialogBase::OnListItemActivated ), NULL, this ); -} - -CEditProductDialogBase::CEditProductDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) -{ - this->SetSizeHints( wxDefaultSize, wxDefaultSize ); - this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); - - wxBoxSizer* bSizer20; - bSizer20 = new wxBoxSizer( wxVERTICAL ); - - m_scrolledWindow = new wxScrolledWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL ); - m_scrolledWindow->SetScrollRate( 5, 5 ); - m_scrolledWindow->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW ) ); - - wxBoxSizer* bSizer21; - bSizer21 = new wxBoxSizer( wxVERTICAL ); - - wxFlexGridSizer* fgSizer8; - fgSizer8 = new wxFlexGridSizer( 0, 2, 0, 0 ); - fgSizer8->AddGrowableCol( 1 ); - fgSizer8->SetFlexibleDirection( wxBOTH ); - fgSizer8->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED ); - - m_staticText106 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Category"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT ); - m_staticText106->Wrap( -1 ); - fgSizer8->Add( m_staticText106, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); - - m_comboBoxCategory = new wxComboBox( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0, NULL, 0 ); - m_comboBoxCategory->SetMinSize( wxSize( 180,-1 ) ); - - fgSizer8->Add( m_comboBoxCategory, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_staticText108 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Title"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT ); - m_staticText108->Wrap( -1 ); - fgSizer8->Add( m_staticText108, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); - - m_textCtrlTitle = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - fgSizer8->Add( m_textCtrlTitle, 1, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - m_staticText107 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Price"), wxDefaultPosition, wxDefaultSize, wxALIGN_RIGHT ); - m_staticText107->Wrap( -1 ); - fgSizer8->Add( m_staticText107, 0, wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT, 5 ); - - m_textCtrlPrice = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlPrice->SetMinSize( wxSize( 105,-1 ) ); - - fgSizer8->Add( m_textCtrlPrice, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - bSizer21->Add( fgSizer8, 0, wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 5 ); - - m_staticText22 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Page 1: Description"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText22->Wrap( -1 ); - bSizer21->Add( m_staticText22, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlDescription = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE ); - m_textCtrlDescription->SetMinSize( wxSize( -1,170 ) ); - - bSizer21->Add( m_textCtrlDescription, 0, wxALL|wxEXPAND, 5 ); - - m_staticText23 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Page 2: Order Form"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText23->Wrap( -1 ); - bSizer21->Add( m_staticText23, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlInstructions = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE ); - m_textCtrlInstructions->SetMinSize( wxSize( -1,120 ) ); - - bSizer21->Add( m_textCtrlInstructions, 0, wxEXPAND|wxALL, 5 ); - - fgSizer5 = new wxFlexGridSizer( 0, 3, 0, 0 ); - fgSizer5->AddGrowableCol( 1 ); - fgSizer5->SetFlexibleDirection( wxBOTH ); - fgSizer5->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED ); - - m_staticText24 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Label"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText24->Wrap( -1 ); - fgSizer5->Add( m_staticText24, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5 ); - - m_staticText25 = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Comma separated list of choices, or leave blank for text field"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText25->Wrap( -1 ); - fgSizer5->Add( m_staticText25, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT, 5 ); - - - fgSizer5->Add( 0, 0, 1, wxEXPAND, 5 ); - - m_textCtrlLabel0 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel0->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel0, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField0 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField0, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel0 = new wxButton( m_scrolledWindow, wxID_DEL0, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel0, 0, wxRIGHT|wxLEFT|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlLabel1 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel1->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel1, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField1 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField1, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel1 = new wxButton( m_scrolledWindow, wxID_DEL1, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel1, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel2 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel2->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel2, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField2 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField2, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel2 = new wxButton( m_scrolledWindow, wxID_DEL2, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel2, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel3 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel3->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel3, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField3 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField3, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel3 = new wxButton( m_scrolledWindow, wxID_DEL3, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel3, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel4 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel4->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel4, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField4 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField4, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel4 = new wxButton( m_scrolledWindow, wxID_DEL4, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel4, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel5 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel5->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel5, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField5 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField5, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel5 = new wxButton( m_scrolledWindow, wxID_DEL5, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel5, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel6 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel6->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel6, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField6 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField6, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel6 = new wxButton( m_scrolledWindow, wxID_DEL6, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel6, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel7 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel7->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel7, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField7 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField7, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel7 = new wxButton( m_scrolledWindow, wxID_DEL7, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel7, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel8 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel8->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel8, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField8 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField8, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel8 = new wxButton( m_scrolledWindow, wxID_DEL8, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel8, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel9 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel9->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel9, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField9 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField9, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel9 = new wxButton( m_scrolledWindow, wxID_DEL9, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel9, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel10 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel10->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel10, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField10 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField10, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel10 = new wxButton( m_scrolledWindow, wxID_DEL10, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel10, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel11 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel11->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel11, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField11 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField11, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel11 = new wxButton( m_scrolledWindow, wxID_DEL11, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel11, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel12 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel12->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel12, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField12 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField12, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel12 = new wxButton( m_scrolledWindow, wxID_DEL12, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel12, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel13 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel13->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel13, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField13 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField13, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel13 = new wxButton( m_scrolledWindow, wxID_DEL13, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel13, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel14 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel14->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel14, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField14 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField14, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel14 = new wxButton( m_scrolledWindow, wxID_DEL14, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel14, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel15 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel15->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel15, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField15 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField15, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel15 = new wxButton( m_scrolledWindow, wxID_DEL15, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel15, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel16 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel16->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel16, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField16 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField16, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel16 = new wxButton( m_scrolledWindow, wxID_DEL16, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel16, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel17 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel17->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel17, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField17 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField17, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel17 = new wxButton( m_scrolledWindow, wxID_DEL17, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel17, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel18 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel18->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel18, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField18 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField18, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel18 = new wxButton( m_scrolledWindow, wxID_DEL18, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel18, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlLabel19 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_textCtrlLabel19->SetMinSize( wxSize( 150,-1 ) ); - - fgSizer5->Add( m_textCtrlLabel19, 0, wxALL|wxALIGN_CENTER_VERTICAL, 5 ); - - m_textCtrlField19 = new wxTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,-1 ), 0 ); - fgSizer5->Add( m_textCtrlField19, 0, wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL, 5 ); - - m_buttonDel19 = new wxButton( m_scrolledWindow, wxID_DEL19, wxT("Delete"), wxDefaultPosition, wxSize( 60,20 ), 0 ); - fgSizer5->Add( m_buttonDel19, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - - bSizer21->Add( fgSizer5, 0, wxEXPAND, 5 ); - - wxBoxSizer* bSizer25; - bSizer25 = new wxBoxSizer( wxHORIZONTAL ); - - m_buttonAddField = new wxButton( m_scrolledWindow, wxID_ANY, wxT("&Add Field"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer25->Add( m_buttonAddField, 0, wxALL, 5 ); - - bSizer21->Add( bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5 ); - - m_scrolledWindow->SetSizer( bSizer21 ); - m_scrolledWindow->Layout(); - bSizer21->Fit( m_scrolledWindow ); - bSizer20->Add( m_scrolledWindow, 1, wxEXPAND|wxALL, 5 ); - - wxBoxSizer* bSizer26; - bSizer26 = new wxBoxSizer( wxHORIZONTAL ); - - m_buttonOK = new wxButton( this, wxID_BUTTONSEND, wxT("&Send"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - m_buttonPreview = new wxButton( this, wxID_BUTTONPREVIEW, wxT("&Preview"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonPreview, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); - - this->SetSizer( bSizer20 ); - this->Layout(); - - // Connect Events - m_textCtrlTitle->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlPrice->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlDescription->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlInstructions->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlLabel0->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField0->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel0->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel0 ), NULL, this ); - m_textCtrlLabel1->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField1->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel1->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel1 ), NULL, this ); - m_textCtrlLabel2->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField2->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel2->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel2 ), NULL, this ); - m_textCtrlLabel3->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField3->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel3->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel3 ), NULL, this ); - m_textCtrlLabel4->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField4->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel4->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel4 ), NULL, this ); - m_textCtrlLabel5->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField5->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel5->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel5 ), NULL, this ); - m_textCtrlLabel6->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField6->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel6->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel6 ), NULL, this ); - m_textCtrlLabel7->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField7->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel7->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel7 ), NULL, this ); - m_textCtrlLabel8->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField8->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel8->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel8 ), NULL, this ); - m_textCtrlLabel9->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField9->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel9->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel9 ), NULL, this ); - m_textCtrlLabel10->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField10->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel10->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel10 ), NULL, this ); - m_textCtrlLabel11->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField11->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel11->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel11 ), NULL, this ); - m_textCtrlLabel12->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField12->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel12->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel12 ), NULL, this ); - m_textCtrlLabel13->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField13->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel13->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel13 ), NULL, this ); - m_textCtrlLabel14->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField14->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel14->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel14 ), NULL, this ); - m_textCtrlLabel15->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField15->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel15->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel15 ), NULL, this ); - m_textCtrlLabel16->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField16->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel16->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel16 ), NULL, this ); - m_textCtrlLabel17->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField17->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel17->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel17 ), NULL, this ); - m_textCtrlLabel18->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField18->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel18->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel18 ), NULL, this ); - m_textCtrlLabel19->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField19->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel19->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel19 ), NULL, this ); - m_buttonAddField->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonAddField ), NULL, this ); - m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonSend ), NULL, this ); - m_buttonPreview->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonPreview ), NULL, this ); - m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonCancel ), NULL, this ); -} - -CEditProductDialogBase::~CEditProductDialogBase() -{ - // Disconnect Events - m_textCtrlTitle->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlPrice->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlDescription->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlInstructions->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlLabel0->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField0->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel0->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel0 ), NULL, this ); - m_textCtrlLabel1->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField1->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel1->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel1 ), NULL, this ); - m_textCtrlLabel2->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField2->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel2->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel2 ), NULL, this ); - m_textCtrlLabel3->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField3->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel3->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel3 ), NULL, this ); - m_textCtrlLabel4->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField4->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel4->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel4 ), NULL, this ); - m_textCtrlLabel5->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField5->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel5->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel5 ), NULL, this ); - m_textCtrlLabel6->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField6->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel6->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel6 ), NULL, this ); - m_textCtrlLabel7->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField7->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel7->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel7 ), NULL, this ); - m_textCtrlLabel8->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField8->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel8->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel8 ), NULL, this ); - m_textCtrlLabel9->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField9->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel9->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel9 ), NULL, this ); - m_textCtrlLabel10->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField10->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel10->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel10 ), NULL, this ); - m_textCtrlLabel11->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField11->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel11->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel11 ), NULL, this ); - m_textCtrlLabel12->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField12->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel12->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel12 ), NULL, this ); - m_textCtrlLabel13->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField13->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel13->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel13 ), NULL, this ); - m_textCtrlLabel14->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField14->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel14->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel14 ), NULL, this ); - m_textCtrlLabel15->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField15->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel15->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel15 ), NULL, this ); - m_textCtrlLabel16->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField16->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel16->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel16 ), NULL, this ); - m_textCtrlLabel17->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField17->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel17->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel17 ), NULL, this ); - m_textCtrlLabel18->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField18->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel18->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel18 ), NULL, this ); - m_textCtrlLabel19->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_textCtrlField19->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditProductDialogBase::OnKeyDown ), NULL, this ); - m_buttonDel19->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonDel19 ), NULL, this ); - m_buttonAddField->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonAddField ), NULL, this ); - m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonSend ), NULL, this ); - m_buttonPreview->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonPreview ), NULL, this ); - m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditProductDialogBase::OnButtonCancel ), NULL, this ); -} - -CViewProductDialogBase::CViewProductDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) -{ - this->SetSizeHints( wxDefaultSize, wxDefaultSize ); - this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); - - wxBoxSizer* bSizer20; - bSizer20 = new wxBoxSizer( wxVERTICAL ); - - wxBoxSizer* bSizer116; - bSizer116 = new wxBoxSizer( wxHORIZONTAL ); - - m_htmlWinReviews = new wxHtmlWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO ); - m_htmlWinReviews->Hide(); - - bSizer116->Add( m_htmlWinReviews, 1, wxALL|wxEXPAND, 5 ); - - m_scrolledWindow = new wxScrolledWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL ); - m_scrolledWindow->SetScrollRate( 5, 5 ); - m_scrolledWindow->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW ) ); - - wxBoxSizer* bSizer21; - bSizer21 = new wxBoxSizer( wxVERTICAL ); - - m_richTextHeading = new wxRichTextCtrl( m_scrolledWindow, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize( -1,50 ), wxTE_READONLY|wxNO_BORDER ); - bSizer21->Add( m_richTextHeading, 0, wxEXPAND, 5 ); - - m_staticTextInstructions = new wxStaticText( m_scrolledWindow, wxID_ANY, wxT("Order Form instructions here\nmultiple lines\n1\n2\n3\n4\n5\n6"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticTextInstructions->Wrap( -1 ); - bSizer21->Add( m_staticTextInstructions, 0, wxALL|wxEXPAND, 5 ); - - wxBoxSizer* bSizer25; - bSizer25 = new wxBoxSizer( wxHORIZONTAL ); - - m_buttonSubmitForm = new wxButton( m_scrolledWindow, wxID_BUTTONSAMPLE, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer25->Add( m_buttonSubmitForm, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - m_buttonCancelForm = new wxButton( m_scrolledWindow, wxID_CANCEL2, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer25->Add( m_buttonCancelForm, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - bSizer21->Add( bSizer25, 0, wxALIGN_CENTER_HORIZONTAL, 5 ); - - m_scrolledWindow->SetSizer( bSizer21 ); - m_scrolledWindow->Layout(); - bSizer21->Fit( m_scrolledWindow ); - bSizer116->Add( m_scrolledWindow, 1, wxEXPAND|wxALL, 5 ); - - bSizer20->Add( bSizer116, 1, wxEXPAND, 5 ); - - wxBoxSizer* bSizer26; - bSizer26 = new wxBoxSizer( wxHORIZONTAL ); - - m_buttonBack = new wxButton( this, wxID_BUTTONBACK, wxT("< &Back "), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonBack->Enable( false ); - - bSizer26->Add( m_buttonBack, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - m_buttonNext = new wxButton( this, wxID_BUTTONNEXT, wxT(" &Next >"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonNext, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); - - this->SetSizer( bSizer20 ); - this->Layout(); - - // Connect Events - m_buttonSubmitForm->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonSubmitForm ), NULL, this ); - m_buttonCancelForm->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancelForm ), NULL, this ); - m_buttonBack->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonBack ), NULL, this ); - m_buttonNext->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonNext ), NULL, this ); - m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancel ), NULL, this ); -} - -CViewProductDialogBase::~CViewProductDialogBase() -{ - // Disconnect Events - m_buttonSubmitForm->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonSubmitForm ), NULL, this ); - m_buttonCancelForm->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancelForm ), NULL, this ); - m_buttonBack->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonBack ), NULL, this ); - m_buttonNext->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonNext ), NULL, this ); - m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewProductDialogBase::OnButtonCancel ), NULL, this ); -} - -CViewOrderDialogBase::CViewOrderDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) -{ - this->SetSizeHints( wxDefaultSize, wxDefaultSize ); - this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); - - wxBoxSizer* bSizer20; - bSizer20 = new wxBoxSizer( wxVERTICAL ); - - wxBoxSizer* bSizer116; - bSizer116 = new wxBoxSizer( wxHORIZONTAL ); - - m_htmlWin = new wxHtmlWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO ); - bSizer116->Add( m_htmlWin, 1, wxALL|wxEXPAND, 5 ); - - bSizer20->Add( bSizer116, 1, wxEXPAND, 5 ); - - wxBoxSizer* bSizer26; - bSizer26 = new wxBoxSizer( wxHORIZONTAL ); - - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer26->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - bSizer20->Add( bSizer26, 0, wxALIGN_RIGHT, 5 ); - - this->SetSizer( bSizer20 ); - this->Layout(); - - // Connect Events - m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewOrderDialogBase::OnButtonOK ), NULL, this ); -} - -CViewOrderDialogBase::~CViewOrderDialogBase() -{ - // Disconnect Events - m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CViewOrderDialogBase::OnButtonOK ), NULL, this ); -} - -CEditReviewDialogBase::CEditReviewDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style ) -{ - this->SetSizeHints( wxDefaultSize, wxDefaultSize ); - this->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); - - wxBoxSizer* bSizer112; - bSizer112 = new wxBoxSizer( wxVERTICAL ); - - - bSizer112->Add( 0, 3, 0, 0, 5 ); - - m_staticTextSeller = new wxStaticText( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0 ); - m_staticTextSeller->Wrap( -1 ); - bSizer112->Add( m_staticTextSeller, 0, wxALL|wxEXPAND, 5 ); - - - bSizer112->Add( 0, 3, 0, 0, 5 ); - - m_staticText110 = new wxStaticText( this, wxID_ANY, wxT("Rating"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText110->Wrap( -1 ); - bSizer112->Add( m_staticText110, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); - - wxString m_choiceStarsChoices[] = { wxT(" 1 star"), wxT(" 2 stars"), wxT(" 3 stars"), wxT(" 4 stars"), wxT(" 5 stars") }; - int m_choiceStarsNChoices = sizeof( m_choiceStarsChoices ) / sizeof( wxString ); - m_choiceStars = new wxChoice( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, m_choiceStarsNChoices, m_choiceStarsChoices, 0 ); - m_choiceStars->SetSelection( 0 ); - bSizer112->Add( m_choiceStars, 0, wxALL, 5 ); - - m_staticText43 = new wxStaticText( this, wxID_ANY, wxT("Review"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText43->Wrap( -1 ); - bSizer112->Add( m_staticText43, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); - - m_textCtrlReview = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE ); - bSizer112->Add( m_textCtrlReview, 1, wxALL|wxEXPAND, 5 ); - - wxBoxSizer* bSizer113; - bSizer113 = new wxBoxSizer( wxHORIZONTAL ); - - m_buttonSubmit = new wxButton( this, wxID_SUBMIT, wxT("&Submit"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer113->Add( m_buttonSubmit, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer113->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - - bSizer112->Add( bSizer113, 0, wxALIGN_RIGHT, 5 ); - - this->SetSizer( bSizer112 ); - this->Layout(); - - // Connect Events - m_textCtrlReview->Connect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditReviewDialogBase::OnKeyDown ), NULL, this ); - m_buttonSubmit->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonSubmit ), NULL, this ); - m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonCancel ), NULL, this ); -} - -CEditReviewDialogBase::~CEditReviewDialogBase() -{ - // Disconnect Events - m_textCtrlReview->Disconnect( wxEVT_KEY_DOWN, wxKeyEventHandler( CEditReviewDialogBase::OnKeyDown ), NULL, this ); - m_buttonSubmit->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonSubmit ), NULL, this ); - m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CEditReviewDialogBase::OnButtonCancel ), NULL, this ); -} - CGetTextFromUserDialogBase::CGetTextFromUserDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) { this->SetSizeHints( wxDefaultSize, wxDefaultSize ); @@ -1671,10 +964,10 @@ CGetTextFromUserDialogBase::CGetTextFromUserDialogBase( wxWindow* parent, wxWind bSizer80->Add( 0, 0, 1, wxEXPAND, 5 ); - m_buttonOK = new wxButton( this, wxID_OK, wxT("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + m_buttonOK = new wxButton( this, wxID_OK, _("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer80->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonCancel = new wxButton( this, wxID_CANCEL, wxT("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); + m_buttonCancel = new wxButton( this, wxID_CANCEL, _("Cancel"), wxDefaultPosition, wxDefaultSize, 0 ); bSizer80->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); bSizer79->Add( bSizer80, 0, wxEXPAND, 5 ); diff --git a/uibase.h b/uibase.h index faef9829..2dfc2af1 100644 --- a/uibase.h +++ b/uibase.h @@ -11,6 +11,8 @@ #ifndef __uibase__ #define __uibase__ +#include + #include #include #include @@ -37,8 +39,6 @@ #include #include #include -#include -#include /////////////////////////////////////////////////////////////////////////// @@ -49,8 +49,8 @@ #define wxID_BUTTONSEND 1004 #define wxID_BUTTONRECEIVE 1005 #define wxID_TEXTCTRLADDRESS 1006 -#define wxID_BUTTONCOPY 1007 -#define wxID_BUTTONCHANGE 1008 +#define wxID_BUTTONCHANGE 1007 +#define wxID_BUTTONCOPY 1008 #define wxID_TRANSACTIONFEE 1009 #define wxID_PROXYIP 1010 #define wxID_PROXYPORT 1011 @@ -64,33 +64,7 @@ #define wxID_BUTTONNEW 1019 #define wxID_BUTTONEDIT 1020 #define wxID_BUTTONDELETE 1021 -#define wxID_DEL0 1022 -#define wxID_DEL1 1023 -#define wxID_DEL2 1024 -#define wxID_DEL3 1025 -#define wxID_DEL4 1026 -#define wxID_DEL5 1027 -#define wxID_DEL6 1028 -#define wxID_DEL7 1029 -#define wxID_DEL8 1030 -#define wxID_DEL9 1031 -#define wxID_DEL10 1032 -#define wxID_DEL11 1033 -#define wxID_DEL12 1034 -#define wxID_DEL13 1035 -#define wxID_DEL14 1036 -#define wxID_DEL15 1037 -#define wxID_DEL16 1038 -#define wxID_DEL17 1039 -#define wxID_DEL18 1040 -#define wxID_DEL19 1041 -#define wxID_BUTTONPREVIEW 1042 -#define wxID_BUTTONSAMPLE 1043 -#define wxID_CANCEL2 1044 -#define wxID_BUTTONBACK 1045 -#define wxID_BUTTONNEXT 1046 -#define wxID_SUBMIT 1047 -#define wxID_TEXTCTRL 1048 +#define wxID_TEXTCTRL 1022 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase @@ -109,8 +83,8 @@ class CMainFrameBase : public wxFrame wxStaticText* m_staticText32; wxTextCtrl* m_textCtrlAddress; + wxButton* m_buttonNew; wxButton* m_buttonCopy; - wxButton* m_button91; wxPanel* m_panel14; wxStaticText* m_staticText41; @@ -137,8 +111,8 @@ class CMainFrameBase : public wxFrame virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } virtual void OnMouseEventsAddress( wxMouseEvent& event ){ event.Skip(); } virtual void OnSetFocusAddress( wxFocusEvent& event ){ event.Skip(); } - virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonChange( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } virtual void OnListColBeginDrag( wxListEvent& event ){ event.Skip(); } virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } virtual void OnPaintListCtrl( wxPaintEvent& event ){ event.Skip(); } @@ -147,7 +121,7 @@ class CMainFrameBase : public wxFrame public: wxMenu* m_menuOptions; wxListCtrl* m_listCtrl; - CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = wxT("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 712,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = _("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 712,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); ~CMainFrameBase(); }; @@ -168,7 +142,7 @@ class CTxDetailsDialogBase : public wxDialog public: - CTxDetailsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Transaction Details"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 620,450 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + CTxDetailsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("Transaction Details"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 620,450 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); ~CTxDetailsDialogBase(); }; @@ -223,7 +197,7 @@ class COptionsDialogBase : public wxDialog public: - COptionsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Options"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 540,360 ), long style = wxDEFAULT_DIALOG_STYLE ); + COptionsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("Options"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 540,360 ), long style = wxDEFAULT_DIALOG_STYLE ); ~COptionsDialogBase(); }; @@ -251,7 +225,7 @@ class CAboutDialogBase : public wxDialog public: wxStaticText* m_staticTextVersion; - CAboutDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("About Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 532,329 ), long style = wxDEFAULT_DIALOG_STYLE ); + CAboutDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("About Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 532,329 ), long style = wxDEFAULT_DIALOG_STYLE ); ~CAboutDialogBase(); }; @@ -298,7 +272,7 @@ class CSendDialogBase : public wxDialog public: - CSendDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Send Coins"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 675,312 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + CSendDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("Send Coins"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 675,312 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); ~CSendDialogBase(); }; @@ -325,7 +299,7 @@ class CSendingDialogBase : public wxDialog public: - CSendingDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Sending..."), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 442,151 ), long style = wxDEFAULT_DIALOG_STYLE ); + CSendingDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("Sending..."), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 442,151 ), long style = wxDEFAULT_DIALOG_STYLE ); ~CSendingDialogBase(); }; @@ -361,7 +335,7 @@ class CYourAddressDialogBase : public wxDialog public: - CYourAddressDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Your Bitcoin Addresses"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 610,390 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + CYourAddressDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("Your Bitcoin Addresses"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 610,390 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); ~CYourAddressDialogBase(); }; @@ -397,242 +371,11 @@ class CAddressBookDialogBase : public wxDialog public: wxButton* m_buttonCancel; - CAddressBookDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Address Book"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 610,390 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); + CAddressBookDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("Address Book"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 610,390 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); ~CAddressBookDialogBase(); }; -/////////////////////////////////////////////////////////////////////////////// -/// Class CProductsDialogBase -/////////////////////////////////////////////////////////////////////////////// -class CProductsDialogBase : public wxDialog -{ - private: - - protected: - wxComboBox* m_comboBoxCategory; - wxTextCtrl* m_textCtrlSearch; - wxButton* m_buttonSearch; - wxListCtrl* m_listCtrl; - - // Virtual event handlers, overide them in your derived class - virtual void OnCombobox( wxCommandEvent& event ){ event.Skip(); } - virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } - virtual void OnButtonSearch( wxCommandEvent& event ){ event.Skip(); } - virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } - - - public: - CProductsDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Marketplace"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 708,535 ), long style = wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER ); - ~CProductsDialogBase(); - -}; - -/////////////////////////////////////////////////////////////////////////////// -/// Class CEditProductDialogBase -/////////////////////////////////////////////////////////////////////////////// -class CEditProductDialogBase : public wxFrame -{ - private: - - protected: - wxScrolledWindow* m_scrolledWindow; - wxStaticText* m_staticText106; - wxComboBox* m_comboBoxCategory; - wxStaticText* m_staticText108; - wxTextCtrl* m_textCtrlTitle; - wxStaticText* m_staticText107; - wxTextCtrl* m_textCtrlPrice; - wxStaticText* m_staticText22; - wxTextCtrl* m_textCtrlDescription; - wxStaticText* m_staticText23; - wxTextCtrl* m_textCtrlInstructions; - wxStaticText* m_staticText24; - wxStaticText* m_staticText25; - - wxTextCtrl* m_textCtrlLabel0; - wxTextCtrl* m_textCtrlField0; - wxButton* m_buttonDel0; - wxTextCtrl* m_textCtrlLabel1; - wxTextCtrl* m_textCtrlField1; - wxButton* m_buttonDel1; - wxTextCtrl* m_textCtrlLabel2; - wxTextCtrl* m_textCtrlField2; - wxButton* m_buttonDel2; - wxTextCtrl* m_textCtrlLabel3; - wxTextCtrl* m_textCtrlField3; - wxButton* m_buttonDel3; - wxTextCtrl* m_textCtrlLabel4; - wxTextCtrl* m_textCtrlField4; - wxButton* m_buttonDel4; - wxTextCtrl* m_textCtrlLabel5; - wxTextCtrl* m_textCtrlField5; - wxButton* m_buttonDel5; - wxTextCtrl* m_textCtrlLabel6; - wxTextCtrl* m_textCtrlField6; - wxButton* m_buttonDel6; - wxTextCtrl* m_textCtrlLabel7; - wxTextCtrl* m_textCtrlField7; - wxButton* m_buttonDel7; - wxTextCtrl* m_textCtrlLabel8; - wxTextCtrl* m_textCtrlField8; - wxButton* m_buttonDel8; - wxTextCtrl* m_textCtrlLabel9; - wxTextCtrl* m_textCtrlField9; - wxButton* m_buttonDel9; - wxTextCtrl* m_textCtrlLabel10; - wxTextCtrl* m_textCtrlField10; - wxButton* m_buttonDel10; - wxTextCtrl* m_textCtrlLabel11; - wxTextCtrl* m_textCtrlField11; - wxButton* m_buttonDel11; - wxTextCtrl* m_textCtrlLabel12; - wxTextCtrl* m_textCtrlField12; - wxButton* m_buttonDel12; - wxTextCtrl* m_textCtrlLabel13; - wxTextCtrl* m_textCtrlField13; - wxButton* m_buttonDel13; - wxTextCtrl* m_textCtrlLabel14; - wxTextCtrl* m_textCtrlField14; - wxButton* m_buttonDel14; - wxTextCtrl* m_textCtrlLabel15; - wxTextCtrl* m_textCtrlField15; - wxButton* m_buttonDel15; - wxTextCtrl* m_textCtrlLabel16; - wxTextCtrl* m_textCtrlField16; - wxButton* m_buttonDel16; - wxTextCtrl* m_textCtrlLabel17; - wxTextCtrl* m_textCtrlField17; - wxButton* m_buttonDel17; - wxTextCtrl* m_textCtrlLabel18; - wxTextCtrl* m_textCtrlField18; - wxButton* m_buttonDel18; - wxTextCtrl* m_textCtrlLabel19; - wxTextCtrl* m_textCtrlField19; - wxButton* m_buttonDel19; - wxButton* m_buttonAddField; - wxButton* m_buttonOK; - wxButton* m_buttonPreview; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } - virtual void OnButtonDel0( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel1( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel2( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel3( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel4( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel5( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel6( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel7( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel8( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel9( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel10( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel11( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel12( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel13( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel14( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel15( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel16( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel17( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel18( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDel19( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonAddField( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonSend( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonPreview( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } - - - public: - wxFlexGridSizer* fgSizer5; - CEditProductDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Edit Product"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 660,640 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); - ~CEditProductDialogBase(); - -}; - -/////////////////////////////////////////////////////////////////////////////// -/// Class CViewProductDialogBase -/////////////////////////////////////////////////////////////////////////////// -class CViewProductDialogBase : public wxFrame -{ - private: - - protected: - wxHtmlWindow* m_htmlWinReviews; - wxScrolledWindow* m_scrolledWindow; - wxRichTextCtrl* m_richTextHeading; - wxStaticText* m_staticTextInstructions; - wxButton* m_buttonSubmitForm; - wxButton* m_buttonCancelForm; - wxButton* m_buttonBack; - wxButton* m_buttonNext; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnButtonSubmitForm( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonCancelForm( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonBack( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonNext( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } - - - public: - CViewProductDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Order Form"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 630,520 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); - ~CViewProductDialogBase(); - -}; - -/////////////////////////////////////////////////////////////////////////////// -/// Class CViewOrderDialogBase -/////////////////////////////////////////////////////////////////////////////// -class CViewOrderDialogBase : public wxFrame -{ - private: - - protected: - wxHtmlWindow* m_htmlWin; - wxButton* m_buttonOK; - - // Virtual event handlers, overide them in your derived class - virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } - - - public: - CViewOrderDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("View Order"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 630,520 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); - ~CViewOrderDialogBase(); - -}; - -/////////////////////////////////////////////////////////////////////////////// -/// Class CEditReviewDialogBase -/////////////////////////////////////////////////////////////////////////////// -class CEditReviewDialogBase : public wxFrame -{ - private: - - protected: - - wxStaticText* m_staticTextSeller; - - wxStaticText* m_staticText110; - wxChoice* m_choiceStars; - wxStaticText* m_staticText43; - wxTextCtrl* m_textCtrlReview; - wxButton* m_buttonSubmit; - wxButton* m_buttonCancel; - - // Virtual event handlers, overide them in your derived class - virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } - virtual void OnButtonSubmit( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } - - - public: - CEditReviewDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxT("Enter Review"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 630,440 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); - ~CEditReviewDialogBase(); - -}; - /////////////////////////////////////////////////////////////////////////////// /// Class CGetTextFromUserDialogBase /////////////////////////////////////////////////////////////////////////////// diff --git a/uiproject.fbp b/uiproject.fbp index c24f6962..5c6eaa55 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -10,7 +10,7 @@ uibase 1000 none - 0 + 1 . @@ -18,7 +18,7 @@ 1 0 0 - + wxSYS_COLOUR_BTNFACE @@ -478,7 +478,7 @@ 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT + wxRIGHT 0 @@ -487,22 +487,22 @@ 1 - 0 - wxID_BUTTONCOPY - &Copy to Clipboard + 1 + wxID_BUTTONCHANGE + &New... - m_buttonCopy + m_buttonNew protected - wxBU_EXACTFIT + - OnButtonCopy + OnButtonChange @@ -530,7 +530,7 @@ 5 - wxRIGHT + wxALIGN_CENTER_VERTICAL|wxRIGHT 0 @@ -539,22 +539,22 @@ 1 - 1 - wxID_BUTTONCHANGE - C&hange... + 0 + wxID_BUTTONCOPY + &Copy to Clipboard - m_button91 + m_buttonCopy protected - + wxBU_EXACTFIT - OnButtonChange + OnButtonCopy @@ -5239,5779 +5239,6 @@ - - - - - 1 - - - - 0 - wxID_ANY - - - CProductsDialogBase - - 708,535 - wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER - - Marketplace - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer22 - wxVERTICAL - none - - 5 - wxEXPAND|wxTOP|wxBOTTOM|wxRIGHT - 0 - - - bSizer23 - wxHORIZONTAL - none - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - "(Any Category)" - - 1 - - - 0 - wxID_ANY - - - m_comboBoxCategory - protected - - 150,-1 - - - - (Any Category) - - - - - OnCombobox - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 1 - - - - 1 - - - 0 - wxID_ANY - - 0 - - m_textCtrlSearch - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 0 - 1 - - - 0 - wxID_ANY - &Search - - - m_buttonSearch - protected - - - - - - - - - OnButtonSearch - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_listCtrl - protected - - - wxLC_NO_SORT_HEADER|wxLC_REPORT - - - - - - - - - - - - - - - - - - - - - - - - - - - - - OnListItemActivated - - - - - - - - - - - - - - - - - - - - - - - - - wxSYS_COLOUR_MENU - - - 1 - - - - 0 - wxID_ANY - - - CEditProductDialogBase - - 660,640 - wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER - - Edit Product - - - - wxTAB_TRAVERSAL - 1 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer20 - wxVERTICAL - none - - 5 - wxEXPAND|wxALL - 1 - - wxSYS_COLOUR_WINDOW - - 1 - - - 0 - wxID_ANY - - - m_scrolledWindow - protected - - 5 - 5 - - - - - - wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer21 - wxVERTICAL - none - - 5 - wxEXPAND|wxTOP|wxRIGHT|wxLEFT - 0 - - 2 - wxBOTH - 1 - - 0 - - fgSizer8 - wxFLEX_GROWMODE_SPECIFIED - none - 0 - 0 - - 5 - wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Category - - - m_staticText106 - protected - - - wxALIGN_RIGHT - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - - 1 - - - 0 - wxID_ANY - - 180,-1 - m_comboBoxCategory - protected - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Title - - - m_staticText108 - protected - - - wxALIGN_RIGHT - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - 0 - - m_textCtrlTitle - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxALIGN_RIGHT|wxTOP|wxBOTTOM|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Price - - - m_staticText107 - protected - - - wxALIGN_RIGHT - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 105,-1 - m_textCtrlPrice - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxTOP|wxRIGHT|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Page 1: Description - - - m_staticText22 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - -1,170 - m_textCtrlDescription - protected - - - wxTE_MULTILINE - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxTOP|wxRIGHT|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Page 2: Order Form - - - m_staticText23 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxEXPAND|wxALL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - -1,120 - m_textCtrlInstructions - protected - - - wxTE_MULTILINE - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxEXPAND - 0 - - 3 - wxBOTH - 1 - - 0 - - fgSizer5 - wxFLEX_GROWMODE_SPECIFIED - public - 0 - 0 - - 5 - wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Label - - - m_staticText24 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxTOP|wxRIGHT|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Comma separated list of choices, or leave blank for text field - - - m_staticText25 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxEXPAND - 1 - - 0 - protected - 0 - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel0 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField0 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxRIGHT|wxLEFT|wxALIGN_CENTER_VERTICAL - 0 - - - - 0 - 1 - - - 0 - wxID_DEL0 - Delete - - -1,-1 - m_buttonDel0 - protected - - 60,20 - - - - - - - OnButtonDel0 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel1 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField1 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL1 - Delete - - -1,-1 - m_buttonDel1 - protected - - 60,20 - - - - - - - OnButtonDel1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel2 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField2 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL2 - Delete - - -1,-1 - m_buttonDel2 - protected - - 60,20 - - - - - - - OnButtonDel2 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel3 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField3 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL3 - Delete - - -1,-1 - m_buttonDel3 - protected - - 60,20 - - - - - - - OnButtonDel3 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel4 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField4 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL4 - Delete - - -1,-1 - m_buttonDel4 - protected - - 60,20 - - - - - - - OnButtonDel4 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel5 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField5 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL5 - Delete - - -1,-1 - m_buttonDel5 - protected - - 60,20 - - - - - - - OnButtonDel5 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel6 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField6 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL6 - Delete - - -1,-1 - m_buttonDel6 - protected - - 60,20 - - - - - - - OnButtonDel6 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel7 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField7 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL7 - Delete - - -1,-1 - m_buttonDel7 - protected - - 60,20 - - - - - - - OnButtonDel7 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel8 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField8 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL8 - Delete - - -1,-1 - m_buttonDel8 - protected - - 60,20 - - - - - - - OnButtonDel8 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel9 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField9 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL9 - Delete - - -1,-1 - m_buttonDel9 - protected - - 60,20 - - - - - - - OnButtonDel9 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel10 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField10 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL10 - Delete - - -1,-1 - m_buttonDel10 - protected - - 60,20 - - - - - - - OnButtonDel10 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel11 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField11 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL11 - Delete - - -1,-1 - m_buttonDel11 - protected - - 60,20 - - - - - - - OnButtonDel11 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel12 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField12 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL12 - Delete - - -1,-1 - m_buttonDel12 - protected - - 60,20 - - - - - - - OnButtonDel12 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel13 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField13 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL13 - Delete - - -1,-1 - m_buttonDel13 - protected - - 60,20 - - - - - - - OnButtonDel13 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel14 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField14 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL14 - Delete - - -1,-1 - m_buttonDel14 - protected - - 60,20 - - - - - - - OnButtonDel14 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel15 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField15 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL15 - Delete - - -1,-1 - m_buttonDel15 - protected - - 60,20 - - - - - - - OnButtonDel15 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel16 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField16 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL16 - Delete - - -1,-1 - m_buttonDel16 - protected - - 60,20 - - - - - - - OnButtonDel16 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel17 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField17 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL17 - Delete - - -1,-1 - m_buttonDel17 - protected - - 60,20 - - - - - - - OnButtonDel17 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel18 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField18 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL18 - Delete - - -1,-1 - m_buttonDel18 - protected - - 60,20 - - - - - - - OnButtonDel18 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - - 0 - 150,-1 - m_textCtrlLabel19 - protected - - - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND|wxALIGN_CENTER_VERTICAL - 0 - - - - 1 - - - 0 - wxID_ANY - -1,-1 - 0 - -1,-1 - m_textCtrlField19 - protected - - -1,-1 - - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT - 0 - - - - 0 - 1 - - - 0 - wxID_DEL19 - Delete - - -1,-1 - m_buttonDel19 - protected - - 60,20 - - - - - - - OnButtonDel19 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_HORIZONTAL - 0 - - - bSizer25 - wxHORIZONTAL - none - - 5 - wxALL - 0 - - - - 0 - 1 - - - 0 - wxID_ANY - &Add Field - - - m_buttonAddField - protected - - - - - - - - - OnButtonAddField - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_RIGHT - 0 - - - bSizer26 - wxHORIZONTAL - none - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_BUTTONSEND - &Send - - -1,-1 - m_buttonOK - protected - - - - - - - - - OnButtonSend - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_BUTTONPREVIEW - &Preview - - -1,-1 - m_buttonPreview - protected - - - - - - - - - OnButtonPreview - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_CANCEL - Cancel - - -1,-1 - m_buttonCancel - protected - - - - - - - - - OnButtonCancel - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - wxSYS_COLOUR_MENU - - - 1 - - - - 0 - wxID_ANY - - - CViewProductDialogBase - - 630,520 - wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER - - Order Form - - - - wxTAB_TRAVERSAL - 1 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer20 - wxVERTICAL - none - - 5 - wxEXPAND - 1 - - - bSizer116 - wxHORIZONTAL - none - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 1 - wxID_ANY - - - m_htmlWinReviews - protected - - - wxHW_SCROLLBAR_AUTO - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxEXPAND|wxALL - 1 - - wxSYS_COLOUR_WINDOW - - 1 - - - 0 - wxID_ANY - - - m_scrolledWindow - protected - - 5 - 5 - - - - - - wxHSCROLL|wxTAB_TRAVERSAL|wxVSCROLL - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer21 - wxVERTICAL - none - - 5 - wxEXPAND - 0 - - - - 1 - - - 0 - wxID_ANY - - -1,-1 - m_richTextHeading - protected - - -1,50 - wxTE_READONLY - - - - - wxNO_BORDER - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND - 0 - - - - 1 - - - 0 - wxID_ANY - Order Form instructions here multiple lines 1 2 3 4 5 6 - - - m_staticTextInstructions - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_CENTER_HORIZONTAL - 0 - - - bSizer25 - wxHORIZONTAL - none - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_BUTTONSAMPLE - &Submit - - -1,-1 - m_buttonSubmitForm - protected - - - - - - - - - OnButtonSubmitForm - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_CANCEL2 - Cancel - - - m_buttonCancelForm - protected - - - - - - - - - OnButtonCancelForm - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_RIGHT - 0 - - - bSizer26 - wxHORIZONTAL - none - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 0 - - - 0 - wxID_BUTTONBACK - < &Back - - -1,-1 - m_buttonBack - protected - - - - - - - - - OnButtonBack - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_BUTTONNEXT - &Next > - - -1,-1 - m_buttonNext - protected - - - - - - - - - OnButtonNext - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_CANCEL - Cancel - - -1,-1 - m_buttonCancel - protected - - - - - - - - - OnButtonCancel - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - wxSYS_COLOUR_MENU - - - 1 - - - - 0 - wxID_ANY - - - CViewOrderDialogBase - - 630,520 - wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER - - View Order - - - - wxTAB_TRAVERSAL - 1 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer20 - wxVERTICAL - none - - 5 - wxEXPAND - 1 - - - bSizer116 - wxHORIZONTAL - none - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - - m_htmlWin - protected - - - wxHW_SCROLLBAR_AUTO - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_RIGHT - 0 - - - bSizer26 - wxHORIZONTAL - none - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_OK - OK - - -1,-1 - m_buttonOK - protected - - - - - - - - - OnButtonOK - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - wxSYS_COLOUR_MENU - - - 1 - - - - 0 - wxID_ANY - - - CEditReviewDialogBase - - 630,440 - wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER - - Enter Review - - - - wxTAB_TRAVERSAL - 1 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bSizer112 - wxVERTICAL - none - - 5 - - 0 - - 3 - protected - 0 - - - - 5 - wxALL|wxEXPAND - 0 - - - - 1 - - - 0 - wxID_ANY - - - - m_staticTextSeller - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - - 0 - - 3 - protected - 0 - - - - 5 - wxTOP|wxRIGHT|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Rating - - - m_staticText110 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - " 1 star" " 2 stars" " 3 stars" " 4 stars" " 5 stars" - - 1 - - - 0 - wxID_ANY - - - m_choiceStars - protected - - 0 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxTOP|wxRIGHT|wxLEFT - 0 - - - - 1 - - - 0 - wxID_ANY - Review - - - m_staticText43 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND - 1 - - - - 1 - - - 0 - wxID_ANY - - 0 - - m_textCtrlReview - protected - - - wxTE_MULTILINE - - - - - - - - - - OnKeyDown - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALIGN_RIGHT - 0 - - - bSizer113 - wxHORIZONTAL - none - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_SUBMIT - &Submit - - -1,-1 - m_buttonSubmit - protected - - - - - - - - - OnButtonSubmit - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND - 0 - - - - 0 - 1 - - - 0 - wxID_CANCEL - Cancel - - -1,-1 - m_buttonCancel - protected - - - - - - - - - OnButtonCancel - - - - - - - - - - - - - - - - - - - - - - - - - - - - - From 2b63e68bbfebc370aa6f2d542419a6e5680f33c5 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Mon, 15 Feb 2010 20:36:55 +0000 Subject: [PATCH 053/133] fix mingw makefile to use correct wxWidgets include, this is the last version of makefile and makefile.vc for wxWidgets-2.8 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@65 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- makefile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/makefile b/makefile index fb8dd274..71f37449 100644 --- a/makefile +++ b/makefile @@ -15,7 +15,7 @@ endif -INCLUDEPATHS=-I"/boost" -I"/db/build_unix" -I"/openssl/include" -I"/wxwidgets/lib/vc_lib/mswd" -I"/wxwidgets/include" +INCLUDEPATHS=-I"/boost" -I"/db/build_unix" -I"/openssl/include" -I"/wxwidgets/lib/gcc_lib/mswd" -I"/wxwidgets/include" LIBPATHS=-L"/boost/stage/lib" -L"/db/build_unix" -L"/openssl/out" -L"/wxwidgets/lib/gcc_lib" LIBS= \ -l libboost_system-mgw34-mt-d -l libboost_filesystem-mgw34-mt-d \ From e4806597650d8a17f1da7a5b5501eb367ea786fc Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 17 Feb 2010 17:22:01 +0000 Subject: [PATCH 054/133] all builds are now with wxWidgets-2.9.0, we are now using UTF-8, added support for language translation file locale//LC_MESSAGES/bitcoin.mo -- version 0.2.2 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@66 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-msw.txt | 24 +- build-unix.txt | 22 +- locale/readme.txt | 5 + main.cpp | 30 +-- makefile => makefile.mingw | 27 ++- makefile.unix.wx2.9 => makefile.unix | 14 +- makefile.unix.wx2.8 | 88 ------- makefile.vc | 35 +-- serialize.h | 4 +- ui.cpp | 337 +++++++++++++++------------ ui.h | 12 +- uibase.cpp | 17 +- uibase.h | 5 - uiproject.fbp | 108 +-------- util.cpp | 6 +- util.h | 9 +- 16 files changed, 281 insertions(+), 462 deletions(-) create mode 100644 locale/readme.txt rename makefile => makefile.mingw (78%) rename makefile.unix.wx2.9 => makefile.unix (89%) delete mode 100644 makefile.unix.wx2.8 diff --git a/build-msw.txt b/build-msw.txt index 9786255e..adb526a3 100644 --- a/build-msw.txt +++ b/build-msw.txt @@ -29,11 +29,11 @@ Dependencies ------------ Libraries you need to download separately and build: - default path download -wxWidgets \wxwidgets prebuilt: http://wxpack.sourceforge.net -OpenSSL \openssl http://www.openssl.org/source/ -Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html -Boost \boost http://www.boost.org/users/download/ + default path download +wxWidgets-2.9 \wxwidgets http://www.wxwidgets.org/downloads/ +OpenSSL \openssl http://www.openssl.org/source/ +Berkeley DB \db http://www.oracle.com/technology/software/products/berkeley-db/index.html +Boost \boost http://www.boost.org/users/download/ Their licenses: wxWidgets LGPL 2.1 with very liberal exceptions @@ -43,10 +43,10 @@ Boost MIT-like license Versions used in this release: MinGW GCC 3.4.5 -wxWidgets 2.8.9 +wxWidgets 2.9.0 OpenSSL 0.9.8k Berkeley DB 4.7.25.NC -Boost 1.34.1 +Boost 1.42.1 Notes @@ -59,6 +59,14 @@ The release is built with GCC and then "strip bitcoin.exe" to strip the debug symbols, which reduces the executable size by about 90%. +wxWidgets +--------- +cd \wxwidgets\build\msw +make -f makefile.gcc + or +nmake -f makefile.vc + + OpenSSL ------- Bitcoin does not use any encryption. If you want to do a no-everything @@ -104,5 +112,5 @@ download bjam.exe from http://sourceforge.net/project/showfiles.php?group_id=7586&package_id=72941 cd \boost bjam toolset=gcc --build-type=complete stage -or + or bjam toolset=msvc --build-type=complete stage diff --git a/build-unix.txt b/build-unix.txt index f1a72eaa..5e3f574a 100644 --- a/build-unix.txt +++ b/build-unix.txt @@ -20,15 +20,11 @@ sudo apt-get install libdb4.7-dev sudo apt-get install libdb4.7++-dev sudo apt-get install libboost-dev -The release was built with wxWidgets 2.8.9 ansi on 32-bit. The current -sourcecode can be built on 64-bit with wxWidgets 2.9.0. +We're now using wxWidgets 2.9, which uses UTF-8. -There is currently no libwxgtk2.8-ansi-dev debian package for Karmic. -libwxgtk2.8-dev is the "unicode" build, but for wxWidgets 2.8 "unicode" -means wchar, not UTF-8. wchar wxString doesn't convert to std::string. -We haven't been able to compile the 2.8 versions on 64-bit. - -wxWidgets 2.9 is UTF-8 and compiles on 64-bit. +There isn't currently a debian package of wxWidgets we can use. The 2.8 +packages for Karmic are UTF-16 unicode and won't work for us, and we've had +trouble building 2.8 on 64-bit. You need to download wxWidgets from http://www.wxwidgets.org/downloads/ and build it yourself. See the build instructions and configure parameters @@ -42,9 +38,9 @@ Boost MIT-like license Versions used in this release: GCC 4.3.3 OpenSSL 0.9.8k -wxWidgets 2.8.9 +wxWidgets 2.9.0 Berkeley DB 4.7.25.NC -Boost 1.40.0 +Boost 1.38.0 Notes @@ -59,7 +55,9 @@ symbols, which reduces the executable size by about 90%. wxWidgets --------- -cd /usr/local/wxWidgets-2.8.9 or 2.9.0 +cd /usr/local +tar -xzvf wxWidgets-2.9.0.tar.gz +cd /usr/local/wxWidgets-2.9.0 mkdir buildgtk cd buildgtk ../configure --with-gtk --enable-debug --disable-shared --enable-monolithic @@ -72,7 +70,7 @@ ldconfig Boost ----- If you want to build Boost yourself, -cd /usr/local/boost_1_40_0 +cd /usr/local/boost_1_42_0 su ./bootstrap.sh ./bjam install diff --git a/locale/readme.txt b/locale/readme.txt new file mode 100644 index 00000000..9fca3ce6 --- /dev/null +++ b/locale/readme.txt @@ -0,0 +1,5 @@ +put bitcoin.po and bitcoin.mo files at: +locale//LC_MESSAGES/bitcoin.mo and .po + +.po is the sourcefile +.mo is the compiled translation diff --git a/main.cpp b/main.cpp index 3391a299..665a78e2 100644 --- a/main.cpp +++ b/main.cpp @@ -1460,27 +1460,13 @@ bool ScanMessageStart(Stream& s) bool CheckDiskSpace(int64 nAdditionalBytes) { -#ifdef __WXMSW__ - uint64 nFreeBytesAvailable = 0; // bytes available to caller - uint64 nTotalNumberOfBytes = 0; // bytes on disk - uint64 nTotalNumberOfFreeBytes = 0; // free bytes on disk - if (!GetDiskFreeSpaceEx(GetDataDir().c_str(), - (PULARGE_INTEGER)&nFreeBytesAvailable, - (PULARGE_INTEGER)&nTotalNumberOfBytes, - (PULARGE_INTEGER)&nTotalNumberOfFreeBytes)) - { - printf("ERROR: GetDiskFreeSpaceEx() failed\n"); - return true; - } -#else uint64 nFreeBytesAvailable = filesystem::space(GetDataDir()).available; -#endif // Check for 15MB because database could create another 10MB log file at any time if (nFreeBytesAvailable < (int64)15000000 + nAdditionalBytes) { fShutdown = true; - ThreadSafeMessageBox("Warning: Your disk space is low ", "Bitcoin", wxOK | wxICON_EXCLAMATION); + ThreadSafeMessageBox(_("Warning: Disk space is low "), "Bitcoin", wxOK | wxICON_EXCLAMATION); CreateThread(Shutdown, NULL); return false; } @@ -2962,16 +2948,16 @@ string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) { string strError; if (nValue + nFeeRequired > GetBalance()) - strError = strprintf("Error: This is an oversized transaction that requires a transaction fee of %s ", FormatMoney(nFeeRequired).c_str()); + strError = strprintf(_("Error: This is an oversized transaction that requires a transaction fee of %s "), FormatMoney(nFeeRequired).c_str()); else - strError = "Error: Transaction creation failed "; + strError = _("Error: Transaction creation failed "); printf("SendMoney() : %s", strError.c_str()); return strError; } if (!CommitTransactionSpent(wtxNew, key)) { printf("SendMoney() : Error finalizing transaction"); - return "Error finalizing transaction"; + return _("Error finalizing transaction"); } // Track how many getdata requests our transaction gets @@ -2985,7 +2971,7 @@ string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) { // This must not fail. The transaction has already been signed and recorded. printf("SendMoney() : Error: Transaction not valid"); - return "Error: The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here."; + return _("Error: The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here."); } wtxNew.RelayWalletTransaction(); } @@ -2999,14 +2985,14 @@ string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtx { // Check amount if (nValue <= 0) - return "Invalid amount"; + return _("Invalid amount"); if (nValue + nTransactionFee > GetBalance()) - return "You don't have enough money"; + return _("You don't have enough money"); // Parse bitcoin address CScript scriptPubKey; if (!scriptPubKey.SetBitcoinAddress(strAddress)) - return "Invalid bitcoin address"; + return _("Invalid bitcoin address"); return SendMoney(scriptPubKey, nValue, wtxNew); } diff --git a/makefile b/makefile.mingw similarity index 78% rename from makefile rename to makefile.mingw index 71f37449..2644cbaa 100644 --- a/makefile +++ b/makefile.mingw @@ -3,27 +3,30 @@ # file license.txt or http://www.opensource.org/licenses/mit-license.php. -ifneq "$(BUILD)" "debug" -ifneq "$(BUILD)" "release" -BUILD=debug -endif -endif -ifeq "$(BUILD)" "debug" -D=d -DEBUGFLAGS=-g -D__WXDEBUG__ -endif +# for wxWidgets-2.8.x, search and replace "mswud"->"mswd" and "29u"->"28" +INCLUDEPATHS= \ + -I"/boost" \ + -I"/db/build_unix" \ + -I"/openssl/include" \ + -I"/wxwidgets/lib/gcc_lib/mswud" \ + -I"/wxwidgets/include" +LIBPATHS= \ + -L"/boost/stage/lib" \ + -L"/db/build_unix" \ + -L"/openssl/out" \ + -L"/wxwidgets/lib/gcc_lib" -INCLUDEPATHS=-I"/boost" -I"/db/build_unix" -I"/openssl/include" -I"/wxwidgets/lib/gcc_lib/mswd" -I"/wxwidgets/include" -LIBPATHS=-L"/boost/stage/lib" -L"/db/build_unix" -L"/openssl/out" -L"/wxwidgets/lib/gcc_lib" LIBS= \ -l libboost_system-mgw34-mt-d -l libboost_filesystem-mgw34-mt-d \ -l db_cxx \ -l eay32 \ - -l wxmsw28$(D)_richtext -l wxmsw28$(D)_html -l wxmsw28$(D)_core -l wxmsw28$(D)_adv -l wxbase28$(D) -l wxtiff$(D) -l wxjpeg$(D) -l wxpng$(D) -l wxzlib$(D) -l wxregex$(D) -l wxexpat$(D) \ + -l wxmsw29ud_html -l wxmsw29ud_core -l wxmsw29ud_adv -l wxbase29ud -l wxtiffd -l wxjpegd -l wxpngd -l wxzlibd \ -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 -l shlwapi + WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH +DEBUGFLAGS=-g -D__WXDEBUG__ CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h diff --git a/makefile.unix.wx2.9 b/makefile.unix similarity index 89% rename from makefile.unix.wx2.9 rename to makefile.unix index 0b3b09b5..a8ec9367 100644 --- a/makefile.unix.wx2.9 +++ b/makefile.unix @@ -3,17 +3,6 @@ # file license.txt or http://www.opensource.org/licenses/mit-license.php. -ifneq "$(BUILD)" "debug" -ifneq "$(BUILD)" "release" -BUILD=debug -endif -endif -ifeq "$(BUILD)" "debug" -D=d -DEBUGFLAGS=-g -D__WXDEBUG__ -endif - - INCLUDEPATHS= \ -I"/usr/include" \ @@ -28,12 +17,13 @@ LIBS= \ -Wl,-Bstatic \ -l boost_system-mt -l boost_filesystem-mt \ -l db_cxx \ - -l wx_gtk2u$(D)-2.9 \ + -l wx_gtk2ud-2.9 \ -Wl,-Bdynamic \ -l crypto \ -l gtk-x11-2.0 -l gthread-2.0 -l SM WXDEFS=-D__WXGTK__ -DNOPCH +DEBUGFLAGS=-g -D__WXDEBUG__ CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h diff --git a/makefile.unix.wx2.8 b/makefile.unix.wx2.8 deleted file mode 100644 index 38d9bc0e..00000000 --- a/makefile.unix.wx2.8 +++ /dev/null @@ -1,88 +0,0 @@ -# Copyright (c) 2009-2010 Satoshi Nakamoto -# Distributed under the MIT/X11 software license, see the accompanying -# file license.txt or http://www.opensource.org/licenses/mit-license.php. - - -ifneq "$(BUILD)" "debug" -ifneq "$(BUILD)" "release" -BUILD=debug -endif -endif -ifeq "$(BUILD)" "debug" -D=d -DEBUGFLAGS=-g -D__WXDEBUG__ -endif - - - -INCLUDEPATHS= \ - -I"/usr/include" \ - -I"/usr/local/include/wx-2.8" \ - -I"/usr/local/lib/wx/include/gtk2-ansi-debug-static-2.8" - -LIBPATHS= \ - -L"/usr/lib" \ - -L"/usr/local/lib" - -LIBS= \ - -Wl,-Bstatic \ - -l boost_system -l boost_filesystem \ - -l db_cxx \ - -l wx_gtk2$(D)-2.8 \ - -Wl,-Bdynamic \ - -l crypto \ - -l gtk-x11-2.0 -l gthread-2.0 -l SM - -WXDEFS=-D__WXGTK__ -DNOPCH -CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h - - - -all: bitcoin - - -headers.h.gch: headers.h $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/util.o: util.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/script.o: script.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/db.o: db.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/net.o: net.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/main.o: main.cpp $(HEADERS) sha.h - g++ -c $(CFLAGS) -o $@ $< - -obj/ui.o: ui.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/uibase.o: uibase.cpp uibase.h - g++ -c $(CFLAGS) -o $@ $< - -obj/sha.o: sha.cpp sha.h - g++ -c $(CFLAGS) -O3 -o $@ $< - -obj/irc.o: irc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/rpc.o: rpc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - - - -OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o - -bitcoin: headers.h.gch $(OBJS) - g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) - -clean: - -rm obj/* - -rm headers.h.gch diff --git a/makefile.vc b/makefile.vc index 5ca8de46..c407be3d 100644 --- a/makefile.vc +++ b/makefile.vc @@ -3,26 +3,31 @@ # file license.txt or http://www.opensource.org/licenses/mit-license.php. -!IF "$(BUILD)" != "debug" && "$(BUILD)" != "release" -BUILD=debug -!ENDIF -!IF "$(BUILD)" == "debug" -D=d -DEBUGFLAGS=/Zi /Od /D__WXDEBUG__ -!ENDIF +# for wxWidgets-2.8.x, search and replace "mswud"->"mswd" and "29u"->"28" +INCLUDEPATHS= \ + /I"/boost" \ + /I"/db/build_windows" \ + /I"/openssl/include" \ + /I"/wxwidgets/lib/vc_lib/mswud" \ + /I"/wxwidgets/include" +LIBPATHS= \ + /LIBPATH:"/boost/stage/lib" \ + /LIBPATH:"/db/build_windows/debug" \ + /LIBPATH:"/openssl/out" \ + /LIBPATH:"/wxwidgets/lib/vc_lib" -INCLUDEPATHS=/I"/boost" /I"/db/build_windows" /I"/openssl/include" /I"/wxwidgets/lib/vc_lib/mswd" /I"/wxwidgets/include" -LIBPATHS=/LIBPATH:"/boost/stage/lib" /LIBPATH:"/db/build_windows/$(BUILD)" /LIBPATH:"/openssl/out" /LIBPATH:"/wxwidgets/lib/vc_lib" LIBS= \ - libboost_system-vc80-mt-gd.lib libboost_filesystem-vc80-mt-gd.lib \ - libdb47s$(D).lib \ - libeay32.lib \ - wxmsw28$(D)_richtext.lib wxmsw28$(D)_html.lib wxmsw28$(D)_core.lib wxmsw28$(D)_adv.lib wxbase28$(D).lib wxtiff$(D).lib wxjpeg$(D).lib wxpng$(D).lib wxzlib$(D).lib wxregex$(D).lib wxexpat$(D).lib \ - kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib shlwapi.lib + libboost_system-vc80-mt-gd.lib libboost_filesystem-vc80-mt-gd.lib \ + libdb47sd.lib \ + libeay32.lib \ + wxmsw29ud_html.lib wxmsw29ud_core.lib wxmsw29ud_adv.lib wxbase29ud.lib wxtiffd.lib wxjpegd.lib wxpngd.lib wxzlibd.lib \ + kernel32.lib user32.lib gdi32.lib comdlg32.lib winspool.lib winmm.lib shell32.lib comctl32.lib ole32.lib oleaut32.lib uuid.lib rpcrt4.lib advapi32.lib ws2_32.lib shlwapi.lib + WXDEFS=/DWIN32 /D__WXMSW__ /D_WINDOWS /DNOPCH -CFLAGS=/c /nologo /Ob0 /MD$(D) /EHsc /GR /Zm300 $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +DEBUGFLAGS=/Zi /Od /D__WXDEBUG__ +CFLAGS=/c /nologo /Ob0 /MDd /EHsc /GR /Zm300 $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h diff --git a/serialize.h b/serialize.h index 12b58ad1..f3b99978 100644 --- a/serialize.h +++ b/serialize.h @@ -19,8 +19,8 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 201; -static const char* pszSubVer = ".2"; +static const int VERSION = 202; +static const char* pszSubVer = ".0"; diff --git a/ui.cpp b/ui.cpp index f506fcf3..87ba19e0 100644 --- a/ui.cpp +++ b/ui.cpp @@ -229,8 +229,8 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_staticTextBalance->SetSize(140, 17); // & underlines don't work on the toolbar buttons on gtk m_toolBar->ClearTools(); - m_toolBar->AddTool(wxID_BUTTONSEND, "Send Coins", wxBitmap(send20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); - m_toolBar->AddTool(wxID_BUTTONRECEIVE, "Address Book", wxBitmap(addressbook20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + m_toolBar->AddTool(wxID_BUTTONSEND, _("Send Coins"), wxBitmap(send20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); + m_toolBar->AddTool(wxID_BUTTONRECEIVE, _("Address Book"), wxBitmap(addressbook20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); m_toolBar->Realize(); // resize to fit ubuntu's huge default font dResize = 1.20; @@ -244,13 +244,13 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) int nDateWidth = DateTimeStr(1229413914).size() * 6 + 8; if (!strstr(DateTimeStr(1229413914).c_str(), "2008")) nDateWidth += 12; - m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, dResize * 0); - m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, dResize * 0); - m_listCtrl->InsertColumn(2, "Status", wxLIST_FORMAT_LEFT, dResize * 110); - m_listCtrl->InsertColumn(3, "Date", wxLIST_FORMAT_LEFT, dResize * nDateWidth); - m_listCtrl->InsertColumn(4, "Description", wxLIST_FORMAT_LEFT, dResize * 409 - nDateWidth); - m_listCtrl->InsertColumn(5, "Debit", wxLIST_FORMAT_RIGHT, dResize * 79); - m_listCtrl->InsertColumn(6, "Credit", wxLIST_FORMAT_RIGHT, dResize * 79); + m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, dResize * 0); + m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, dResize * 0); + m_listCtrl->InsertColumn(2, _("Status"), wxLIST_FORMAT_LEFT, dResize * 110); + m_listCtrl->InsertColumn(3, _("Date"), wxLIST_FORMAT_LEFT, dResize * nDateWidth); + m_listCtrl->InsertColumn(4, _("Description"), wxLIST_FORMAT_LEFT, dResize * 409 - nDateWidth); + m_listCtrl->InsertColumn(5, _("Debit"), wxLIST_FORMAT_RIGHT, dResize * 79); + m_listCtrl->InsertColumn(6, _("Credit"), wxLIST_FORMAT_RIGHT, dResize * 79); // Init status bar int pnWidths[3] = { -100, 88, 290 }; @@ -450,19 +450,19 @@ string FormatTxStatus(const CWalletTx& wtx) if (!wtx.IsFinal()) { if (wtx.nLockTime < 500000000) - return strprintf("Open for %d blocks", nBestHeight - wtx.nLockTime); + return strprintf(_("Open for %d blocks"), nBestHeight - wtx.nLockTime); else - return strprintf("Open until %s", DateTimeStr(wtx.nLockTime).c_str()); + return strprintf(_("Open until %s"), DateTimeStr(wtx.nLockTime).c_str()); } else { int nDepth = wtx.GetDepthInMainChain(); if (GetAdjustedTime() - wtx.nTimeReceived > 2 * 60 && wtx.GetRequestCount() == 0) - return strprintf("%d/offline?", nDepth); + return strprintf(_("%d/offline?"), nDepth); else if (nDepth < 6) - return strprintf("%d/unconfirmed", nDepth); + return strprintf(_("%d/unconfirmed"), nDepth); else - return strprintf("%d confirmations", nDepth); + return strprintf(_("%d confirmations"), nDepth); } } @@ -547,7 +547,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) if (wtx.IsCoinBase()) { // Coinbase - strDescription = "Generated"; + strDescription = _("Generated"); if (nCredit == 0) { int64 nUnmatured = 0; @@ -555,15 +555,15 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) nUnmatured += txout.GetCredit(); if (wtx.IsInMainChain()) { - strDescription = strprintf("Generated (%s matures in %d more blocks)", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); + strDescription = strprintf(_("Generated (%s matures in %d more blocks)"), FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); // Check if the block was requested by anyone if (GetAdjustedTime() - wtx.nTimeReceived > 2 * 60 && wtx.GetRequestCount() == 0) - strDescription = "Generated - Warning: This block was not received by any other nodes and will probably not be accepted!"; + strDescription = _("Generated - Warning: This block was not received by any other nodes and will probably not be accepted!"); } else { - strDescription = "Generated (not accepted)"; + strDescription = _("Generated (not accepted)"); } } } @@ -571,7 +571,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) { // Online transaction if (!mapValue["from"].empty()) - strDescription += "From: " + mapValue["from"]; + strDescription += _("From: ") + mapValue["from"]; if (!mapValue["message"].empty()) { if (!strDescription.empty()) @@ -594,9 +594,9 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) string strAddress = PubKeyToAddress(vchPubKey); if (mapAddressBook.count(strAddress)) { - //strDescription += "Received payment to "; - //strDescription += "Received with address "; - strDescription += "From: unknown, To: "; + //strDescription += _("Received payment to "); + //strDescription += _("Received with address "); + strDescription += _("From: unknown, To: "); strDescription += strAddress; /// The labeling feature is just too confusing, so I hid it /// by putting it at the end where it runs off the screen. @@ -636,7 +636,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) InsertLine(fNew, nIndex, hash, strSort, strStatus, nTime ? DateTimeStr(nTime) : "", - "Payment to yourself", + _("Payment to yourself"), "", ""); /// issue: can't tell which is the payment and which is the change anymore @@ -670,7 +670,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) strAddress = Hash160ToAddress(hash160); } - string strDescription = "To: "; + string strDescription = _("To: "); CRITICAL_BLOCK(cs_mapAddressBook) if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) strDescription += mapAddressBook[strAddress] + " "; @@ -960,12 +960,12 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) // Update status bar string strGen = ""; if (fGenerateBitcoins) - strGen = " Generating"; + strGen = _(" Generating"); if (fGenerateBitcoins && vNodes.empty()) - strGen = "(not connected)"; + strGen = _("(not connected)"); m_statusBar->SetStatusText(strGen, 1); - string strStatus = strprintf(" %d connections %d blocks %d transactions", vNodes.size(), nBestHeight + 1, nTransactionCount); + string strStatus = strprintf(_(" %d connections %d blocks %d transactions"), vNodes.size(), nBestHeight + 1, nTransactionCount); m_statusBar->SetStatusText(strStatus, 2); if (fDebug && GetTime() - nThreadSocketHandlerHeartbeat > 60) @@ -1160,20 +1160,20 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails - strHTML += "Status: " + FormatTxStatus(wtx); + strHTML += _("Status: ") + FormatTxStatus(wtx); int nRequests = wtx.GetRequestCount(); if (nRequests != -1) { if (nRequests == 0) - strHTML += ", has not been successfully broadcast yet"; + strHTML += _(", has not been successfully broadcast yet"); else if (nRequests == 1) - strHTML += strprintf(", broadcast through %d node", nRequests); + strHTML += strprintf(_(", broadcast through %d node"), nRequests); else - strHTML += strprintf(", broadcast through %d nodes", nRequests); + strHTML += strprintf(_(", broadcast through %d nodes"), nRequests); } strHTML += "
"; - strHTML += "Date: " + (nTime ? DateTimeStr(nTime) : "") + "
"; + strHTML += _("Date: ") + (nTime ? DateTimeStr(nTime) : "") + "
"; // @@ -1181,13 +1181,13 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails // if (wtx.IsCoinBase()) { - strHTML += "Source: Generated
"; + strHTML += _("Source: Generated
"); } else if (!wtx.mapValue["from"].empty()) { // Online transaction if (!wtx.mapValue["from"].empty()) - strHTML += "From: " + HtmlEscape(wtx.mapValue["from"]) + "
"; + strHTML += _("From: ") + HtmlEscape(wtx.mapValue["from"]) + "
"; } else { @@ -1205,13 +1205,13 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails string strAddress = PubKeyToAddress(vchPubKey); if (mapAddressBook.count(strAddress)) { - strHTML += "From: unknown
"; - strHTML += "To: "; + strHTML += string() + _("From: ") + _("unknown") + "
"; + strHTML += _("To: "); strHTML += HtmlEscape(strAddress); if (!mapAddressBook[strAddress].empty()) - strHTML += " (yours, label: " + mapAddressBook[strAddress] + ")"; + strHTML += _(" (yours, label: ") + mapAddressBook[strAddress] + ")"; else - strHTML += " (yours)"; + strHTML += _(" (yours)"); strHTML += "
"; } } @@ -1230,7 +1230,7 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails { // Online transaction strAddress = wtx.mapValue["to"]; - strHTML += "To: "; + strHTML += _("To: "); if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) strHTML += mapAddressBook[strAddress] + " "; strHTML += HtmlEscape(strAddress) + "
"; @@ -1248,17 +1248,19 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails int64 nUnmatured = 0; foreach(const CTxOut& txout, wtx.vout) nUnmatured += txout.GetCredit(); + strHTML += _("Credit: "); if (wtx.IsInMainChain()) - strHTML += strprintf("Credit: (%s matures in %d more blocks)
", FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); + strHTML += strprintf(_("(%s matures in %d more blocks)"), FormatMoney(nUnmatured).c_str(), wtx.GetBlocksToMaturity()); else - strHTML += "Credit: (not accepted)
"; + strHTML += _("(not accepted)"); + strHTML += "
"; } else if (nNet > 0) { // // Credit // - strHTML += "Credit: " + FormatMoney(nNet) + "
"; + strHTML += _("Credit: ") + FormatMoney(nNet) + "
"; } else { @@ -1287,7 +1289,7 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails if (ExtractHash160(txout.scriptPubKey, hash160)) { string strAddress = Hash160ToAddress(hash160); - strHTML += "To: "; + strHTML += _("To: "); if (mapAddressBook.count(strAddress) && !mapAddressBook[strAddress].empty()) strHTML += mapAddressBook[strAddress] + " "; strHTML += strAddress; @@ -1295,7 +1297,7 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails } } - strHTML += "Debit: " + FormatMoney(-txout.nValue) + "
"; + strHTML += _("Debit: ") + FormatMoney(-txout.nValue) + "
"; } if (fAllToMe) @@ -1303,13 +1305,13 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails // Payment to self /// issue: can't tell which is the payment and which is the change anymore //int64 nValue = wtx.vout[0].nValue; - //strHTML += "Debit: " + FormatMoney(-nValue) + "
"; - //strHTML += "Credit: " + FormatMoney(nValue) + "
"; + //strHTML += _("Debit: ") + FormatMoney(-nValue) + "
"; + //strHTML += _("Credit: ") + FormatMoney(nValue) + "
"; } int64 nTxFee = nDebit - wtx.GetValueOut(); if (nTxFee > 0) - strHTML += "Transaction fee: " + FormatMoney(-nTxFee) + "
"; + strHTML += _("Transaction fee: ") + FormatMoney(-nTxFee) + "
"; } else { @@ -1318,24 +1320,24 @@ CTxDetailsDialog::CTxDetailsDialog(wxWindow* parent, CWalletTx wtx) : CTxDetails // foreach(const CTxIn& txin, wtx.vin) if (txin.IsMine()) - strHTML += "Debit: " + FormatMoney(-txin.GetDebit()) + "
"; + strHTML += _("Debit: ") + FormatMoney(-txin.GetDebit()) + "
"; foreach(const CTxOut& txout, wtx.vout) if (txout.IsMine()) - strHTML += "Credit: " + FormatMoney(txout.GetCredit()) + "
"; + strHTML += _("Credit: ") + FormatMoney(txout.GetCredit()) + "
"; } } - strHTML += "Net amount: " + FormatMoney(nNet, true) + "
"; + strHTML += _("Net amount: ") + FormatMoney(nNet, true) + "
"; // // Message // if (!wtx.mapValue["message"].empty()) - strHTML += "
Message:
" + HtmlEscape(wtx.mapValue["message"], true) + "
"; + strHTML += string() + "
" + _("Message:") + "
" + HtmlEscape(wtx.mapValue["message"], true) + "
"; if (wtx.IsCoinBase()) - strHTML += "
Generated coins must wait 120 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, it will change to \"not accepted\" and not be spendable. This may occasionally happen if another node generates a block within a few seconds of yours.
"; + strHTML += string() + "
" + _("Generated coins must wait 120 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, it will change to \"not accepted\" and not be spendable. This may occasionally happen if another node generates a block within a few seconds of yours.") + "
"; // @@ -1402,12 +1404,12 @@ void CTxDetailsDialog::OnButtonOK(wxCommandEvent& event) COptionsDialog::COptionsDialog(wxWindow* parent) : COptionsDialogBase(parent) { // Set up list box of page choices - m_listBox->Append("Main"); - //m_listBox->Append("Test 2"); + m_listBox->Append(_("Main")); + //m_listBox->Append(_("Test 2")); m_listBox->SetSelection(0); SelectPage(0); #ifndef __WXMSW__ - m_checkBoxMinimizeOnClose->SetLabel("&Minimize on close"); + m_checkBoxMinimizeOnClose->SetLabel(_("&Minimize on close")); m_checkBoxStartOnSystemStartup->Enable(false); // not implemented yet #endif @@ -1559,16 +1561,16 @@ void COptionsDialog::OnButtonApply(wxCommandEvent& event) CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) { - m_staticTextVersion->SetLabel(strprintf("version 0.%d.%d beta", VERSION/100, VERSION%100)); + m_staticTextVersion->SetLabel(strprintf(_("version 0.%d.%d beta"), VERSION/100, VERSION%100)); -#if !wxUSE_UNICODE - // Workaround until upgrade to wxWidgets supporting UTF-8 - // Hack to change the (c) character from UTF-8 back to ANSI + // Change (c) into UTF-8 or ANSI copyright symbol wxString str = m_staticTextMain->GetLabel(); - if (str.Find('\xC2') != wxNOT_FOUND) - str.Remove(str.Find('\xC2'), 1); - m_staticTextMain->SetLabel(str); +#if wxUSE_UNICODE + str.Replace("(c)", wxString::FromUTF8("\xC2\xA9")); +#else + str.Replace("(c)", "\xA9"); #endif + m_staticTextMain->SetLabel(str); #ifndef __WXMSW__ // Resize on Linux to make the window fit the text. // The text was wrapped manually rather than using the Wrap setting because @@ -1577,7 +1579,7 @@ CAboutDialog::CAboutDialog(wxWindow* parent) : CAboutDialogBase(parent) if (fontTmp.GetPointSize() > 8); fontTmp.SetPointSize(8); m_staticTextMain->SetFont(fontTmp); - SetSize(GetSize().GetWidth() + 44, GetSize().GetHeight() - 4); + SetSize(GetSize().GetWidth() + 44, GetSize().GetHeight() + 10); #endif } @@ -1644,8 +1646,8 @@ void CSendDialog::OnTextAddress(wxCommandEvent& event) { strFromSave = m_textCtrlFrom->GetValue(); strMessageSave = m_textCtrlMessage->GetValue(); - m_textCtrlFrom->SetValue("Will appear as \"From: Unknown\""); - m_textCtrlMessage->SetValue("Can't include a message when sending to a Bitcoin address"); + m_textCtrlFrom->SetValue(_("Will appear as \"From: Unknown\"")); + m_textCtrlMessage->SetValue(_("Can't include a message when sending to a Bitcoin address")); } else if (fEnable && !fEnabledPrev) { @@ -1697,17 +1699,17 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event) int64 nValue = 0; if (!ParseMoney(m_textCtrlAmount->GetValue(), nValue) || nValue <= 0) { - wxMessageBox("Error in amount ", "Send Coins"); + wxMessageBox(_("Error in amount "), _("Send Coins")); return; } if (nValue > GetBalance()) { - wxMessageBox("Amount exceeds your balance ", "Send Coins"); + wxMessageBox(_("Amount exceeds your balance "), _("Send Coins")); return; } if (nValue + nTransactionFee > GetBalance()) { - wxMessageBox(string("Total exceeds your balance when the ") + FormatMoney(nTransactionFee) + " transaction fee is included ", "Send Coins"); + wxMessageBox(string(_("Total exceeds your balance when the ")) + FormatMoney(nTransactionFee) + _(" transaction fee is included "), _("Send Coins")); return; } @@ -1723,9 +1725,9 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event) string strError = SendMoney(scriptPubKey, nValue, wtx); if (strError != "") - wxMessageBox(strError + " ", "Sending..."); + wxMessageBox(strError + " ", _("Sending...")); else - wxMessageBox("Payment sent ", "Sending..."); + wxMessageBox(_("Payment sent "), _("Sending...")); } else { @@ -1733,7 +1735,7 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event) CAddress addr(strAddress); if (!addr.IsValid()) { - wxMessageBox("Invalid address ", "Send Coins"); + wxMessageBox(_("Invalid address "), _("Send Coins")); return; } @@ -1787,7 +1789,7 @@ CSendingDialog::CSendingDialog(wxWindow* parent, const CAddress& addrIn, int64 n SetSize(1.2 * GetSize().GetWidth(), 1.08 * GetSize().GetHeight()); #endif - SetTitle(strprintf("Sending %s to %s", FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); + SetTitle(strprintf(_("Sending %s to %s"), FormatMoney(nPrice).c_str(), wtx.mapValue["to"].c_str())); m_textCtrlStatus->SetValue(""); CreateThread(SendingDialogStartTransfer, this); @@ -1859,13 +1861,13 @@ void CSendingDialog::OnPaint(wxPaintEvent& event) } if (fAbort && fCanCancel && IsShown()) { - strcpy(pszStatus, "CANCELLED"); + strcpy(pszStatus, _("CANCELLED")); m_buttonOK->Enable(true); m_buttonOK->SetFocus(); m_buttonCancel->Enable(false); - m_buttonCancel->SetLabel("Cancelled"); + m_buttonCancel->SetLabel(_("Cancelled")); Close(); - wxMessageBox("Transfer cancelled ", "Sending...", wxOK, this); + wxMessageBox(_("Transfer cancelled "), _("Sending..."), wxOK, this); } event.Skip(); } @@ -1893,7 +1895,7 @@ bool CSendingDialog::Status() if (fAbort && fCanCancel) { memset(pszStatus, 0, 10); - strcpy(pszStatus, "CANCELLED"); + strcpy(pszStatus, _("CANCELLED")); Repaint(); fWorkDone = true; return false; @@ -1919,7 +1921,7 @@ bool CSendingDialog::Error(const string& str) { fCanCancel = false; fWorkDone = true; - Status(string("Error: ") + str); + Status(string(_("Error: ")) + str); return false; } @@ -1933,22 +1935,22 @@ void CSendingDialog::StartTransfer() // Make sure we have enough money if (nPrice + nTransactionFee > GetBalance()) { - Error("You don't have enough money"); + Error(_("You don't have enough money")); return; } // We may have connected already for product details - if (!Status("Connecting...")) + if (!Status(_("Connecting..."))) return; CNode* pnode = ConnectNode(addr, 15 * 60); if (!pnode) { - Error("Unable to connect"); + Error(_("Unable to connect")); return; } // Send order to seller, with response going to OnReply2 via event handler - if (!Status("Requesting public key...")) + if (!Status(_("Requesting public key..."))) return; pnode->PushRequest("checkorder", wtx, SendingDialogOnReply2, this); } @@ -1960,7 +1962,7 @@ void SendingDialogOnReply2(void* parg, CDataStream& vRecv) void CSendingDialog::OnReply2(CDataStream& vRecv) { - if (!Status("Received public key...")) + if (!Status(_("Received public key..."))) return; CScript scriptPubKey; @@ -1972,7 +1974,7 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) { string strMessage; vRecv >> strMessage; - Error("Transfer was not accepted"); + Error(_("Transfer was not accepted")); //// todo: enlarge the window and enable a hidden white box to put seller's message return; } @@ -1981,7 +1983,7 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) catch (...) { //// what do we want to do about this? - Error("Invalid response received"); + Error(_("Invalid response received")); return; } @@ -1996,11 +1998,11 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) CRITICAL_BLOCK(cs_main) { // Pay - if (!Status("Creating transaction...")) + if (!Status(_("Creating transaction..."))) return; if (nPrice + nTransactionFee > GetBalance()) { - Error("You don't have enough money"); + Error(_("You don't have enough money")); return; } CKey key; @@ -2008,9 +2010,9 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) if (!CreateTransaction(scriptPubKey, nPrice, wtx, key, nFeeRequired)) { if (nPrice + nFeeRequired > GetBalance()) - Error(strprintf("This is an oversized transaction that requires a transaction fee of %s", FormatMoney(nFeeRequired).c_str())); + Error(strprintf(_("This is an oversized transaction that requires a transaction fee of %s"), FormatMoney(nFeeRequired).c_str())); else - Error("Transaction creation failed"); + Error(_("Transaction creation failed")); return; } @@ -2018,7 +2020,7 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) CNode* pnode = ConnectNode(addr, 2 * 60 * 60); if (!pnode) { - Error("Lost connection, transaction cancelled"); + Error(_("Lost connection, transaction cancelled")); return; } @@ -2034,13 +2036,13 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) return; fCanCancel = false; } - if (!Status("Sending payment...")) + if (!Status(_("Sending payment..."))) return; // Commit if (!CommitTransactionSpent(wtx, key)) { - Error("Error finalizing payment"); + Error(_("Error finalizing payment")); return; } @@ -2052,7 +2054,7 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) printf("ERROR: CSendingDialog : wtxNew.AcceptTransaction() %s failed\n", wtx.GetHash().ToString().c_str()); wtx.RelayWalletTransaction(); - Status("Waiting for confirmation..."); + Status(_("Waiting for confirmation...")); MainFrameRepaint(); } } @@ -2070,22 +2072,22 @@ void CSendingDialog::OnReply3(CDataStream& vRecv) vRecv >> nRet; if (nRet > 0) { - Error("The payment was sent, but the recipient was unable to verify it.\n" - "The transaction is recorded and will credit to the recipient,\n" - "but the comment information will be blank."); + Error(_("The payment was sent, but the recipient was unable to verify it.\n" + "The transaction is recorded and will credit to the recipient,\n" + "but the comment information will be blank.")); return; } } catch (...) { //// what do we want to do about this? - Error("Payment was sent, but an invalid response was received"); + Error(_("Payment was sent, but an invalid response was received")); return; } fSuccess = true; fWorkDone = true; - Status("Payment completed"); + Status(_("Payment completed")); } @@ -2101,8 +2103,8 @@ void CSendingDialog::OnReply3(CDataStream& vRecv) CYourAddressDialog::CYourAddressDialog(wxWindow* parent, const string& strInitSelected) : CYourAddressDialogBase(parent) { // Init column headers - m_listCtrl->InsertColumn(0, "Label", wxLIST_FORMAT_LEFT, 200); - m_listCtrl->InsertColumn(1, "Bitcoin Address", wxLIST_FORMAT_LEFT, 350); + m_listCtrl->InsertColumn(0, _("Label"), wxLIST_FORMAT_LEFT, 200); + m_listCtrl->InsertColumn(1, _("Bitcoin Address"), wxLIST_FORMAT_LEFT, 350); m_listCtrl->SetFocus(); // Fill listctrl with address book data @@ -2162,7 +2164,7 @@ void CYourAddressDialog::OnButtonRename(wxCommandEvent& event) return; string strName = (string)m_listCtrl->GetItemText(nIndex); string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); - CGetTextFromUserDialog dialog(this, "Edit Address Label", "New Label", strName); + CGetTextFromUserDialog dialog(this, _("Edit Address Label"), _("New Label"), strName); if (!dialog.ShowModal()) return; strName = dialog.GetValue(); @@ -2176,7 +2178,7 @@ void CYourAddressDialog::OnButtonRename(wxCommandEvent& event) void CYourAddressDialog::OnButtonNew(wxCommandEvent& event) { // Ask name - CGetTextFromUserDialog dialog(this, "New Bitcoin Address", "Label", ""); + CGetTextFromUserDialog dialog(this, _("New Bitcoin Address"), _("Label"), ""); if (!dialog.ShowModal()) return; string strName = dialog.GetValue(); @@ -2236,8 +2238,8 @@ CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInit m_buttonCancel->Show(false); // Init column headers - m_listCtrl->InsertColumn(0, "Name", wxLIST_FORMAT_LEFT, 200); - m_listCtrl->InsertColumn(1, "Address", wxLIST_FORMAT_LEFT, 350); + m_listCtrl->InsertColumn(0, _("Name"), wxLIST_FORMAT_LEFT, 200); + m_listCtrl->InsertColumn(1, _("Address"), wxLIST_FORMAT_LEFT, 350); m_listCtrl->SetFocus(); // Set Icon @@ -2307,7 +2309,7 @@ bool CAddressBookDialog::CheckIfMine(const string& strAddress, const string& str uint160 hash160; bool fMine = (AddressToHash160(strAddress, hash160) && mapPubKeys.count(hash160)); if (fMine) - wxMessageBox("This is one of your own addresses for receiving payments and cannot be entered in the address book. ", strTitle); + wxMessageBox(_("This is one of your own addresses for receiving payments and cannot be entered in the address book. "), strTitle); return fMine; } @@ -2322,13 +2324,13 @@ void CAddressBookDialog::OnButtonEdit(wxCommandEvent& event) string strAddressOrg = strAddress; do { - CGetTextFromUserDialog dialog(this, "Edit Address", "Name", strName, "Address", strAddress); + CGetTextFromUserDialog dialog(this, _("Edit Address"), _("Name"), strName, _("Address"), strAddress); if (!dialog.ShowModal()) return; strName = dialog.GetValue1(); strAddress = dialog.GetValue2(); } - while (CheckIfMine(strAddress, "Edit Address")); + while (CheckIfMine(strAddress, _("Edit Address"))); // Change name if (strAddress != strAddressOrg) @@ -2346,13 +2348,13 @@ void CAddressBookDialog::OnButtonNew(wxCommandEvent& event) string strAddress; do { - CGetTextFromUserDialog dialog(this, "New Address", "Name", strName, "Address", strAddress); + CGetTextFromUserDialog dialog(this, _("New Address"), _("Name"), strName, _("Address"), strAddress); if (!dialog.ShowModal()) return; strName = dialog.GetValue1(); strAddress = dialog.GetValue2(); } - while (CheckIfMine(strAddress, "New Address")); + while (CheckIfMine(strAddress, _("New Address"))); // Add to list and select it SetAddressBookName(strAddress, strName); @@ -2436,11 +2438,11 @@ void CMyTaskBarIcon::Show(bool fShow) static char pszPrevTip[200]; if (fShow) { - string strTooltip = "Bitcoin"; + string strTooltip = _("Bitcoin"); if (fGenerateBitcoins) - strTooltip = "Bitcoin - Generating"; + strTooltip = _("Bitcoin - Generating"); if (fGenerateBitcoins && vNodes.empty()) - strTooltip = "Bitcoin - (not connected)"; + strTooltip = _("Bitcoin - (not connected)"); // Optimization, only update when changed, using char array to be reentrant if (strncmp(pszPrevTip, strTooltip.c_str(), sizeof(pszPrevTip)-1) != 0) @@ -2515,12 +2517,12 @@ void CMyTaskBarIcon::UpdateTooltip() wxMenu* CMyTaskBarIcon::CreatePopupMenu() { wxMenu* pmenu = new wxMenu; - pmenu->Append(ID_TASKBAR_RESTORE, "&Open Bitcoin"); - pmenu->Append(ID_TASKBAR_OPTIONS, "O&ptions..."); - pmenu->AppendCheckItem(ID_TASKBAR_GENERATE, "&Generate Coins")->Check(fGenerateBitcoins); + pmenu->Append(ID_TASKBAR_RESTORE, _("&Open Bitcoin")); + pmenu->Append(ID_TASKBAR_OPTIONS, _("O&ptions...")); + pmenu->AppendCheckItem(ID_TASKBAR_GENERATE, _("&Generate Coins"))->Check(fGenerateBitcoins); #ifndef __WXMAC_OSX__ // Mac has built-in quit menu pmenu->AppendSeparator(); - pmenu->Append(ID_TASKBAR_EXIT, "E&xit"); + pmenu->Append(ID_TASKBAR_EXIT, _("E&xit")); #endif return pmenu; } @@ -2542,7 +2544,9 @@ wxMenu* CMyTaskBarIcon::CreatePopupMenu() // Define a new application class CMyApp: public wxApp { - public: +public: + wxLocale m_locale; + CMyApp(){}; ~CMyApp(){}; bool OnInit(); @@ -2584,9 +2588,9 @@ bool CMyApp::OnInit() bool CMyApp::OnInit2() { #ifdef _MSC_VER - // Turn off microsoft heap dump noise for now + // Turn off microsoft heap dump noise _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE); - _CrtSetReportFile(_CRT_WARN, CreateFile("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); + _CrtSetReportFile(_CRT_WARN, CreateFileA("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); #endif #if defined(__WXMSW__) && defined(__WXDEBUG__) // Disable malfunctioning wxWidgets debug assertion @@ -2599,6 +2603,31 @@ bool CMyApp::OnInit2() SetAppName("bitcoin"); umask(077); #endif +#ifdef __WXMSW__ +#if wxUSE_UNICODE + // Hack to set wxConvLibc codepage to UTF-8 on Windows, + // may break if wxMBConv_win32 implementation in strconv.cpp changes. + class wxMBConv_win32 : public wxMBConv + { + public: + long m_CodePage; + size_t m_minMBCharWidth; + }; + if (((wxMBConv_win32*)&wxConvLibc)->m_CodePage == CP_ACP) + ((wxMBConv_win32*)&wxConvLibc)->m_CodePage = CP_UTF8; +#endif +#endif + + // Load locale//LC_MESSAGES/bitcoin.mo language file + m_locale.Init(wxLANGUAGE_DEFAULT, 0); + m_locale.AddCatalogLookupPathPrefix("locale"); + if (!fWindows) + { + m_locale.AddCatalogLookupPathPrefix("/usr/share/locale"); + m_locale.AddCatalogLookupPathPrefix("/usr/local/share/locale"); + } + m_locale.AddCatalog("wxstd"); // wxWidgets standard translations, if any + m_locale.AddCatalog("bitcoin"); // // Parameters @@ -2612,33 +2641,31 @@ bool CMyApp::OnInit2() ParseParameters(argc, argv); if (mapArgs.count("-?") || mapArgs.count("--help")) { -#ifdef __WXMSW__ - string strUsage = - "Usage: bitcoin [options]\t\t\t\t\t\t\n" - "Options:\n" - " -gen\t\t Generate coins\n" - " -gen=0\t\t Don't generate coins\n" - " -min\t\t Start minimized\n" - " -datadir=\t Specify data directory\n" - " -proxy=\t Connect through socks4 proxy\n" - " -addnode=\t Add a node to connect to\n" - " -connect=\t Connect only to the specified node\n" - " -?\t\t This help message\n"; - wxMessageBox(strUsage, "Bitcoin", wxOK); -#else - string strUsage = - "Usage: bitcoin [options]\n" - "Options:\n" - " -gen Generate coins\n" - " -gen=0 Don't generate coins\n" - " -min Start minimized\n" - " -datadir= Specify data directory\n" - " -proxy= Connect through socks4 proxy\n" - " -addnode= Add a node to connect to\n" - " -connect= Connect only to the specified node\n" - " -? This help message\n"; - fprintf(stderr, "%s", strUsage.c_str()); -#endif + wxString strUsage = string() + + _("Usage: bitcoin [options]") + "\t\t\t\t\t\t\n" + + _("Options:\n") + + " -gen \t\t " + _("Generate coins\n") + + " -gen=0 \t\t " + _("Don't generate coins\n") + + " -min \t\t " + _("Start minimized\n") + + " -datadir= \t " + _("Specify data directory\n") + + " -proxy=\t " + _("Connect through socks4 proxy\n") + + " -addnode= \t " + _("Add a node to connect to\n") + + " -connect= \t " + _("Connect only to the specified node\n") + + " -? \t\t " + _("This help message\n"); + + if (fWindows) + { + // Remove spaces, the tabs make the columns line up in the message box + for (int i = 0; i < 50; i++) + strUsage.Replace(" \t", "\t"); + wxMessageBox(strUsage, "Bitcoin", wxOK); + } + else + { + // Remove tabs + strUsage.Replace("\t", ""); + fprintf(stderr, "%s", ((string)strUsage).c_str()); + } return false; } @@ -2661,7 +2688,9 @@ bool CMyApp::OnInit2() if (!fDebug && !pszSetDataDir[0]) ShrinkDebugFile(); printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); - printf("Bitcoin version %d%s, OS version %s\n", VERSION, pszSubVer, ((string)wxGetOsDescription()).c_str()); + printf("Bitcoin version 0.%d.%d%s beta, OS version %s\n", VERSION/100, VERSION%100, pszSubVer, ((string)wxGetOsDescription()).c_str()); + printf("System default language is %d %s\n", m_locale.GetSystemLanguage(), ((string)m_locale.GetSysName()).c_str()); + printf("Language file %s (%s)\n", (string("locale/") + (string)m_locale.GetCanonicalName() + "/LC_MESSAGES/bitcoin.mo").c_str(), ((string)m_locale.GetLocale()).c_str()); if (mapArgs.count("-loadblockindextest")) { @@ -2691,7 +2720,7 @@ bool CMyApp::OnInit2() { // TODO: find out how to do this in Linux, or replace with wxWidgets commands // Show the previous instance and exit - HWND hwndPrev = FindWindow("wxWindowClassNR", "Bitcoin"); + HWND hwndPrev = FindWindowA("wxWindowClassNR", "Bitcoin"); if (hwndPrev) { if (IsIconic(hwndPrev)) @@ -2732,19 +2761,19 @@ bool CMyApp::OnInit2() printf("Loading addresses...\n"); nStart = GetTimeMillis(); if (!LoadAddresses()) - strErrors += "Error loading addr.dat \n"; + strErrors += _("Error loading addr.dat \n"); printf(" addresses %15"PRI64d"ms\n", GetTimeMillis() - nStart); printf("Loading block index...\n"); nStart = GetTimeMillis(); if (!LoadBlockIndex()) - strErrors += "Error loading blkindex.dat \n"; + strErrors += _("Error loading blkindex.dat \n"); printf(" block index %15"PRI64d"ms\n", GetTimeMillis() - nStart); printf("Loading wallet...\n"); nStart = GetTimeMillis(); if (!LoadWallet(fFirstRun)) - strErrors += "Error loading wallet.dat \n"; + strErrors += _("Error loading wallet.dat \n"); printf(" wallet %15"PRI64d"ms\n", GetTimeMillis() - nStart); printf("Done loading\n"); @@ -2812,7 +2841,7 @@ bool CMyApp::OnInit2() addrProxy = CAddress(mapArgs["-proxy"]); if (!addrProxy.IsValid()) { - wxMessageBox("Invalid -proxy address", "Bitcoin"); + wxMessageBox(_("Invalid -proxy address"), "Bitcoin"); return false; } } @@ -2947,7 +2976,7 @@ void CMyApp::OnUnhandledException() void CMyApp::OnFatalException() { - wxMessageBox("Program has crashed and will terminate. ", "Bitcoin", wxOK | wxICON_ERROR); + wxMessageBox(_("Program has crashed and will terminate. "), "Bitcoin", wxOK | wxICON_ERROR); } @@ -2962,7 +2991,7 @@ string MyGetSpecialFolderPath(int nFolder, bool fCreate) char pszPath[MAX_PATH+100] = ""; // SHGetSpecialFolderPath is not usually available on NT 4.0 - HMODULE hShell32 = LoadLibrary("shell32.dll"); + HMODULE hShell32 = LoadLibraryA("shell32.dll"); if (hShell32) { PSHGETSPECIALFOLDERPATHA pSHGetSpecialFolderPath = @@ -3017,7 +3046,7 @@ void SetStartOnSystemStartup(bool fAutoStart) if (SUCCEEDED(hres)) { // Get the current executable path - char pszExePath[MAX_PATH]; + TCHAR pszExePath[MAX_PATH]; GetModuleFileName(NULL, pszExePath, sizeof(pszExePath)); // Set the path to the shortcut target diff --git a/ui.h b/ui.h index da1ada7d..50d39ab7 100644 --- a/ui.h +++ b/ui.h @@ -16,12 +16,12 @@ extern int fMinimizeOnClose; -extern void HandleCtrlA(wxKeyEvent& event); -extern string FormatTxStatus(const CWalletTx& wtx); -extern void UIThreadCall(boost::function0); -extern void MainFrameRepaint(); -extern void Shutdown(void* parg); -extern int ThreadSafeMessageBox(const string& message, const string& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1); +void HandleCtrlA(wxKeyEvent& event); +string FormatTxStatus(const CWalletTx& wtx); +void UIThreadCall(boost::function0); +void MainFrameRepaint(); +void Shutdown(void* parg); +int ThreadSafeMessageBox(const string& message, const string& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1); diff --git a/uibase.cpp b/uibase.cpp index 03bd7912..5b3f9abc 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -1,6 +1,3 @@ -// Copyright (c) 2009-2010 Satoshi Nakamoto -// Distributed under the MIT/X11 software license, see the accompanying -// file license.txt or http://www.opensource.org/licenses/mit-license.php. /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -406,22 +403,14 @@ COptionsDialogBase::COptionsDialogBase( wxWindow* parent, wxWindowID id, const w bSizer64->Add( 0, 16, 0, wxEXPAND, 5 ); - m_staticText321 = new wxStaticText( m_panelTest2, wxID_ANY, _("Test panel 2 for future expansion"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText321 = new wxStaticText( m_panelTest2, wxID_ANY, _("// [don't translate] Test panel 2 for future expansion"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText321->Wrap( -1 ); bSizer64->Add( m_staticText321, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 ); - m_staticText69 = new wxStaticText( m_panelTest2, wxID_ANY, _("Let's not start multiple pages until the first page is filled up"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText69 = new wxStaticText( m_panelTest2, wxID_ANY, _("// [don't translate] Let's not start multiple pages until the first page is filled up"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText69->Wrap( -1 ); bSizer64->Add( m_staticText69, 0, wxALL, 5 ); - m_staticText70 = new wxStaticText( m_panelTest2, wxID_ANY, _("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText70->Wrap( -1 ); - bSizer64->Add( m_staticText70, 0, wxALL, 5 ); - - m_staticText71 = new wxStaticText( m_panelTest2, wxID_ANY, _("MyLabel"), wxDefaultPosition, wxDefaultSize, 0 ); - m_staticText71->Wrap( -1 ); - bSizer64->Add( m_staticText71, 0, wxALL, 5 ); - m_panelTest2->SetSizer( bSizer64 ); m_panelTest2->Layout(); bSizer64->Fit( m_panelTest2 ); @@ -521,7 +510,7 @@ CAboutDialogBase::CAboutDialogBase( wxWindow* parent, wxWindowID id, const wxStr bSizer631->Add( 0, 4, 0, wxEXPAND, 5 ); - m_staticTextMain = new wxStaticText( this, wxID_ANY, _("Copyright © 2009-2010 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file \nlicense.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the \nOpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by \nEric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticTextMain = new wxStaticText( this, wxID_ANY, _("Copyright (c) 2009-2010 Satoshi Nakamoto.\n\nThis is experimental software. Do not rely on it for actual financial transactions.\n\nDistributed under the MIT/X11 software license, see the accompanying file \nlicense.txt or http://www.opensource.org/licenses/mit-license.php.\n\nThis product includes software developed by the OpenSSL Project for use in the \nOpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by \nEric Young (eay@cryptsoft.com)."), wxDefaultPosition, wxDefaultSize, 0 ); m_staticTextMain->Wrap( -1 ); bSizer631->Add( m_staticTextMain, 0, wxALL, 5 ); diff --git a/uibase.h b/uibase.h index 2dfc2af1..494ed10f 100644 --- a/uibase.h +++ b/uibase.h @@ -1,6 +1,3 @@ -// Copyright (c) 2009-2010 Satoshi Nakamoto -// Distributed under the MIT/X11 software license, see the accompanying -// file license.txt or http://www.opensource.org/licenses/mit-license.php. /////////////////////////////////////////////////////////////////////////// // C++ code generated with wxFormBuilder (version Apr 16 2008) // http://www.wxformbuilder.org/ @@ -178,8 +175,6 @@ class COptionsDialogBase : public wxDialog wxStaticText* m_staticText321; wxStaticText* m_staticText69; - wxStaticText* m_staticText70; - wxStaticText* m_staticText71; wxButton* m_buttonOK; wxButton* m_buttonCancel; wxButton* m_buttonApply; diff --git a/uiproject.fbp b/uiproject.fbp index 5c6eaa55..988df2a9 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -2173,7 +2173,7 @@ 0 wxID_ANY - Test panel 2 for future expansion + // [don't translate] Test panel 2 for future expansion m_staticText321 @@ -2224,7 +2224,7 @@ 0 wxID_ANY - Let's not start multiple pages until the first page is filled up + // [don't translate] Let's not start multiple pages until the first page is filled up m_staticText69 @@ -2263,108 +2263,6 @@
- - 5 - wxALL - 0 - - - - 1 - - - 0 - wxID_ANY - MyLabel - - - m_staticText70 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL - 0 - - - - 1 - - - 0 - wxID_ANY - MyLabel - - - m_staticText71 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - -
@@ -2820,7 +2718,7 @@ 0 wxID_ANY - Copyright © 2009-2010 Satoshi Nakamoto. This is experimental software. Do not rely on it for actual financial transactions. Distributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php. This product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com). + Copyright (c) 2009-2010 Satoshi Nakamoto. This is experimental software. Do not rely on it for actual financial transactions. Distributed under the MIT/X11 software license, see the accompanying file license.txt or http://www.opensource.org/licenses/mit-license.php. This product includes software developed by the OpenSSL Project for use in the OpenSSL Toolkit (http://www.openssl.org/) and cryptographic software written by Eric Young (eay@cryptsoft.com). m_staticTextMain diff --git a/util.cpp b/util.cpp index 2f5be222..266c1dd7 100644 --- a/util.cpp +++ b/util.cpp @@ -90,7 +90,7 @@ void RandAddSeedPerfmon() unsigned char pdata[250000]; memset(pdata, 0, sizeof(pdata)); unsigned long nSize = sizeof(pdata); - long ret = RegQueryValueEx(HKEY_PERFORMANCE_DATA, "Global", NULL, NULL, pdata, &nSize); + long ret = RegQueryValueExA(HKEY_PERFORMANCE_DATA, "Global", NULL, NULL, pdata, &nSize); RegCloseKey(HKEY_PERFORMANCE_DATA); if (ret == ERROR_SUCCESS) { @@ -193,7 +193,7 @@ inline int OutputDebugStringF(const char* pszFormat, ...) p2++; char c = *p2; *p2 = '\0'; - OutputDebugString(p1); + OutputDebugStringA(p1); *p2 = c; p1 = p2; } @@ -441,7 +441,7 @@ void FormatException(char* pszMessage, std::exception* pex, const char* pszThrea #ifdef __WXMSW__ char pszModule[MAX_PATH]; pszModule[0] = '\0'; - GetModuleFileName(NULL, pszModule, sizeof(pszModule)); + GetModuleFileNameA(NULL, pszModule, sizeof(pszModule)); #else // might not be thread safe, uses wxString //const char* pszModule = wxStandardPaths::Get().GetExecutablePath().mb_str(); diff --git a/util.h b/util.h index 933c71be..188a9ece 100644 --- a/util.h +++ b/util.h @@ -340,10 +340,11 @@ void skipspaces(T& it) ++it; } - - - - +inline const char* wxGetTranslation(const char* psz) +{ + // Return translated UTF-8 const char* + return wxGetTranslation(wxString(psz, wxConvUTF8)).utf8_str(); +} From c6ab3cf6d9cf5f5ade3b8fc48590e1bc4794cb43 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 17 Feb 2010 23:55:43 +0000 Subject: [PATCH 055/133] safer wxGetTranslation wrapper git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@67 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- serialize.h | 2 +- util.cpp | 29 +++++++++++++++++++++++++++++ util.h | 9 ++++----- 3 files changed, 34 insertions(+), 6 deletions(-) diff --git a/serialize.h b/serialize.h index f3b99978..aaf3f990 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 202; +static const int VERSION = 203; static const char* pszSubVer = ".0"; diff --git a/util.cpp b/util.cpp index 266c1dd7..f4ce5566 100644 --- a/util.cpp +++ b/util.cpp @@ -431,6 +431,35 @@ void ParseParameters(int argc, char* argv[]) } +const char* wxGetTranslation(const char* pszEnglish) +{ + // Wrapper of wxGetTranslation returning the same const char* type as was passed in + static CCriticalSection cs; + CRITICAL_BLOCK(cs) + { + // Look in cache + static map mapCache; + map::iterator mi = mapCache.find(pszEnglish); + if (mi != mapCache.end()) + return (*mi).second; + + // wxWidgets translation + const char* pszTranslated = wxGetTranslation(wxString(pszEnglish, wxConvUTF8)).utf8_str(); + if (strcmp(pszEnglish, pszTranslated) == 0) + return pszEnglish; + + // Add to cache, memory doesn't need to be freed + char* pszCached = new char[strlen(pszTranslated)+1]; + strcpy(pszCached, pszTranslated); + mapCache[pszEnglish] = pszCached; + return pszCached; + } +} + + + + + diff --git a/util.h b/util.h index 188a9ece..8aed902f 100644 --- a/util.h +++ b/util.h @@ -136,6 +136,7 @@ bool ParseMoney(const char* pszIn, int64& nRet); vector ParseHex(const char* psz); vector ParseHex(const std::string& str); void ParseParameters(int argc, char* argv[]); +const char* wxGetTranslation(const char* psz); int GetFilesize(FILE* file); void GetDataDir(char* pszDirRet); string GetDataDir(); @@ -340,11 +341,9 @@ void skipspaces(T& it) ++it; } -inline const char* wxGetTranslation(const char* psz) -{ - // Return translated UTF-8 const char* - return wxGetTranslation(wxString(psz, wxConvUTF8)).utf8_str(); -} + + + From 75199de534a590329a3b636d63fade2e4544ce97 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sat, 20 Feb 2010 21:59:59 +0000 Subject: [PATCH 056/133] Address Book with tabs instead of separate Your Address book, with live update of default address in main window, New... button on main window for creating new receiving address, made receiving address labels more visible, ask user before paying transaction fee, when sending to bitcoin address also use a bitcoin address for the change, added some event.Skip() to fix UI glitches -- version 0.2.4 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@68 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 112 +++++----- main.h | 7 +- script.h | 5 + serialize.h | 2 +- ui.cpp | 580 ++++++++++++++++++++++++-------------------------- ui.h | 62 +++--- uibase.cpp | 100 ++++++--- uibase.h | 36 +++- uiproject.fbp | 580 ++++++++++++++++++++++++++++++++++++++++---------- util.cpp | 6 +- 10 files changed, 941 insertions(+), 549 deletions(-) diff --git a/main.cpp b/main.cpp index 665a78e2..e5cc881b 100644 --- a/main.cpp +++ b/main.cpp @@ -2645,7 +2645,12 @@ void BitcoinMiner() do { pindexTmp = pindexBest; - Sleep(10000); + for (int i = 0; i < 10; i++) + { + Sleep(1000); + if (fShutdown) + return; + } } while (pindexTmp != pindexBest); } @@ -2852,10 +2857,13 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CK if (keyRet.IsNull()) keyRet.MakeNewKey(); - // Fill a vout to ourself - CScript scriptPubKey; - scriptPubKey << keyRet.GetPubKey() << OP_CHECKSIG; - wtxNew.vout.push_back(CTxOut(nValueIn - nTotalValue, scriptPubKey)); + // Fill a vout to ourself, using same address type as the payment + CScript scriptChange; + if (scriptPubKey.GetBitcoinAddressHash160() != 0) + scriptChange.SetBitcoinAddress(keyRet.GetPubKey()); + else + scriptChange << keyRet.GetPubKey() << OP_CHECKSIG; + wtxNew.vout.push_back(CTxOut(nValueIn - nTotalValue, scriptChange)); } // Fill a vout to the payee @@ -2894,42 +2902,50 @@ bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CK } // Call after CreateTransaction unless you want to abort -bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key) +bool CommitTransaction(CWalletTx& wtxNew, const CKey& key) { CRITICAL_BLOCK(cs_main) - CRITICAL_BLOCK(cs_mapWallet) { - //// old: eventually should make this transactional, never want to add a - //// transaction without marking spent transactions, although the risk of - //// interruption during this step is remote. - //// update: This matters even less now that fSpent can get corrected - //// when transactions are seen in VerifySignature. The remote chance of - //// unmarked fSpent will be handled by that. Don't need to make this - //// transactional. Pls delete this comment block later. - - // This is only to keep the database open to defeat the auto-flush for the - // duration of this scope. This is the only place where this optimization - // maybe makes sense; please don't do it anywhere else. - CWalletDB walletdb("r"); - - // Add the change's private key to wallet - if (!key.IsNull() && !AddKey(key)) - throw runtime_error("CommitTransactionSpent() : AddKey failed\n"); - - // Add tx to wallet, because if it has change it's also ours, - // otherwise just for transaction history. - AddToWallet(wtxNew); + printf("CommitTransaction:\n%s", wtxNew.ToString().c_str()); + CRITICAL_BLOCK(cs_mapWallet) + { + // This is only to keep the database open to defeat the auto-flush for the + // duration of this scope. This is the only place where this optimization + // maybe makes sense; please don't do it anywhere else. + CWalletDB walletdb("r"); + + // Add the change's private key to wallet + if (!key.IsNull() && !AddKey(key)) + throw runtime_error("CommitTransaction() : AddKey failed\n"); + + // Add tx to wallet, because if it has change it's also ours, + // otherwise just for transaction history. + AddToWallet(wtxNew); + + // Mark old coins as spent + set setCoins; + foreach(const CTxIn& txin, wtxNew.vin) + setCoins.insert(&mapWallet[txin.prevout.hash]); + foreach(CWalletTx* pcoin, setCoins) + { + pcoin->fSpent = true; + pcoin->WriteToDisk(); + vWalletUpdated.push_back(pcoin->GetHash()); + } + } + + // Track how many getdata requests our transaction gets + CRITICAL_BLOCK(cs_mapRequestCount) + mapRequestCount[wtxNew.GetHash()] = 0; - // Mark old coins as spent - set setCoins; - foreach(const CTxIn& txin, wtxNew.vin) - setCoins.insert(&mapWallet[txin.prevout.hash]); - foreach(CWalletTx* pcoin, setCoins) + // Broadcast + if (!wtxNew.AcceptTransaction()) { - pcoin->fSpent = true; - pcoin->WriteToDisk(); - vWalletUpdated.push_back(pcoin->GetHash()); + // This must not fail. The transaction has already been signed and recorded. + printf("CommitTransaction() : Error: Transaction not valid"); + return false; } + wtxNew.RelayWalletTransaction(); } MainFrameRepaint(); return true; @@ -2938,7 +2954,7 @@ bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key) -string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) +string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, bool fAskFee) { CRITICAL_BLOCK(cs_main) { @@ -2954,26 +2970,12 @@ string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) printf("SendMoney() : %s", strError.c_str()); return strError; } - if (!CommitTransactionSpent(wtxNew, key)) - { - printf("SendMoney() : Error finalizing transaction"); - return _("Error finalizing transaction"); - } - - // Track how many getdata requests our transaction gets - CRITICAL_BLOCK(cs_mapRequestCount) - mapRequestCount[wtxNew.GetHash()] = 0; - printf("SendMoney: %s\n", wtxNew.GetHash().ToString().substr(0,6).c_str()); + if (fAskFee && !ThreadSafeAskFee(nFeeRequired, _("Sending..."), NULL)) + return "ABORTED"; - // Broadcast - if (!wtxNew.AcceptTransaction()) - { - // This must not fail. The transaction has already been signed and recorded. - printf("SendMoney() : Error: Transaction not valid"); + if (!CommitTransaction(wtxNew, key)) return _("Error: The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here."); - } - wtxNew.RelayWalletTransaction(); } MainFrameRepaint(); return ""; @@ -2981,7 +2983,7 @@ string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew) -string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtxNew) +string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtxNew, bool fAskFee) { // Check amount if (nValue <= 0) @@ -2994,5 +2996,5 @@ string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtx if (!scriptPubKey.SetBitcoinAddress(strAddress)) return _("Invalid bitcoin address"); - return SendMoney(scriptPubKey, nValue, wtxNew); + return SendMoney(scriptPubKey, nValue, wtxNew, fAskFee); } diff --git a/main.h b/main.h index bcb6ec7d..4027f87e 100644 --- a/main.h +++ b/main.h @@ -67,9 +67,10 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv); bool SendMessages(CNode* pto); int64 GetBalance(); bool CreateTransaction(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, CKey& keyRet, int64& nFeeRequiredRet); -bool CommitTransactionSpent(const CWalletTx& wtxNew, const CKey& key); -string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew); -string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtxNew); +bool CommitTransaction(CWalletTx& wtxNew, const CKey& key); +bool BroadcastTransaction(CWalletTx& wtxNew); +string SendMoney(CScript scriptPubKey, int64 nValue, CWalletTx& wtxNew, bool fAskFee=false); +string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtxNew, bool fAskFee=false); void GenerateBitcoins(bool fGenerate); void ThreadBitcoinMiner(void* parg); void BitcoinMiner(); diff --git a/script.h b/script.h index 9e418891..dc47e1d8 100644 --- a/script.h +++ b/script.h @@ -580,6 +580,11 @@ public: *this << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; } + void SetBitcoinAddress(const vector& vchPubKey) + { + SetBitcoinAddress(Hash160(vchPubKey)); + } + bool SetBitcoinAddress(const string& strAddress) { this->clear(); diff --git a/serialize.h b/serialize.h index aaf3f990..23d61fe9 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 203; +static const int VERSION = 204; static const char* pszSubVer = ".0"; diff --git a/ui.cpp b/ui.cpp index 87ba19e0..9bd34e09 100644 --- a/ui.cpp +++ b/ui.cpp @@ -35,13 +35,54 @@ int fMinimizeOnClose = true; // Util // +void ExitTimeout(void* parg) +{ +#ifdef __WXMSW__ + Sleep(5000); + ExitProcess(0); +#endif +} + +void Shutdown(void* parg) +{ + static CCriticalSection cs_Shutdown; + static bool fTaken; + bool fFirstThread; + CRITICAL_BLOCK(cs_Shutdown) + { + fFirstThread = !fTaken; + fTaken = true; + } + static bool fExit; + if (fFirstThread) + { + fShutdown = true; + nTransactionsUpdated++; + DBFlush(false); + StopNode(); + DBFlush(true); + CreateThread(ExitTimeout, NULL); + Sleep(50); + printf("Bitcoin exiting\n\n"); + fExit = true; + exit(0); + } + else + { + while (!fExit) + Sleep(500); + Sleep(100); + ExitThread(0); + } +} + void HandleCtrlA(wxKeyEvent& event) { // Ctrl-a select all + event.Skip(); wxTextCtrl* textCtrl = (wxTextCtrl*)event.GetEventObject(); if (event.GetModifiers() == wxMOD_CONTROL && event.GetKeyCode() == 'A') textCtrl->SetSelection(-1, -1); - event.Skip(); } bool Is24HourTime() @@ -194,6 +235,35 @@ int ThreadSafeMessageBox(const string& message, const string& caption, int style #endif } +bool ThreadSafeAskFee(int64 nFeeRequired, const string& strCaption, wxWindow* parent) +{ + if (nFeeRequired == 0 || fDaemon) + return true; + string strMessage = strprintf( + _("This transaction is over the size limit. You can still send it for a fee of %s, " + "which goes to the nodes that process your transaction and helps to support the network. " + "Do you want to pay the fee?"), + FormatMoney(nFeeRequired).c_str()); + return (ThreadSafeMessageBox(strMessage, strCaption, wxYES_NO, parent) == wxYES); +} + +void SetDefaultReceivingAddress(const string& strAddress) +{ + // Update main window address and database + if (pframeMain == NULL) + return; + if (strAddress != pframeMain->m_textCtrlAddress->GetValue()) + { + uint160 hash160; + if (!AddressToHash160(strAddress, hash160)) + return; + if (!mapPubKeys.count(hash160)) + return; + CWalletDB().WriteDefaultKey(mapPubKeys[hash160]); + pframeMain->m_textCtrlAddress->SetValue(strAddress); + } +} + @@ -227,11 +297,6 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) fontTmp.SetFamily(wxFONTFAMILY_TELETYPE); m_staticTextBalance->SetFont(fontTmp); m_staticTextBalance->SetSize(140, 17); - // & underlines don't work on the toolbar buttons on gtk - m_toolBar->ClearTools(); - m_toolBar->AddTool(wxID_BUTTONSEND, _("Send Coins"), wxBitmap(send20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); - m_toolBar->AddTool(wxID_BUTTONRECEIVE, _("Address Book"), wxBitmap(addressbook20_xpm), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString); - m_toolBar->Realize(); // resize to fit ubuntu's huge default font dResize = 1.20; SetSize((dResize + 0.02) * GetSize().GetWidth(), 1.09 * GetSize().GetHeight()); @@ -276,47 +341,6 @@ CMainFrame::~CMainFrame() ptaskbaricon = NULL; } -void ExitTimeout(void* parg) -{ -#ifdef __WXMSW__ - Sleep(5000); - ExitProcess(0); -#endif -} - -void Shutdown(void* parg) -{ - static CCriticalSection cs_Shutdown; - static bool fTaken; - bool fFirstThread; - CRITICAL_BLOCK(cs_Shutdown) - { - fFirstThread = !fTaken; - fTaken = true; - } - static bool fExit; - if (fFirstThread) - { - fShutdown = true; - nTransactionsUpdated++; - DBFlush(false); - StopNode(); - DBFlush(true); - CreateThread(ExitTimeout, NULL); - Sleep(50); - printf("Bitcoin exiting\n\n"); - fExit = true; - exit(0); - } - else - { - while (!fExit) - Sleep(500); - Sleep(100); - ExitThread(0); - } -} - void CMainFrame::OnClose(wxCloseEvent& event) { if (fMinimizeOnClose && event.CanVeto() && !IsIconized()) @@ -335,6 +359,7 @@ void CMainFrame::OnClose(wxCloseEvent& event) void CMainFrame::OnIconize(wxIconizeEvent& event) { + event.Skip(); // Hide the task bar button when minimized. // Event is sent when the frame is minimized or restored. // wxWidgets 2.8.9 doesn't have IsIconized() so there's no way @@ -342,7 +367,7 @@ void CMainFrame::OnIconize(wxIconizeEvent& event) if (!event.Iconized()) fClosedToTray = false; #ifndef __WXMSW__ - // Tray is not reliable on Linux gnome + // Tray is not reliable on ubuntu 9.10 gnome fClosedToTray = false; #endif if (fMinimizeToTray && event.Iconized()) @@ -353,6 +378,7 @@ void CMainFrame::OnIconize(wxIconizeEvent& event) void CMainFrame::OnMouseEvents(wxMouseEvent& event) { + event.Skip(); RandAddSeed(); RAND_add(&event.m_x, sizeof(event.m_x), 0.25); RAND_add(&event.m_y, sizeof(event.m_y), 0.25); @@ -360,9 +386,11 @@ void CMainFrame::OnMouseEvents(wxMouseEvent& event) void CMainFrame::OnListColBeginDrag(wxListEvent& event) { - // Hidden columns not resizeable - if (event.GetColumn() <= 1 && !fDebug) + // Hidden columns not resizeable + if (event.GetColumn() <= 1 && !fDebug) event.Veto(); + else + event.Skip(); } int CMainFrame::GetSortIndex(const string& strSort) @@ -546,7 +574,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) if (wtx.IsCoinBase()) { - // Coinbase + // Generated strDescription = _("Generated"); if (nCredit == 0) { @@ -569,7 +597,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) } else if (!mapValue["from"].empty() || !mapValue["message"].empty()) { - // Online transaction + // Received by IP connection if (!mapValue["from"].empty()) strDescription += _("From: ") + mapValue["from"]; if (!mapValue["message"].empty()) @@ -581,7 +609,7 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) } else { - // Offline transaction + // Received by Bitcoin Address foreach(const CTxOut& txout, wtx.vout) { if (txout.IsMine()) @@ -591,20 +619,19 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) { CRITICAL_BLOCK(cs_mapAddressBook) { + //strDescription += _("Received payment to "); + //strDescription += _("Received with address "); + strDescription += _("From: unknown, Received with: "); string strAddress = PubKeyToAddress(vchPubKey); - if (mapAddressBook.count(strAddress)) + map::iterator mi = mapAddressBook.find(strAddress); + if (mi != mapAddressBook.end() && !(*mi).second.empty()) { - //strDescription += _("Received payment to "); - //strDescription += _("Received with address "); - strDescription += _("From: unknown, To: "); - strDescription += strAddress; - /// The labeling feature is just too confusing, so I hid it - /// by putting it at the end where it runs off the screen. - /// It can still be seen by widening the column, or in the - /// details dialog. - if (!mapAddressBook[strAddress].empty()) - strDescription += " (" + mapAddressBook[strAddress] + ")"; + string strLabel = (*mi).second; + strDescription += strAddress.substr(0,12) + "... "; + strDescription += "(" + strLabel + ")"; } + else + strDescription += strAddress; } } break; @@ -659,12 +686,12 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) string strAddress; if (!mapValue["to"].empty()) { - // Online transaction + // Sent to IP strAddress = mapValue["to"]; } else { - // Offline transaction + // Sent to Bitcoin Address uint160 hash160; if (ExtractHash160(txout.scriptPubKey, hash160)) strAddress = Hash160ToAddress(hash160); @@ -683,8 +710,11 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) } int64 nValue = txout.nValue; - if (nOut == 0 && nTxFee > 0) + if (nTxFee > 0) + { nValue += nTxFee; + nTxFee = 0; + } InsertLine(fNew, nIndex, hash, strprintf("%s-%d", strSort.c_str(), nOut), strStatus, @@ -846,12 +876,12 @@ void CMainFrame::RefreshStatusColumn() void CMainFrame::OnPaint(wxPaintEvent& event) { + event.Skip(); if (fRefresh) { fRefresh = false; Refresh(); } - event.Skip(); } @@ -903,6 +933,9 @@ void MainFrameRepaint() void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) { + // Skip lets the listctrl do the paint, we're just hooking the message + event.Skip(); + if (ptaskbaricon) ptaskbaricon->UpdateTooltip(); @@ -970,11 +1003,6 @@ void CMainFrame::OnPaintListCtrl(wxPaintEvent& event) if (fDebug && GetTime() - nThreadSocketHandlerHeartbeat > 60) m_statusBar->SetStatusText(" ERROR: ThreadSocketHandler has stopped", 0); - - // Pass through to listctrl to actually do the paint, we're just hooking the message - m_listCtrl->Disconnect(wxEVT_PAINT, (wxObjectEventFunction)NULL, NULL, this); - m_listCtrl->GetEventHandler()->ProcessEvent(event); - m_listCtrl->Connect(wxEVT_PAINT, wxPaintEventHandler(CMainFrame::OnPaintListCtrl), NULL, this); } @@ -1033,8 +1061,10 @@ void CMainFrame::OnUpdateUIOptionsGenerate(wxUpdateUIEvent& event) void CMainFrame::OnMenuOptionsChangeYourAddress(wxCommandEvent& event) { - // Options->Change Your Address - OnButtonChange(event); + // Options->Your Receiving Addresses + CAddressBookDialog dialog(this, "", CAddressBookDialog::RECEIVING, false); + if (!dialog.ShowModal()) + return; } void CMainFrame::OnMenuOptionsOptions(wxCommandEvent& event) @@ -1061,11 +1091,11 @@ void CMainFrame::OnButtonSend(wxCommandEvent& event) void CMainFrame::OnButtonAddressBook(wxCommandEvent& event) { // Toolbar: Address Book - CAddressBookDialog dialogAddr(this, "", false); + CAddressBookDialog dialogAddr(this, "", CAddressBookDialog::SENDING, false); if (dialogAddr.ShowModal() == 2) { // Send - CSendDialog dialogSend(this, dialogAddr.GetAddress()); + CSendDialog dialogSend(this, dialogAddr.GetSelectedAddress()); dialogSend.ShowModal(); } } @@ -1073,35 +1103,36 @@ void CMainFrame::OnButtonAddressBook(wxCommandEvent& event) void CMainFrame::OnSetFocusAddress(wxFocusEvent& event) { // Automatically select-all when entering window + event.Skip(); m_textCtrlAddress->SetSelection(-1, -1); fOnSetFocusAddress = true; - event.Skip(); } void CMainFrame::OnMouseEventsAddress(wxMouseEvent& event) { + event.Skip(); if (fOnSetFocusAddress) m_textCtrlAddress->SetSelection(-1, -1); fOnSetFocusAddress = false; - event.Skip(); } -void CMainFrame::OnButtonChange(wxCommandEvent& event) +void CMainFrame::OnButtonNew(wxCommandEvent& event) { - CYourAddressDialog dialog(this, string(m_textCtrlAddress->GetValue())); + // Ask name + CGetTextFromUserDialog dialog(this, + _("New Receiving Address"), + _("It's good policy to use a new address for each payment you receive.\n\nLabel"), + ""); if (!dialog.ShowModal()) return; - string strAddress = (string)dialog.GetAddress(); - if (strAddress != m_textCtrlAddress->GetValue()) - { - uint160 hash160; - if (!AddressToHash160(strAddress, hash160)) - return; - if (!mapPubKeys.count(hash160)) - return; - CWalletDB().WriteDefaultKey(mapPubKeys[hash160]); - m_textCtrlAddress->SetValue(strAddress); - } + string strName = dialog.GetValue(); + + // Generate new key + string strAddress = PubKeyToAddress(GenerateNewKey()); + + // Save + SetAddressBookName(strAddress, strName); + SetDefaultReceivingAddress(strAddress); } void CMainFrame::OnButtonCopy(wxCommandEvent& event) @@ -1139,7 +1170,6 @@ void CMainFrame::OnListItemActivated(wxListEvent& event) - ////////////////////////////////////////////////////////////////////////////// // // CTxDetailsDialog @@ -1452,6 +1482,7 @@ void COptionsDialog::OnListBox(wxCommandEvent& event) void COptionsDialog::OnKillFocusTransactionFee(wxFocusEvent& event) { + event.Skip(); int64 nTmp = nTransactionFee; ParseMoney(m_textCtrlTransactionFee->GetValue(), nTmp); m_textCtrlTransactionFee->SetValue(FormatMoney(nTmp)); @@ -1485,6 +1516,7 @@ CAddress COptionsDialog::GetProxyAddr() void COptionsDialog::OnKillFocusProxy(wxFocusEvent& event) { + event.Skip(); m_textCtrlProxyIP->SetValue(GetProxyAddr().ToStringIP()); m_textCtrlProxyPort->SetValue(GetProxyAddr().ToStringPort()); } @@ -1632,6 +1664,7 @@ CSendDialog::CSendDialog(wxWindow* parent, const wxString& strAddress) : CSendDi void CSendDialog::OnTextAddress(wxCommandEvent& event) { // Check mark + event.Skip(); bool fBitcoinAddress = IsValidBitcoinAddress(m_textCtrlAddress->GetValue()); m_bitmapCheckMark->Show(fBitcoinAddress); @@ -1660,6 +1693,7 @@ void CSendDialog::OnTextAddress(wxCommandEvent& event) void CSendDialog::OnKillFocusAmount(wxFocusEvent& event) { // Reformat the amount + event.Skip(); if (m_textCtrlAmount->GetValue().Trim().empty()) return; int64 nTmp; @@ -1670,9 +1704,9 @@ void CSendDialog::OnKillFocusAmount(wxFocusEvent& event) void CSendDialog::OnButtonAddressBook(wxCommandEvent& event) { // Open address book - CAddressBookDialog dialog(this, m_textCtrlAddress->GetValue(), true); + CAddressBookDialog dialog(this, m_textCtrlAddress->GetValue(), CAddressBookDialog::SENDING, true); if (dialog.ShowModal()) - m_textCtrlAddress->SetValue(dialog.GetAddress()); + m_textCtrlAddress->SetValue(dialog.GetSelectedAddress()); } void CSendDialog::OnButtonPaste(wxCommandEvent& event) @@ -1723,11 +1757,11 @@ void CSendDialog::OnButtonSend(wxCommandEvent& event) CScript scriptPubKey; scriptPubKey << OP_DUP << OP_HASH160 << hash160 << OP_EQUALVERIFY << OP_CHECKSIG; - string strError = SendMoney(scriptPubKey, nValue, wtx); - if (strError != "") - wxMessageBox(strError + " ", _("Sending...")); - else + string strError = SendMoney(scriptPubKey, nValue, wtx, true); + if (strError == "") wxMessageBox(_("Payment sent "), _("Sending...")); + else if (strError != "ABORTED") + wxMessageBox(strError + " ", _("Sending...")); } else { @@ -1846,6 +1880,7 @@ void CSendingDialog::OnButtonCancel(wxCommandEvent& event) void CSendingDialog::OnPaint(wxPaintEvent& event) { + event.Skip(); if (strlen(pszStatus) > 130) m_textCtrlStatus->SetValue(string("\n") + pszStatus); else @@ -1869,7 +1904,6 @@ void CSendingDialog::OnPaint(wxPaintEvent& event) Close(); wxMessageBox(_("Transfer cancelled "), _("Sending..."), wxOK, this); } - event.Skip(); } @@ -2016,6 +2050,13 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) return; } + // Transaction fee + if (!ThreadSafeAskFee(nFeeRequired, _("Sending..."), this)) + { + Error(_("Transaction aborted")); + return; + } + // Make sure we're still connected CNode* pnode = ConnectNode(addr, 2 * 60 * 60); if (!pnode) @@ -2040,20 +2081,15 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) return; // Commit - if (!CommitTransactionSpent(wtx, key)) + if (!CommitTransaction(wtx, key)) { - Error(_("Error finalizing payment")); + Error(_("The transaction was rejected. This might happen if some of the coins in your wallet were already spent, such as if you used a copy of wallet.dat and coins were spent in the copy but not marked as spent here.")); return; } // Send payment tx to seller, with response going to OnReply3 via event handler pnode->PushRequest("submitorder", wtx, SendingDialogOnReply3, this); - // Accept and broadcast transaction - if (!wtx.AcceptTransaction()) - printf("ERROR: CSendingDialog : wtxNew.AcceptTransaction() %s failed\n", wtx.GetHash().ToString().c_str()); - wtx.RelayWalletTransaction(); - Status(_("Waiting for confirmation...")); MainFrameRepaint(); } @@ -2097,37 +2133,54 @@ void CSendingDialog::OnReply3(CDataStream& vRecv) ////////////////////////////////////////////////////////////////////////////// // -// CYourAddressDialog +// CAddressBookDialog // -CYourAddressDialog::CYourAddressDialog(wxWindow* parent, const string& strInitSelected) : CYourAddressDialogBase(parent) +CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInitSelected, int nPageIn, bool fDuringSendIn) : CAddressBookDialogBase(parent) { + // Set initially selected page + wxNotebookEvent event; + event.SetSelection(nPageIn); + OnNotebookPageChanged(event); + m_notebook->ChangeSelection(nPageIn); + + fDuringSend = fDuringSendIn; + if (!fDuringSend) + m_buttonCancel->Show(false); + + // Set Icon + wxIcon iconAddressBook; + iconAddressBook.CopyFromBitmap(wxBitmap(addressbook16_xpm)); + SetIcon(iconAddressBook); + // Init column headers - m_listCtrl->InsertColumn(0, _("Label"), wxLIST_FORMAT_LEFT, 200); - m_listCtrl->InsertColumn(1, _("Bitcoin Address"), wxLIST_FORMAT_LEFT, 350); - m_listCtrl->SetFocus(); + m_listCtrlSending->InsertColumn(0, _("Name"), wxLIST_FORMAT_LEFT, 200); + m_listCtrlSending->InsertColumn(1, _("Address"), wxLIST_FORMAT_LEFT, 350); + m_listCtrlSending->SetFocus(); + m_listCtrlReceiving->InsertColumn(0, _("Label"), wxLIST_FORMAT_LEFT, 200); + m_listCtrlReceiving->InsertColumn(1, _("Bitcoin Address"), wxLIST_FORMAT_LEFT, 350); + m_listCtrlReceiving->SetFocus(); // Fill listctrl with address book data CRITICAL_BLOCK(cs_mapKeys) CRITICAL_BLOCK(cs_mapAddressBook) { + string strDefaultReceiving = (string)pframeMain->m_textCtrlAddress->GetValue(); foreach(const PAIRTYPE(string, string)& item, mapAddressBook) { string strAddress = item.first; string strName = item.second; uint160 hash160; bool fMine = (AddressToHash160(strAddress, hash160) && mapPubKeys.count(hash160)); - if (fMine) - { - int nIndex = InsertLine(m_listCtrl, strName, strAddress); - if (strAddress == strInitSelected) - m_listCtrl->SetItemState(nIndex, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED); - } + wxListCtrl* plistCtrl = fMine ? m_listCtrlReceiving : m_listCtrlSending; + int nIndex = InsertLine(plistCtrl, strName, strAddress); + if (strAddress == (fMine ? strDefaultReceiving : strInitSelected)) + plistCtrl->SetItemState(nIndex, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED); } } } -wxString CYourAddressDialog::GetAddress() +wxString CAddressBookDialog::GetSelectedAddress() { int nIndex = GetSelection(m_listCtrl); if (nIndex == -1) @@ -2135,172 +2188,92 @@ wxString CYourAddressDialog::GetAddress() return GetItemText(m_listCtrl, nIndex, 1); } -void CYourAddressDialog::OnListEndLabelEdit(wxListEvent& event) +wxString CAddressBookDialog::GetSelectedSendingAddress() { - // Update address book with edited name - if (event.IsEditCancelled()) - return; - string strAddress = (string)GetItemText(m_listCtrl, event.GetIndex(), 1); - SetAddressBookName(strAddress, string(event.GetText())); - pframeMain->RefreshListCtrl(); + int nIndex = GetSelection(m_listCtrlSending); + if (nIndex == -1) + return ""; + return GetItemText(m_listCtrlSending, nIndex, 1); } -void CYourAddressDialog::OnListItemSelected(wxListEvent& event) +wxString CAddressBookDialog::GetSelectedReceivingAddress() { + int nIndex = GetSelection(m_listCtrlReceiving); + if (nIndex == -1) + return ""; + return GetItemText(m_listCtrlReceiving, nIndex, 1); } -void CYourAddressDialog::OnListItemActivated(wxListEvent& event) +void CAddressBookDialog::OnNotebookPageChanged(wxNotebookEvent& event) { - // Doubleclick edits item - wxCommandEvent event2; - OnButtonRename(event2); + event.Skip(); + nPage = event.GetSelection(); + if (nPage == SENDING) + m_listCtrl = m_listCtrlSending; + else if (nPage == RECEIVING) + m_listCtrl = m_listCtrlReceiving; + m_buttonDelete->Show(nPage == SENDING); + m_buttonCopy->Show(nPage == RECEIVING); + this->Layout(); + m_listCtrl->SetFocus(); } -void CYourAddressDialog::OnButtonRename(wxCommandEvent& event) +void CAddressBookDialog::OnListEndLabelEdit(wxListEvent& event) { - // Ask new name - int nIndex = GetSelection(m_listCtrl); - if (nIndex == -1) - return; - string strName = (string)m_listCtrl->GetItemText(nIndex); - string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); - CGetTextFromUserDialog dialog(this, _("Edit Address Label"), _("New Label"), strName); - if (!dialog.ShowModal()) + // Update address book with edited name + event.Skip(); + if (event.IsEditCancelled()) return; - strName = dialog.GetValue(); - - // Change name - SetAddressBookName(strAddress, strName); - m_listCtrl->SetItemText(nIndex, strName); + string strAddress = (string)GetItemText(m_listCtrl, event.GetIndex(), 1); + SetAddressBookName(strAddress, string(event.GetText())); pframeMain->RefreshListCtrl(); } -void CYourAddressDialog::OnButtonNew(wxCommandEvent& event) +void CAddressBookDialog::OnListItemSelected(wxListEvent& event) { - // Ask name - CGetTextFromUserDialog dialog(this, _("New Bitcoin Address"), _("Label"), ""); - if (!dialog.ShowModal()) - return; - string strName = dialog.GetValue(); - - // Generate new key - string strAddress = PubKeyToAddress(GenerateNewKey()); - SetAddressBookName(strAddress, strName); - - // Add to list and select it - int nIndex = InsertLine(m_listCtrl, strName, strAddress); - SetSelection(m_listCtrl, nIndex); - m_listCtrl->SetFocus(); + event.Skip(); + if (nPage == RECEIVING) + SetDefaultReceivingAddress((string)GetSelectedReceivingAddress()); } -void CYourAddressDialog::OnButtonCopy(wxCommandEvent& event) +void CAddressBookDialog::OnListItemActivated(wxListEvent& event) { - // Copy address box to clipboard - if (wxTheClipboard->Open()) + event.Skip(); + if (fDuringSend) { - wxTheClipboard->SetData(new wxTextDataObject(GetAddress())); - wxTheClipboard->Close(); + // Doubleclick returns selection + EndModal(GetSelectedAddress() != "" ? 2 : 0); + return; } -} -void CYourAddressDialog::OnButtonOK(wxCommandEvent& event) -{ - // OK - EndModal(true); -} - -void CYourAddressDialog::OnButtonCancel(wxCommandEvent& event) -{ - // Cancel - EndModal(false); -} - -void CYourAddressDialog::OnClose(wxCloseEvent& event) -{ - // Close - EndModal(false); + // Doubleclick edits item + wxCommandEvent event2; + OnButtonEdit(event2); } - - - - - -////////////////////////////////////////////////////////////////////////////// -// -// CAddressBookDialog -// - -CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInitSelected, bool fSendingIn) : CAddressBookDialogBase(parent) +void CAddressBookDialog::OnButtonDelete(wxCommandEvent& event) { - fSending = fSendingIn; - if (!fSending) - m_buttonCancel->Show(false); - - // Init column headers - m_listCtrl->InsertColumn(0, _("Name"), wxLIST_FORMAT_LEFT, 200); - m_listCtrl->InsertColumn(1, _("Address"), wxLIST_FORMAT_LEFT, 350); - m_listCtrl->SetFocus(); - - // Set Icon - wxIcon iconAddressBook; - iconAddressBook.CopyFromBitmap(wxBitmap(addressbook16_xpm)); - SetIcon(iconAddressBook); - - // Fill listctrl with address book data - CRITICAL_BLOCK(cs_mapKeys) - CRITICAL_BLOCK(cs_mapAddressBook) + if (nPage != SENDING) + return; + for (int nIndex = m_listCtrl->GetItemCount()-1; nIndex >= 0; nIndex--) { - foreach(const PAIRTYPE(string, string)& item, mapAddressBook) + if (m_listCtrl->GetItemState(nIndex, wxLIST_STATE_SELECTED)) { - string strAddress = item.first; - string strName = item.second; - uint160 hash160; - bool fMine = (AddressToHash160(strAddress, hash160) && mapPubKeys.count(hash160)); - if (!fMine) - { - int nIndex = InsertLine(m_listCtrl, strName, strAddress); - if (strAddress == strInitSelected) - m_listCtrl->SetItemState(nIndex, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED); - } + string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); + CWalletDB().EraseName(strAddress); + m_listCtrl->DeleteItem(nIndex); } } -} - -wxString CAddressBookDialog::GetAddress() -{ - int nIndex = GetSelection(m_listCtrl); - if (nIndex == -1) - return ""; - return GetItemText(m_listCtrl, nIndex, 1); -} - -void CAddressBookDialog::OnListEndLabelEdit(wxListEvent& event) -{ - // Update address book with edited name - if (event.IsEditCancelled()) - return; - string strAddress = (string)GetItemText(m_listCtrl, event.GetIndex(), 1); - SetAddressBookName(strAddress, string(event.GetText())); pframeMain->RefreshListCtrl(); } -void CAddressBookDialog::OnListItemSelected(wxListEvent& event) -{ -} - -void CAddressBookDialog::OnListItemActivated(wxListEvent& event) +void CAddressBookDialog::OnButtonCopy(wxCommandEvent& event) { - if (fSending) - { - // Doubleclick returns selection - EndModal(GetAddress() != "" ? 2 : 0); - } - else + // Copy address box to clipboard + if (wxTheClipboard->Open()) { - // Doubleclick edits item - wxCommandEvent event2; - OnButtonEdit(event2); + wxTheClipboard->SetData(new wxTextDataObject(GetSelectedAddress())); + wxTheClipboard->Close(); } } @@ -2315,24 +2288,37 @@ bool CAddressBookDialog::CheckIfMine(const string& strAddress, const string& str void CAddressBookDialog::OnButtonEdit(wxCommandEvent& event) { - // Ask new name int nIndex = GetSelection(m_listCtrl); if (nIndex == -1) return; string strName = (string)m_listCtrl->GetItemText(nIndex); string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); string strAddressOrg = strAddress; - do + + if (nPage == SENDING) { - CGetTextFromUserDialog dialog(this, _("Edit Address"), _("Name"), strName, _("Address"), strAddress); + // Ask name and address + do + { + CGetTextFromUserDialog dialog(this, _("Edit Address"), _("Name"), strName, _("Address"), strAddress); + if (!dialog.ShowModal()) + return; + strName = dialog.GetValue1(); + strAddress = dialog.GetValue2(); + } + while (CheckIfMine(strAddress, _("Edit Address"))); + + } + else if (nPage == RECEIVING) + { + // Ask name + CGetTextFromUserDialog dialog(this, _("Edit Address Label"), _("Label"), strName); if (!dialog.ShowModal()) return; - strName = dialog.GetValue1(); - strAddress = dialog.GetValue2(); + strName = dialog.GetValue(); } - while (CheckIfMine(strAddress, _("Edit Address"))); - // Change name + // Write back if (strAddress != strAddressOrg) CWalletDB().EraseName(strAddressOrg); SetAddressBookName(strAddress, strName); @@ -2343,55 +2329,50 @@ void CAddressBookDialog::OnButtonEdit(wxCommandEvent& event) void CAddressBookDialog::OnButtonNew(wxCommandEvent& event) { - // Ask name string strName; string strAddress; - do + + if (nPage == SENDING) { - CGetTextFromUserDialog dialog(this, _("New Address"), _("Name"), strName, _("Address"), strAddress); + // Ask name and address + do + { + CGetTextFromUserDialog dialog(this, _("Add Address"), _("Name"), strName, _("Address"), strAddress); + if (!dialog.ShowModal()) + return; + strName = dialog.GetValue1(); + strAddress = dialog.GetValue2(); + } + while (CheckIfMine(strAddress, _("Add Address"))); + } + else if (nPage == RECEIVING) + { + // Ask name + CGetTextFromUserDialog dialog(this, + _("New Receiving Address"), + _("It's good policy to use a new address for each payment you receive.\n\nLabel"), + ""); if (!dialog.ShowModal()) return; - strName = dialog.GetValue1(); - strAddress = dialog.GetValue2(); + strName = dialog.GetValue(); + + // Generate new key + strAddress = PubKeyToAddress(GenerateNewKey()); } - while (CheckIfMine(strAddress, _("New Address"))); // Add to list and select it SetAddressBookName(strAddress, strName); int nIndex = InsertLine(m_listCtrl, strName, strAddress); SetSelection(m_listCtrl, nIndex); m_listCtrl->SetFocus(); - pframeMain->RefreshListCtrl(); -} - -void CAddressBookDialog::OnButtonDelete(wxCommandEvent& event) -{ - for (int nIndex = m_listCtrl->GetItemCount()-1; nIndex >= 0; nIndex--) - { - if (m_listCtrl->GetItemState(nIndex, wxLIST_STATE_SELECTED)) - { - string strAddress = (string)GetItemText(m_listCtrl, nIndex, 1); - CWalletDB().EraseName(strAddress); - m_listCtrl->DeleteItem(nIndex); - } - } - pframeMain->RefreshListCtrl(); -} - -void CAddressBookDialog::OnButtonCopy(wxCommandEvent& event) -{ - // Copy address box to clipboard - if (wxTheClipboard->Open()) - { - wxTheClipboard->SetData(new wxTextDataObject(GetAddress())); - wxTheClipboard->Close(); - } + if (nPage == SENDING) + pframeMain->RefreshListCtrl(); } void CAddressBookDialog::OnButtonOK(wxCommandEvent& event) { // OK - EndModal(GetAddress() != "" ? 1 : 0); + EndModal(GetSelectedAddress() != "" ? 1 : 0); } void CAddressBookDialog::OnButtonCancel(wxCommandEvent& event) @@ -2644,14 +2625,14 @@ bool CMyApp::OnInit2() wxString strUsage = string() + _("Usage: bitcoin [options]") + "\t\t\t\t\t\t\n" + _("Options:\n") + - " -gen \t\t " + _("Generate coins\n") + - " -gen=0 \t\t " + _("Don't generate coins\n") + - " -min \t\t " + _("Start minimized\n") + - " -datadir= \t " + _("Specify data directory\n") + - " -proxy=\t " + _("Connect through socks4 proxy\n") + - " -addnode= \t " + _("Add a node to connect to\n") + - " -connect= \t " + _("Connect only to the specified node\n") + - " -? \t\t " + _("This help message\n"); + " -gen \t\t " + _("Generate coins\n") + + " -gen=0 \t\t " + _("Don't generate coins\n") + + " -min \t\t " + _("Start minimized\n") + + " -datadir= \t " + _("Specify data directory\n") + + " -proxy=\t " + _("Connect through socks4 proxy\n") + + " -addnode= \t " + _("Add a node to connect to\n") + + " -connect= \t " + _("Connect only to the specified node\n") + + " -? \t\t " + _("This help message\n"); if (fWindows) { @@ -2947,7 +2928,6 @@ bool CMyApp::OnExceptionInMainLoop() Sleep(1000); throw; } - return true; } diff --git a/ui.h b/ui.h index 50d39ab7..317b87e9 100644 --- a/ui.h +++ b/ui.h @@ -2,11 +2,9 @@ // Distributed under the MIT/X11 software license, see the accompanying // file license.txt or http://www.opensource.org/licenses/mit-license.php. - - - DECLARE_EVENT_TYPE(wxEVT_UITHREADCALL, -1) + extern map mapArgs; // Settings @@ -22,6 +20,7 @@ void UIThreadCall(boost::function0); void MainFrameRepaint(); void Shutdown(void* parg); int ThreadSafeMessageBox(const string& message, const string& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1); +bool ThreadSafeAskFee(int64 nFeeRequired, const string& strCaption, wxWindow* parent); @@ -51,7 +50,7 @@ protected: void OnButtonAddressBook(wxCommandEvent& event); void OnSetFocusAddress(wxFocusEvent& event); void OnMouseEventsAddress(wxMouseEvent& event); - void OnButtonChange(wxCommandEvent& event); + void OnButtonNew(wxCommandEvent& event); void OnButtonCopy(wxCommandEvent& event); void OnListColBeginDrag(wxListEvent& event); void OnListItemActivated(wxListEvent& event); @@ -205,53 +204,39 @@ void SendingDialogOnReply3(void* parg, CDataStream& vRecv); -class CYourAddressDialog : public CYourAddressDialogBase -{ -protected: - // Event handlers - void OnListEndLabelEdit(wxListEvent& event); - void OnListItemSelected(wxListEvent& event); - void OnListItemActivated(wxListEvent& event); - void OnButtonRename(wxCommandEvent& event); - void OnButtonNew(wxCommandEvent& event); - void OnButtonCopy(wxCommandEvent& event); - void OnButtonOK(wxCommandEvent& event); - void OnButtonCancel(wxCommandEvent& event); - void OnClose(wxCloseEvent& event); - -public: - /** Constructor */ - CYourAddressDialog(wxWindow* parent); - CYourAddressDialog(wxWindow* parent, const string& strInitSelected); - - // Custom - wxString GetAddress(); -}; - - - class CAddressBookDialog : public CAddressBookDialogBase { protected: // Event handlers + void OnNotebookPageChanged(wxNotebookEvent& event); void OnListEndLabelEdit(wxListEvent& event); void OnListItemSelected(wxListEvent& event); void OnListItemActivated(wxListEvent& event); - void OnButtonEdit(wxCommandEvent& event); void OnButtonDelete(wxCommandEvent& event); - void OnButtonNew(wxCommandEvent& event); void OnButtonCopy(wxCommandEvent& event); + void OnButtonEdit(wxCommandEvent& event); + void OnButtonNew(wxCommandEvent& event); void OnButtonOK(wxCommandEvent& event); void OnButtonCancel(wxCommandEvent& event); void OnClose(wxCloseEvent& event); public: /** Constructor */ - CAddressBookDialog(wxWindow* parent, const wxString& strInitSelected, bool fSendingIn); + CAddressBookDialog(wxWindow* parent, const wxString& strInitSelected, int nPageIn, bool fDuringSendIn); // Custom - bool fSending; + enum + { + SENDING = 0, + RECEIVING = 1, + }; + int nPage; + wxListCtrl* m_listCtrl; + bool fDuringSend; wxString GetAddress(); + wxString GetSelectedAddress(); + wxString GetSelectedSendingAddress(); + wxString GetSelectedReceivingAddress(); bool CheckIfMine(const string& strAddress, const string& strTitle); }; @@ -282,18 +267,25 @@ public: const string& strMessage2="", const string& strValue2="") : CGetTextFromUserDialogBase(parent, wxID_ANY, strCaption) { + int x = GetSize().GetWidth(); + int y = GetSize().GetHeight(); m_staticTextMessage1->SetLabel(strMessage1); m_textCtrl1->SetValue(strValue1); + y += wxString(strMessage1).Freq('\n') * 14; if (!strMessage2.empty()) { m_staticTextMessage2->Show(true); m_staticTextMessage2->SetLabel(strMessage2); m_textCtrl2->Show(true); m_textCtrl2->SetValue(strValue2); - SetSize(wxDefaultCoord, 180); + y += 46 + wxString(strMessage2).Freq('\n') * 14; } if (!fWindows) - SetSize(1.14 * GetSize().GetWidth(), 1.14 * GetSize().GetHeight()); + { + x *= 1.14; + y *= 1.14; + } + SetSize(x, y); } // Custom diff --git a/uibase.cpp b/uibase.cpp index 5b3f9abc..340c7abc 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -42,14 +42,14 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_menuOptions->Append( m_menuOptionsGenerateBitcoins ); wxMenuItem* m_menuOptionsChangeYourAddress; - m_menuOptionsChangeYourAddress = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( _("&Change Your Address...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptionsChangeYourAddress = new wxMenuItem( m_menuOptions, wxID_ANY, wxString( _("&Your Receiving Addresses...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuOptions->Append( m_menuOptionsChangeYourAddress ); wxMenuItem* m_menuOptionsOptions; m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_MENUOPTIONSOPTIONS, wxString( _("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuOptions->Append( m_menuOptionsOptions ); - m_menubar->Append( m_menuOptions, _("&Options") ); + m_menubar->Append( m_menuOptions, _("&Settings") ); m_menuHelp = new wxMenu(); wxMenuItem* m_menuHelpAbout; @@ -65,8 +65,8 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_toolBar->SetToolSeparation( 1 ); m_toolBar->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90, false, wxEmptyString ) ); - m_toolBar->AddTool( wxID_BUTTONSEND, _("&Send Coins"), wxBitmap( send20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); - m_toolBar->AddTool( wxID_BUTTONRECEIVE, _("&Address Book"), wxBitmap( addressbook20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->AddTool( wxID_BUTTONSEND, _("Send Coins"), wxBitmap( send20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); + m_toolBar->AddTool( wxID_BUTTONRECEIVE, _("Address Book"), wxBitmap( addressbook20_xpm ), wxNullBitmap, wxITEM_NORMAL, wxEmptyString, wxEmptyString ); m_toolBar->Realize(); m_statusBar = this->CreateStatusBar( 1, wxST_SIZEGRIP, wxID_ANY ); @@ -86,14 +86,10 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& bSizer85->Add( m_staticText32, 0, wxALIGN_CENTER_VERTICAL|wxLEFT, 5 ); m_textCtrlAddress = new wxTextCtrl( this, wxID_TEXTCTRLADDRESS, wxEmptyString, wxDefaultPosition, wxSize( 340,-1 ), wxTE_READONLY ); - m_textCtrlAddress->SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_MENU ) ); - bSizer85->Add( m_textCtrlAddress, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT, 5 ); - m_buttonNew = new wxButton( this, wxID_BUTTONCHANGE, _("&New..."), wxDefaultPosition, wxDefaultSize, 0 ); - m_buttonNew->Hide(); - - bSizer85->Add( m_buttonNew, 0, wxRIGHT, 5 ); + m_buttonNew = new wxButton( this, wxID_BUTTONNEW, _(" &New... "), wxDefaultPosition, wxSize( -1,-1 ), wxBU_EXACTFIT ); + bSizer85->Add( m_buttonNew, 0, wxRIGHT|wxALIGN_CENTER_VERTICAL, 5 ); m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, _(" &Copy to Clipboard "), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT ); bSizer85->Add( m_buttonCopy, 0, wxALIGN_CENTER_VERTICAL|wxRIGHT, 5 ); @@ -188,7 +184,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_textCtrlAddress->Connect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Connect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); - m_buttonNew->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); + m_buttonNew->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonNew ), NULL, this ); m_buttonCopy->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); m_listCtrl->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); @@ -240,7 +236,7 @@ CMainFrameBase::~CMainFrameBase() m_textCtrlAddress->Disconnect( wxEVT_ENTER_WINDOW, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEventsAddress ), NULL, this ); m_textCtrlAddress->Disconnect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); - m_buttonNew->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonChange ), NULL, this ); + m_buttonNew->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonNew ), NULL, this ); m_buttonCopy->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); @@ -844,20 +840,53 @@ CAddressBookDialogBase::CAddressBookDialogBase( wxWindow* parent, wxWindowID id, { this->SetSizeHints( wxDefaultSize, wxDefaultSize ); + wxBoxSizer* bSizer58; + bSizer58 = new wxBoxSizer( wxVERTICAL ); + + m_notebook = new wxNotebook( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 ); + m_panelSending = new wxPanel( m_notebook, wxID_PANELSENDING, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); wxBoxSizer* bSizer68; bSizer68 = new wxBoxSizer( wxVERTICAL ); - bSizer68->Add( 0, 5, 0, wxEXPAND, 5 ); + bSizer68->Add( 0, 0, 0, wxEXPAND, 5 ); - m_staticText55 = new wxStaticText( this, wxID_ANY, _("Bitcoin Address"), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText55 = new wxStaticText( m_panelSending, wxID_ANY, _("Bitcoin Address"), wxDefaultPosition, wxDefaultSize, 0 ); m_staticText55->Wrap( -1 ); m_staticText55->Hide(); bSizer68->Add( m_staticText55, 0, wxTOP|wxRIGHT|wxLEFT, 5 ); - m_listCtrl = new wxListCtrl( this, wxID_LISTCTRL, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING ); - bSizer68->Add( m_listCtrl, 1, wxALL|wxEXPAND, 5 ); + m_listCtrlSending = new wxListCtrl( m_panelSending, wxID_LISTCTRLSENDING, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING ); + bSizer68->Add( m_listCtrlSending, 1, wxALL|wxEXPAND, 5 ); + + m_panelSending->SetSizer( bSizer68 ); + m_panelSending->Layout(); + bSizer68->Fit( m_panelSending ); + m_notebook->AddPage( m_panelSending, _("Sending"), false ); + m_panelReceiving = new wxPanel( m_notebook, wxID_PANELRECEIVING, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer681; + bSizer681 = new wxBoxSizer( wxVERTICAL ); + + + bSizer681->Add( 0, 0, 0, wxEXPAND, 5 ); + + m_staticText45 = new wxStaticText( m_panelReceiving, wxID_ANY, _("These are your Bitcoin addresses for receiving payments. You can give a different one to each sender to keep track of who is paying you. The highlighted address will be displayed in the main window."), wxDefaultPosition, wxDefaultSize, 0 ); + m_staticText45->Wrap( 570 ); + bSizer681->Add( m_staticText45, 0, wxTOP|wxRIGHT|wxLEFT, 6 ); + + + bSizer681->Add( 0, 2, 0, wxEXPAND, 5 ); + + m_listCtrlReceiving = new wxListCtrl( m_panelReceiving, wxID_LISTCTRLRECEIVING, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING ); + bSizer681->Add( m_listCtrlReceiving, 1, wxALL|wxEXPAND, 5 ); + + m_panelReceiving->SetSizer( bSizer681 ); + m_panelReceiving->Layout(); + bSizer681->Fit( m_panelReceiving ); + m_notebook->AddPage( m_panelReceiving, _("Receiving"), true ); + + bSizer58->Add( m_notebook, 1, wxEXPAND|wxTOP|wxRIGHT|wxLEFT, 5 ); wxBoxSizer* bSizer69; bSizer69 = new wxBoxSizer( wxHORIZONTAL ); @@ -865,34 +894,42 @@ CAddressBookDialogBase::CAddressBookDialogBase( wxWindow* parent, wxWindowID id, bSizer69->Add( 0, 0, 1, wxEXPAND, 5 ); + m_buttonDelete = new wxButton( this, wxID_BUTTONDELETE, _("&Delete"), wxDefaultPosition, wxDefaultSize, 0 ); + bSizer69->Add( m_buttonDelete, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); + + m_buttonCopy = new wxButton( this, wxID_BUTTONCOPY, _(" &Copy to Clipboard "), wxDefaultPosition, wxSize( -1,-1 ), 0 ); + bSizer69->Add( m_buttonCopy, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); + m_buttonEdit = new wxButton( this, wxID_BUTTONEDIT, _("&Edit..."), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_buttonEdit, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonNew = new wxButton( this, wxID_BUTTONNEW, _(" &New Address... "), wxDefaultPosition, wxDefaultSize, 0 ); bSizer69->Add( m_buttonNew, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonDelete = new wxButton( this, wxID_BUTTONDELETE, _("&Delete"), wxDefaultPosition, wxDefaultSize, 0 ); - bSizer69->Add( m_buttonDelete, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - m_buttonOK = new wxButton( this, wxID_OK, _("OK"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer69->Add( m_buttonOK, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); m_buttonCancel = new wxButton( this, wxID_CANCEL, _("Cancel"), wxDefaultPosition, wxSize( -1,-1 ), 0 ); bSizer69->Add( m_buttonCancel, 0, wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND, 5 ); - bSizer68->Add( bSizer69, 0, wxEXPAND, 5 ); + bSizer58->Add( bSizer69, 0, wxEXPAND, 5 ); - this->SetSizer( bSizer68 ); + this->SetSizer( bSizer58 ); this->Layout(); // Connect Events this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CAddressBookDialogBase::OnClose ) ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_notebook->Connect( wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGED, wxNotebookEventHandler( CAddressBookDialogBase::OnNotebookPageChanged ), NULL, this ); + m_listCtrlSending->Connect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrlSending->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrlSending->Connect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_listCtrlReceiving->Connect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrlReceiving->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrlReceiving->Connect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_buttonDelete->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonDelete ), NULL, this ); + m_buttonCopy->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonCopy ), NULL, this ); m_buttonEdit->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonEdit ), NULL, this ); m_buttonNew->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonNew ), NULL, this ); - m_buttonDelete->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonDelete ), NULL, this ); m_buttonOK->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonOK ), NULL, this ); m_buttonCancel->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonCancel ), NULL, this ); } @@ -901,12 +938,17 @@ CAddressBookDialogBase::~CAddressBookDialogBase() { // Disconnect Events this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( CAddressBookDialogBase::OnClose ) ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_notebook->Disconnect( wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGED, wxNotebookEventHandler( CAddressBookDialogBase::OnNotebookPageChanged ), NULL, this ); + m_listCtrlSending->Disconnect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrlSending->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrlSending->Disconnect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_listCtrlReceiving->Disconnect( wxEVT_COMMAND_LIST_END_LABEL_EDIT, wxListEventHandler( CAddressBookDialogBase::OnListEndLabelEdit ), NULL, this ); + m_listCtrlReceiving->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CAddressBookDialogBase::OnListItemActivated ), NULL, this ); + m_listCtrlReceiving->Disconnect( wxEVT_COMMAND_LIST_ITEM_SELECTED, wxListEventHandler( CAddressBookDialogBase::OnListItemSelected ), NULL, this ); + m_buttonDelete->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonDelete ), NULL, this ); + m_buttonCopy->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonCopy ), NULL, this ); m_buttonEdit->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonEdit ), NULL, this ); m_buttonNew->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonNew ), NULL, this ); - m_buttonDelete->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonDelete ), NULL, this ); m_buttonOK->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonOK ), NULL, this ); m_buttonCancel->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CAddressBookDialogBase::OnButtonCancel ), NULL, this ); } diff --git a/uibase.h b/uibase.h index 494ed10f..90abd00f 100644 --- a/uibase.h +++ b/uibase.h @@ -36,6 +36,7 @@ #include #include #include +#include /////////////////////////////////////////////////////////////////////////// @@ -46,7 +47,7 @@ #define wxID_BUTTONSEND 1004 #define wxID_BUTTONRECEIVE 1005 #define wxID_TEXTCTRLADDRESS 1006 -#define wxID_BUTTONCHANGE 1007 +#define wxID_BUTTONNEW 1007 #define wxID_BUTTONCOPY 1008 #define wxID_TRANSACTIONFEE 1009 #define wxID_PROXYIP 1010 @@ -58,10 +59,13 @@ #define wxID_CHOICETRANSFERTYPE 1016 #define wxID_LISTCTRL 1017 #define wxID_BUTTONRENAME 1018 -#define wxID_BUTTONNEW 1019 -#define wxID_BUTTONEDIT 1020 -#define wxID_BUTTONDELETE 1021 -#define wxID_TEXTCTRL 1022 +#define wxID_PANELSENDING 1019 +#define wxID_LISTCTRLSENDING 1020 +#define wxID_PANELRECEIVING 1021 +#define wxID_LISTCTRLRECEIVING 1022 +#define wxID_BUTTONDELETE 1023 +#define wxID_BUTTONEDIT 1024 +#define wxID_TEXTCTRL 1025 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase @@ -79,7 +83,6 @@ class CMainFrameBase : public wxFrame wxStatusBar* m_statusBar; wxStaticText* m_staticText32; - wxTextCtrl* m_textCtrlAddress; wxButton* m_buttonNew; wxButton* m_buttonCopy; @@ -108,7 +111,7 @@ class CMainFrameBase : public wxFrame virtual void OnKeyDown( wxKeyEvent& event ){ event.Skip(); } virtual void OnMouseEventsAddress( wxMouseEvent& event ){ event.Skip(); } virtual void OnSetFocusAddress( wxFocusEvent& event ){ event.Skip(); } - virtual void OnButtonChange( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonNew( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } virtual void OnListColBeginDrag( wxListEvent& event ){ event.Skip(); } virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } @@ -117,6 +120,7 @@ class CMainFrameBase : public wxFrame public: wxMenu* m_menuOptions; + wxTextCtrl* m_textCtrlAddress; wxListCtrl* m_listCtrl; CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = _("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 712,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); ~CMainFrameBase(); @@ -343,23 +347,33 @@ class CAddressBookDialogBase : public wxDialog private: protected: + wxNotebook* m_notebook; + wxPanel* m_panelSending; wxStaticText* m_staticText55; - wxListCtrl* m_listCtrl; + wxListCtrl* m_listCtrlSending; + wxPanel* m_panelReceiving; + + wxStaticText* m_staticText45; + + wxListCtrl* m_listCtrlReceiving; + wxButton* m_buttonDelete; + wxButton* m_buttonCopy; wxButton* m_buttonEdit; wxButton* m_buttonNew; - wxButton* m_buttonDelete; wxButton* m_buttonOK; // Virtual event handlers, overide them in your derived class virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } + virtual void OnNotebookPageChanged( wxNotebookEvent& event ){ event.Skip(); } virtual void OnListEndLabelEdit( wxListEvent& event ){ event.Skip(); } virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } virtual void OnListItemSelected( wxListEvent& event ){ event.Skip(); } + virtual void OnButtonDelete( wxCommandEvent& event ){ event.Skip(); } + virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonEdit( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonNew( wxCommandEvent& event ){ event.Skip(); } - virtual void OnButtonDelete( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonOK( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonCancel( wxCommandEvent& event ){ event.Skip(); } @@ -397,7 +411,7 @@ class CGetTextFromUserDialogBase : public wxDialog public: - CGetTextFromUserDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxEmptyString, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 403,138 ), long style = wxDEFAULT_DIALOG_STYLE ); + CGetTextFromUserDialogBase( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = wxEmptyString, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 440,138 ), long style = wxDEFAULT_DIALOG_STYLE ); ~CGetTextFromUserDialogBase(); }; diff --git a/uiproject.fbp b/uiproject.fbp index 988df2a9..113a1579 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -155,7 +155,7 @@
- &Options + &Settings m_menuOptions public @@ -180,7 +180,7 @@ wxID_ANY wxITEM_NORMAL - &Change Your Address... + &Your Receiving Addresses... m_menuOptionsChangeYourAddress none @@ -276,7 +276,7 @@ xpm/send20.xpm; Load From File wxID_BUTTONSEND wxITEM_NORMAL - &Send Coins + Send Coins m_tool1 @@ -290,7 +290,7 @@ xpm/addressbook20.xpm; Load From File wxID_BUTTONRECEIVE wxITEM_NORMAL - &Address Book + Address Book m_tool2 @@ -426,7 +426,7 @@ wxALIGN_CENTER_VERTICAL|wxRIGHT|wxLEFT 0 - wxSYS_COLOUR_MENU + 1 @@ -437,7 +437,7 @@ 0 -1,-1 m_textCtrlAddress - protected + public 340,-1 wxTE_READONLY @@ -478,7 +478,7 @@ 5 - wxRIGHT + wxRIGHT|wxALIGN_CENTER_VERTICAL 0 @@ -487,22 +487,22 @@ 1 - 1 - wxID_BUTTONCHANGE - &New... + 0 + wxID_BUTTONNEW + &New... m_buttonNew protected - - + -1,-1 + wxBU_EXACTFIT - OnButtonChange + OnButtonNew @@ -4388,7 +4388,7 @@ 5 wxEXPAND 0 - + bSizer69 wxHORIZONTAL @@ -4721,89 +4721,29 @@ - bSizer68 + bSizer58 wxVERTICAL none 5 - wxEXPAND - 0 - - 5 - protected - 0 - - - - 5 - wxTOP|wxRIGHT|wxLEFT - 0 - - - - 1 - - - 1 - wxID_ANY - Bitcoin Address - - - m_staticText55 - protected - - - - - - - - - -1 - - - - - - - - - - - - - - - - - - - - - - - - - - - 5 - wxALL|wxEXPAND + wxEXPAND|wxTOP|wxRIGHT|wxLEFT 1 - + + 1 0 - wxID_LISTCTRL + wxID_ANY - m_listCtrl + m_notebook protected - wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING + @@ -4819,32 +4759,14 @@ - - - - - - - - - - - - OnListEndLabelEdit - - OnListItemActivated - - - - - OnListItemSelected - + OnNotebookPageChanged + @@ -4852,6 +4774,384 @@ + + + Sending + 0 + + + + 1 + + + 0 + wxID_PANELSENDING + + + m_panelSending + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer68 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 0 + protected + 0 + + + + 5 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 1 + wxID_ANY + Bitcoin Address + + + m_staticText55 + protected + + + + + + + + + -1 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_LISTCTRLSENDING + + + m_listCtrlSending + protected + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListEndLabelEdit + + OnListItemActivated + + + + + OnListItemSelected + + + + + + + + + + + + + + + + + + + + + + Receiving + 1 + + + + 1 + + + 0 + wxID_PANELRECEIVING + + + m_panelReceiving + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer681 + wxVERTICAL + none + + 5 + wxEXPAND + 0 + + 0 + protected + 0 + + + + 6 + wxTOP|wxRIGHT|wxLEFT + 0 + + + + 1 + + + 0 + wxID_ANY + These are your Bitcoin addresses for receiving payments. You can give a different one to each sender to keep track of who is paying you. The highlighted address will be displayed in the main window. + + + m_staticText45 + protected + + + + + + + + + 570 + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxEXPAND + 0 + + 2 + protected + 0 + + + + 5 + wxALL|wxEXPAND + 1 + + + + 1 + + + 0 + wxID_LISTCTRLRECEIVING + + + m_listCtrlReceiving + protected + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_ASCENDING + + + + + + + + + + + + + + + + + + + + + + + + + + + OnListEndLabelEdit + + OnListItemActivated + + + + + OnListItemSelected + + + + + + + + + + + + + + + + + + + @@ -4885,11 +5185,11 @@ 0 - wxID_BUTTONEDIT - &Edit... + wxID_BUTTONDELETE + &Delete -1,-1 - m_buttonEdit + m_buttonDelete protected @@ -4899,7 +5199,7 @@ - OnButtonEdit + OnButtonDelete @@ -4937,11 +5237,63 @@ 0 - wxID_BUTTONNEW - &New Address... + wxID_BUTTONCOPY + &Copy to Clipboard -1,-1 - m_buttonNew + m_buttonCopy + protected + + -1,-1 + + + + + + + OnButtonCopy + + + + + + + + + + + + + + + + + + + + + + + + + + + 5 + wxALL|wxALIGN_CENTER_VERTICAL|wxEXPAND + 0 + + + + 0 + 1 + + + 0 + wxID_BUTTONEDIT + &Edit... + + -1,-1 + m_buttonEdit protected @@ -4951,7 +5303,7 @@ - OnButtonNew + OnButtonEdit @@ -4989,11 +5341,11 @@ 0 - wxID_BUTTONDELETE - &Delete + wxID_BUTTONNEW + &New Address... -1,-1 - m_buttonDelete + m_buttonNew protected @@ -5003,7 +5355,7 @@ - OnButtonDelete + OnButtonNew @@ -5151,7 +5503,7 @@ CGetTextFromUserDialogBase - 403,138 + 440,138 wxDEFAULT_DIALOG_STYLE diff --git a/util.cpp b/util.cpp index f4ce5566..40f7af34 100644 --- a/util.cpp +++ b/util.cpp @@ -445,10 +445,14 @@ const char* wxGetTranslation(const char* pszEnglish) // wxWidgets translation const char* pszTranslated = wxGetTranslation(wxString(pszEnglish, wxConvUTF8)).utf8_str(); + + // We don't cache unknown strings because caller might be passing in a + // dynamic string and we would keep allocating memory for each variation. if (strcmp(pszEnglish, pszTranslated) == 0) return pszEnglish; - // Add to cache, memory doesn't need to be freed + // Add to cache, memory doesn't need to be freed. We only cache because + // we must pass back a pointer to permanently allocated memory. char* pszCached = new char[strlen(pszTranslated)+1]; strcpy(pszCached, pszTranslated); mapCache[pszEnglish] = pszCached; From 0184604aaf21f8279df9ecbee7be4fbf0bbf99c1 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 21 Feb 2010 21:42:01 +0000 Subject: [PATCH 057/133] transaction filter tabs instead of view->show generated -- version 0.2.5 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@69 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-unix.txt | 2 +- db.cpp | 2 - main.cpp | 4 +- serialize.h | 2 +- ui.cpp | 87 +++++--- ui.h | 16 +- uibase.cpp | 92 ++++++-- uibase.h | 67 +++--- uiproject.fbp | 553 ++++++++++++++++++++++++++++++++++++++++++++----- 9 files changed, 693 insertions(+), 132 deletions(-) diff --git a/build-unix.txt b/build-unix.txt index 5e3f574a..395ac973 100644 --- a/build-unix.txt +++ b/build-unix.txt @@ -70,7 +70,7 @@ ldconfig Boost ----- If you want to build Boost yourself, -cd /usr/local/boost_1_42_0 +cd /usr/local/boost_1_38_0 su ./bootstrap.sh ./bjam install diff --git a/db.cpp b/db.cpp index bbf24a0e..51cbb309 100644 --- a/db.cpp +++ b/db.cpp @@ -599,7 +599,6 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) ssKey >> strKey; // Menu state - if (strKey == "fShowGenerated") ssValue >> fShowGenerated; if (strKey == "fGenerateBitcoins") ssValue >> fGenerateBitcoins; // Options @@ -618,7 +617,6 @@ bool CWalletDB::LoadWallet(vector& vchDefaultKeyRet) } printf("nFileVersion = %d\n", nFileVersion); - printf("fShowGenerated = %d\n", fShowGenerated); printf("fGenerateBitcoins = %d\n", fGenerateBitcoins); printf("nTransactionFee = %"PRI64d"\n", nTransactionFee); printf("addrIncoming = %s\n", addrIncoming.ToString().c_str()); diff --git a/main.cpp b/main.cpp index e5cc881b..56020a13 100644 --- a/main.cpp +++ b/main.cpp @@ -1335,7 +1335,9 @@ bool CBlock::AcceptBlock() if (!AddToBlockIndex(nFile, nBlockPos)) return error("AcceptBlock() : AddToBlockIndex failed"); - if (hashBestChain == hash && nBestHeight > 28000) + // Don't relay old inventory during initial block download. + // Please keep this constant updated to a few thousand below current block count. + if (hashBestChain == hash && nBestHeight > 40000) RelayInventory(CInv(MSG_BLOCK, hash)); // // Add atoms to user reviews for coins created diff --git a/serialize.h b/serialize.h index 23d61fe9..fb06f881 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 204; +static const int VERSION = 205; static const char* pszSubVer = ".0"; diff --git a/ui.cpp b/ui.cpp index 9bd34e09..68d57b6c 100644 --- a/ui.cpp +++ b/ui.cpp @@ -20,7 +20,6 @@ extern int g_isPainting; bool fClosedToTray = false; // Settings -int fShowGenerated = true; int fMinimizeToTray = true; int fMinimizeOnClose = true; @@ -282,6 +281,12 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) { Connect(wxEVT_UITHREADCALL, wxCommandEventHandler(CMainFrame::OnUIThreadCall), NULL, this); + // Set initially selected page + wxNotebookEvent event; + event.SetSelection(0); + OnNotebookPageChanged(event); + m_notebook->ChangeSelection(0); + // Init fRefreshListCtrl = false; fRefreshListCtrlRunning = false; @@ -298,8 +303,8 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_staticTextBalance->SetFont(fontTmp); m_staticTextBalance->SetSize(140, 17); // resize to fit ubuntu's huge default font - dResize = 1.20; - SetSize((dResize + 0.02) * GetSize().GetWidth(), 1.09 * GetSize().GetHeight()); + dResize = 1.22; + SetSize(dResize * GetSize().GetWidth(), 1.09 * GetSize().GetHeight()); #endif m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); @@ -309,13 +314,17 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) int nDateWidth = DateTimeStr(1229413914).size() * 6 + 8; if (!strstr(DateTimeStr(1229413914).c_str(), "2008")) nDateWidth += 12; - m_listCtrl->InsertColumn(0, "", wxLIST_FORMAT_LEFT, dResize * 0); - m_listCtrl->InsertColumn(1, "", wxLIST_FORMAT_LEFT, dResize * 0); - m_listCtrl->InsertColumn(2, _("Status"), wxLIST_FORMAT_LEFT, dResize * 110); - m_listCtrl->InsertColumn(3, _("Date"), wxLIST_FORMAT_LEFT, dResize * nDateWidth); - m_listCtrl->InsertColumn(4, _("Description"), wxLIST_FORMAT_LEFT, dResize * 409 - nDateWidth); - m_listCtrl->InsertColumn(5, _("Debit"), wxLIST_FORMAT_RIGHT, dResize * 79); - m_listCtrl->InsertColumn(6, _("Credit"), wxLIST_FORMAT_RIGHT, dResize * 79); + wxListCtrl* pplistCtrl[] = {m_listCtrlAll, m_listCtrlSentReceived, m_listCtrlSent, m_listCtrlReceived}; + foreach(wxListCtrl* p, pplistCtrl) + { + p->InsertColumn(0, "", wxLIST_FORMAT_LEFT, dResize * 0); + p->InsertColumn(1, "", wxLIST_FORMAT_LEFT, dResize * 0); + p->InsertColumn(2, _("Status"), wxLIST_FORMAT_LEFT, dResize * 112); + p->InsertColumn(3, _("Date"), wxLIST_FORMAT_LEFT, dResize * nDateWidth); + p->InsertColumn(4, _("Description"), wxLIST_FORMAT_LEFT, dResize * 409 - nDateWidth); + p->InsertColumn(5, _("Debit"), wxLIST_FORMAT_RIGHT, dResize * 79); + p->InsertColumn(6, _("Credit"), wxLIST_FORMAT_RIGHT, dResize * 79); + } // Init status bar int pnWidths[3] = { -100, 88, 290 }; @@ -341,6 +350,42 @@ CMainFrame::~CMainFrame() ptaskbaricon = NULL; } +void CMainFrame::OnNotebookPageChanged(wxNotebookEvent& event) +{ + event.Skip(); + nPage = event.GetSelection(); + if (nPage == ALL) + { + m_listCtrl = m_listCtrlAll; + fShowGenerated = true; + fShowSent = true; + fShowReceived = true; + } + else if (nPage == SENTRECEIVED) + { + m_listCtrl = m_listCtrlSentReceived; + fShowGenerated = false; + fShowSent = true; + fShowReceived = true; + } + else if (nPage == SENT) + { + m_listCtrl = m_listCtrlSent; + fShowGenerated = false; + fShowSent = true; + fShowReceived = false; + } + else if (nPage == RECEIVED) + { + m_listCtrl = m_listCtrlReceived; + fShowGenerated = false; + fShowSent = false; + fShowReceived = true; + } + RefreshListCtrl(); + m_listCtrl->SetFocus(); +} + void CMainFrame::OnClose(wxCloseEvent& event) { if (fMinimizeOnClose && event.CanVeto() && !IsIconized()) @@ -547,7 +592,6 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) return false; } - // View->Show Generated if (!fShowGenerated) return false; } @@ -571,7 +615,6 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) // Credit // string strDescription; - if (wtx.IsCoinBase()) { // Generated @@ -598,6 +641,8 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) else if (!mapValue["from"].empty() || !mapValue["message"].empty()) { // Received by IP connection + if (!fShowReceived) + return false; if (!mapValue["from"].empty()) strDescription += _("From: ") + mapValue["from"]; if (!mapValue["message"].empty()) @@ -610,6 +655,8 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) else { // Received by Bitcoin Address + if (!fShowReceived) + return false; foreach(const CTxOut& txout, wtx.vout) { if (txout.IsMine()) @@ -675,6 +722,9 @@ bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) // // Debit // + if (!fShowSent) + return false; + int64 nTxFee = nDebit - wtx.GetValueOut(); wtx.nLinesDisplayed = 0; for (int nOut = 0; nOut < wtx.vout.size(); nOut++) @@ -1035,19 +1085,6 @@ void CMainFrame::OnMenuFileExit(wxCommandEvent& event) Close(true); } -void CMainFrame::OnMenuViewShowGenerated(wxCommandEvent& event) -{ - // View->Show Generated - fShowGenerated = event.IsChecked(); - CWalletDB().WriteSetting("fShowGenerated", fShowGenerated); - RefreshListCtrl(); -} - -void CMainFrame::OnUpdateUIViewShowGenerated(wxUpdateUIEvent& event) -{ - event.Check(fShowGenerated); -} - void CMainFrame::OnMenuOptionsGenerate(wxCommandEvent& event) { // Options->Generate Coins diff --git a/ui.h b/ui.h index 317b87e9..acabd551 100644 --- a/ui.h +++ b/ui.h @@ -8,7 +8,6 @@ DECLARE_EVENT_TYPE(wxEVT_UITHREADCALL, -1) extern map mapArgs; // Settings -extern int fShowGenerated; extern int fMinimizeToTray; extern int fMinimizeOnClose; @@ -31,6 +30,7 @@ class CMainFrame : public CMainFrameBase { protected: // Event handlers + void OnNotebookPageChanged(wxNotebookEvent& event); void OnClose(wxCloseEvent& event); void OnIconize(wxIconizeEvent& event); void OnMouseEvents(wxMouseEvent& event); @@ -39,8 +39,6 @@ protected: void OnPaint(wxPaintEvent& event); void OnPaintListCtrl(wxPaintEvent& event); void OnMenuFileExit(wxCommandEvent& event); - void OnMenuViewShowGenerated(wxCommandEvent& event); - void OnUpdateUIViewShowGenerated(wxUpdateUIEvent& event); void OnMenuOptionsGenerate(wxCommandEvent& event); void OnUpdateUIOptionsGenerate(wxUpdateUIEvent& event); void OnMenuOptionsChangeYourAddress(wxCommandEvent& event); @@ -64,6 +62,18 @@ public: ~CMainFrame(); // Custom + enum + { + ALL = 0, + SENTRECEIVED = 1, + SENT = 2, + RECEIVED = 3, + }; + int nPage; + wxListCtrl* m_listCtrl; + bool fShowGenerated; + bool fShowSent; + bool fShowReceived; bool fRefreshListCtrl; bool fRefreshListCtrlRunning; bool fOnSetFocusAddress; diff --git a/uibase.cpp b/uibase.cpp index 340c7abc..03ca13db 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -29,13 +29,6 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_menubar->Append( m_menuFile, _("&File") ); - m_menuView = new wxMenu(); - wxMenuItem* m_menuViewShowGenerated; - m_menuViewShowGenerated = new wxMenuItem( m_menuView, wxID_VIEWSHOWGENERATED, wxString( _("&Show Generated Coins") ) , wxEmptyString, wxITEM_CHECK ); - m_menuView->Append( m_menuViewShowGenerated ); - - m_menubar->Append( m_menuView, _("&View") ); - m_menuOptions = new wxMenu(); wxMenuItem* m_menuOptionsGenerateBitcoins; m_menuOptionsGenerateBitcoins = new wxMenuItem( m_menuOptions, wxID_OPTIONSGENERATEBITCOINS, wxString( _("&Generate Coins") ) , wxEmptyString, wxITEM_CHECK ); @@ -135,8 +128,53 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& bSizer2->Add( bSizer3, 0, wxEXPAND, 5 ); - m_listCtrl = new wxListCtrl( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); - bSizer2->Add( m_listCtrl, 1, wxEXPAND, 5 ); + m_notebook = new wxNotebook( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 ); + m_panel9 = new wxPanel( m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer11; + bSizer11 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlAll = new wxListCtrl( m_panel9, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + bSizer11->Add( m_listCtrlAll, 1, wxEXPAND, 5 ); + + m_panel9->SetSizer( bSizer11 ); + m_panel9->Layout(); + bSizer11->Fit( m_panel9 ); + m_notebook->AddPage( m_panel9, _("All Transactions"), true ); + m_panel91 = new wxPanel( m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer111; + bSizer111 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlSentReceived = new wxListCtrl( m_panel91, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + bSizer111->Add( m_listCtrlSentReceived, 1, wxEXPAND, 5 ); + + m_panel91->SetSizer( bSizer111 ); + m_panel91->Layout(); + bSizer111->Fit( m_panel91 ); + m_notebook->AddPage( m_panel91, _("Sent/Received"), false ); + m_panel92 = new wxPanel( m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer112; + bSizer112 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlSent = new wxListCtrl( m_panel92, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + bSizer112->Add( m_listCtrlSent, 1, wxEXPAND, 5 ); + + m_panel92->SetSizer( bSizer112 ); + m_panel92->Layout(); + bSizer112->Fit( m_panel92 ); + m_notebook->AddPage( m_panel92, _("Sent"), false ); + m_panel93 = new wxPanel( m_notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL ); + wxBoxSizer* bSizer113; + bSizer113 = new wxBoxSizer( wxVERTICAL ); + + m_listCtrlReceived = new wxListCtrl( m_panel93, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + bSizer113->Add( m_listCtrlReceived, 1, wxEXPAND, 5 ); + + m_panel93->SetSizer( bSizer113 ); + m_panel93->Layout(); + bSizer113->Fit( m_panel93 ); + m_notebook->AddPage( m_panel93, _("Received"), false ); + + bSizer2->Add( m_notebook, 1, wxEXPAND, 5 ); this->SetSizer( bSizer2 ); this->Layout(); @@ -160,8 +198,6 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& this->Connect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); this->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaint ) ); this->Connect( m_menuFileExit->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuFileExit ) ); - this->Connect( m_menuViewShowGenerated->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuViewShowGenerated ) ); - this->Connect( m_menuViewShowGenerated->GetId(), wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIViewShowGenerated ) ); this->Connect( m_menuOptionsGenerateBitcoins->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsGenerate ) ); this->Connect( m_menuOptionsGenerateBitcoins->GetId(), wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIOptionsGenerate ) ); this->Connect( m_menuOptionsChangeYourAddress->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsChangeYourAddress ) ); @@ -186,9 +222,19 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_textCtrlAddress->Connect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); m_buttonNew->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonNew ), NULL, this ); m_buttonCopy->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); - m_listCtrl->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); - m_listCtrl->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_notebook->Connect( wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGED, wxNotebookEventHandler( CMainFrameBase::OnNotebookPageChanged ), NULL, this ); + m_listCtrlAll->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlAll->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlAll->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlSentReceived->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlSentReceived->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlSentReceived->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlSent->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlSent->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlSent->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlReceived->Connect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlReceived->Connect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlReceived->Connect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); } CMainFrameBase::~CMainFrameBase() @@ -212,8 +258,6 @@ CMainFrameBase::~CMainFrameBase() this->Disconnect( wxEVT_MOUSEWHEEL, wxMouseEventHandler( CMainFrameBase::OnMouseEvents ) ); this->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaint ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuFileExit ) ); - this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuViewShowGenerated ) ); - this->Disconnect( wxID_ANY, wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIViewShowGenerated ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsGenerate ) ); this->Disconnect( wxID_ANY, wxEVT_UPDATE_UI, wxUpdateUIEventHandler( CMainFrameBase::OnUpdateUIOptionsGenerate ) ); this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( CMainFrameBase::OnMenuOptionsChangeYourAddress ) ); @@ -238,9 +282,19 @@ CMainFrameBase::~CMainFrameBase() m_textCtrlAddress->Disconnect( wxEVT_SET_FOCUS, wxFocusEventHandler( CMainFrameBase::OnSetFocusAddress ), NULL, this ); m_buttonNew->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonNew ), NULL, this ); m_buttonCopy->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( CMainFrameBase::OnButtonCopy ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); - m_listCtrl->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_notebook->Disconnect( wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGED, wxNotebookEventHandler( CMainFrameBase::OnNotebookPageChanged ), NULL, this ); + m_listCtrlAll->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlAll->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlAll->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlSentReceived->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlSentReceived->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlSentReceived->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlSent->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlSent->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlSent->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); + m_listCtrlReceived->Disconnect( wxEVT_COMMAND_LIST_COL_BEGIN_DRAG, wxListEventHandler( CMainFrameBase::OnListColBeginDrag ), NULL, this ); + m_listCtrlReceived->Disconnect( wxEVT_COMMAND_LIST_ITEM_ACTIVATED, wxListEventHandler( CMainFrameBase::OnListItemActivated ), NULL, this ); + m_listCtrlReceived->Disconnect( wxEVT_PAINT, wxPaintEventHandler( CMainFrameBase::OnPaintListCtrl ), NULL, this ); } CTxDetailsDialogBase::CTxDetailsDialogBase( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxDialog( parent, id, title, pos, size, style ) diff --git a/uibase.h b/uibase.h index 90abd00f..1686cbfa 100644 --- a/uibase.h +++ b/uibase.h @@ -28,6 +28,7 @@ #include #include #include +#include #include #include #include @@ -36,36 +37,34 @@ #include #include #include -#include /////////////////////////////////////////////////////////////////////////// #define wxID_MAINFRAME 1000 -#define wxID_VIEWSHOWGENERATED 1001 -#define wxID_OPTIONSGENERATEBITCOINS 1002 -#define wxID_MENUOPTIONSOPTIONS 1003 -#define wxID_BUTTONSEND 1004 -#define wxID_BUTTONRECEIVE 1005 -#define wxID_TEXTCTRLADDRESS 1006 -#define wxID_BUTTONNEW 1007 -#define wxID_BUTTONCOPY 1008 -#define wxID_TRANSACTIONFEE 1009 -#define wxID_PROXYIP 1010 -#define wxID_PROXYPORT 1011 -#define wxID_TEXTCTRLPAYTO 1012 -#define wxID_BUTTONPASTE 1013 -#define wxID_BUTTONADDRESSBOOK 1014 -#define wxID_TEXTCTRLAMOUNT 1015 -#define wxID_CHOICETRANSFERTYPE 1016 -#define wxID_LISTCTRL 1017 -#define wxID_BUTTONRENAME 1018 -#define wxID_PANELSENDING 1019 -#define wxID_LISTCTRLSENDING 1020 -#define wxID_PANELRECEIVING 1021 -#define wxID_LISTCTRLRECEIVING 1022 -#define wxID_BUTTONDELETE 1023 -#define wxID_BUTTONEDIT 1024 -#define wxID_TEXTCTRL 1025 +#define wxID_OPTIONSGENERATEBITCOINS 1001 +#define wxID_MENUOPTIONSOPTIONS 1002 +#define wxID_BUTTONSEND 1003 +#define wxID_BUTTONRECEIVE 1004 +#define wxID_TEXTCTRLADDRESS 1005 +#define wxID_BUTTONNEW 1006 +#define wxID_BUTTONCOPY 1007 +#define wxID_TRANSACTIONFEE 1008 +#define wxID_PROXYIP 1009 +#define wxID_PROXYPORT 1010 +#define wxID_TEXTCTRLPAYTO 1011 +#define wxID_BUTTONPASTE 1012 +#define wxID_BUTTONADDRESSBOOK 1013 +#define wxID_TEXTCTRLAMOUNT 1014 +#define wxID_CHOICETRANSFERTYPE 1015 +#define wxID_LISTCTRL 1016 +#define wxID_BUTTONRENAME 1017 +#define wxID_PANELSENDING 1018 +#define wxID_LISTCTRLSENDING 1019 +#define wxID_PANELRECEIVING 1020 +#define wxID_LISTCTRLRECEIVING 1021 +#define wxID_BUTTONDELETE 1022 +#define wxID_BUTTONEDIT 1023 +#define wxID_TEXTCTRL 1024 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase @@ -77,7 +76,6 @@ class CMainFrameBase : public wxFrame protected: wxMenuBar* m_menubar; wxMenu* m_menuFile; - wxMenu* m_menuView; wxMenu* m_menuHelp; wxToolBar* m_toolBar; wxStatusBar* m_statusBar; @@ -91,6 +89,11 @@ class CMainFrameBase : public wxFrame wxStaticText* m_staticTextBalance; wxChoice* m_choiceFilter; + wxNotebook* m_notebook; + wxPanel* m_panel9; + wxPanel* m_panel91; + wxPanel* m_panel92; + wxPanel* m_panel93; // Virtual event handlers, overide them in your derived class virtual void OnClose( wxCloseEvent& event ){ event.Skip(); } @@ -99,8 +102,6 @@ class CMainFrameBase : public wxFrame virtual void OnMouseEvents( wxMouseEvent& event ){ event.Skip(); } virtual void OnPaint( wxPaintEvent& event ){ event.Skip(); } virtual void OnMenuFileExit( wxCommandEvent& event ){ event.Skip(); } - virtual void OnMenuViewShowGenerated( wxCommandEvent& event ){ event.Skip(); } - virtual void OnUpdateUIViewShowGenerated( wxUpdateUIEvent& event ){ event.Skip(); } virtual void OnMenuOptionsGenerate( wxCommandEvent& event ){ event.Skip(); } virtual void OnUpdateUIOptionsGenerate( wxUpdateUIEvent& event ){ event.Skip(); } virtual void OnMenuOptionsChangeYourAddress( wxCommandEvent& event ){ event.Skip(); } @@ -113,6 +114,7 @@ class CMainFrameBase : public wxFrame virtual void OnSetFocusAddress( wxFocusEvent& event ){ event.Skip(); } virtual void OnButtonNew( wxCommandEvent& event ){ event.Skip(); } virtual void OnButtonCopy( wxCommandEvent& event ){ event.Skip(); } + virtual void OnNotebookPageChanged( wxNotebookEvent& event ){ event.Skip(); } virtual void OnListColBeginDrag( wxListEvent& event ){ event.Skip(); } virtual void OnListItemActivated( wxListEvent& event ){ event.Skip(); } virtual void OnPaintListCtrl( wxPaintEvent& event ){ event.Skip(); } @@ -121,8 +123,11 @@ class CMainFrameBase : public wxFrame public: wxMenu* m_menuOptions; wxTextCtrl* m_textCtrlAddress; - wxListCtrl* m_listCtrl; - CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = _("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 712,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); + wxListCtrl* m_listCtrlAll; + wxListCtrl* m_listCtrlSentReceived; + wxListCtrl* m_listCtrlSent; + wxListCtrl* m_listCtrlReceived; + CMainFrameBase( wxWindow* parent, wxWindowID id = wxID_MAINFRAME, const wxString& title = _("Bitcoin"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 723,484 ), long style = wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER|wxTAB_TRAVERSAL ); ~CMainFrameBase(); }; diff --git a/uiproject.fbp b/uiproject.fbp index 113a1579..8643fbab 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -18,7 +18,7 @@ 1 0 0 - + wxSYS_COLOUR_BTNFACE @@ -32,7 +32,7 @@ CMainFrameBase - 712,484 + 723,484 wxDEFAULT_FRAME_STYLE|wxRESIZE_BORDER Bitcoin @@ -134,26 +134,6 @@ - - &View - m_menuView - protected - - - 0 - 1 - - wxID_VIEWSHOWGENERATED - wxITEM_CHECK - &Show Generated Coins - m_menuViewShowGenerated - none - - - OnMenuViewShowGenerated - OnUpdateUIViewShowGenerated - - &Settings m_menuOptions @@ -346,7 +326,7 @@ - + bSizer2 wxVERTICAL @@ -824,8 +804,9 @@ 5 wxEXPAND 1 - + + 1 @@ -834,16 +815,16 @@ wxID_ANY - m_listCtrl - public + m_notebook + protected - wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING + - wxVSCROLL + @@ -854,39 +835,513 @@ - - - - - OnListColBeginDrag - - - - - - - - - OnListItemActivated - - - - - - - OnPaintListCtrl + OnNotebookPageChanged + + + + + All Transactions + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_panel9 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer11 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlAll + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING + + + + + wxVSCROLL + + + + + + + + + + + + + + + OnListColBeginDrag + + + + + + + + + OnListItemActivated + + + + + + + + + + + + + OnPaintListCtrl + + + + + + + + + + + + + + Sent/Received + 0 + + + + 1 + + + 0 + wxID_ANY + + + m_panel91 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer111 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlSentReceived + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING + + + + + wxVSCROLL + + + + + + + + + + + + + + + OnListColBeginDrag + + + + + + + + + OnListItemActivated + + + + + + + + + + + + + OnPaintListCtrl + + + + + + + + + + + + + + Sent + 0 + + + + 1 + + + 0 + wxID_ANY + + + m_panel92 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer112 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlSent + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING + + + + + wxVSCROLL + + + + + + + + + + + + + + + OnListColBeginDrag + + + + + + + + + OnListItemActivated + + + + + + + + + + + + + OnPaintListCtrl + + + + + + + + + + + + + + Received + 0 + + + + 1 + + + 0 + wxID_ANY + + + m_panel93 + protected + + + + + + + wxTAB_TRAVERSAL + + + + + + + + + + + + + + + + + + + + + + + + + + bSizer113 + wxVERTICAL + none + + 5 + wxEXPAND + 1 + + + + 1 + + + 0 + wxID_ANY + + + m_listCtrlReceived + public + + + wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING + + + + + wxVSCROLL + + + + + + + + + + + + + + + OnListColBeginDrag + + + + + + + + + OnListItemActivated + + + + + + + + + + + + + OnPaintListCtrl + + + + + + + + + + + From cb420a1dfc23d3c11c5281ed8f7ae003c2f61594 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Tue, 23 Feb 2010 22:01:39 +0000 Subject: [PATCH 058/133] run as daemon without GUI, hooked wxApp::Initialize to ignore gtk-init-check failure if no GUI, fork to daemonize, rpc getinfo, getconnectioncount, getbalance, getgenerate, setgenerate, -- version 0.2.6 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@70 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- main.cpp | 15 +++--- main.h | 4 ++ rpc.cpp | 134 ++++++++++++++++++++++++++++++++++++++++++++-------- serialize.h | 2 +- ui.cpp | 94 +++++++++++++++++++++++++++++------- util.cpp | 4 +- util.h | 1 + 7 files changed, 207 insertions(+), 47 deletions(-) diff --git a/main.cpp b/main.cpp index 56020a13..bd69de92 100644 --- a/main.cpp +++ b/main.cpp @@ -1315,11 +1315,10 @@ bool CBlock::AcceptBlock() if (nTime <= pindexPrev->GetMedianTimePast()) return error("AcceptBlock() : block's timestamp is too early"); - // Check that all transactions are finalized (starting around Mar 2010) - if (nBestHeight > 36000) - foreach(const CTransaction& tx, vtx) - if (!tx.IsFinal(nTime)) - return error("AcceptBlock() : contains a non-final transaction"); + // Check that all transactions are finalized + foreach(const CTransaction& tx, vtx) + if (!tx.IsFinal(nTime)) + return error("AcceptBlock() : contains a non-final transaction"); // Check proof of work if (nBits != GetNextWorkRequired(pindexPrev)) @@ -1336,7 +1335,7 @@ bool CBlock::AcceptBlock() return error("AcceptBlock() : AddToBlockIndex failed"); // Don't relay old inventory during initial block download. - // Please keep this constant updated to a few thousand below current block count. + // Please keep this number updated to a few thousand below current block count. if (hashBestChain == hash && nBestHeight > 40000) RelayInventory(CInv(MSG_BLOCK, hash)); @@ -1556,8 +1555,8 @@ bool LoadBlockIndex(bool fAllowNew) CTransaction txNew; txNew.vin.resize(1); txNew.vout.resize(1); - txNew.vin[0].scriptSig = CScript() << 486604799 << CBigNum(4) << vector((const unsigned char*)pszTimestamp, (const unsigned char*)pszTimestamp + strlen(pszTimestamp)); - txNew.vout[0].nValue = 50 * COIN; + txNew.vin[0].scriptSig = CScript() << 486604799 << CBigNum(4) << vector((const unsigned char*)pszTimestamp, (const unsigned char*)pszTimestamp + strlen(pszTimestamp)); + txNew.vout[0].nValue = 50 * COIN; CBigNum bnPubKey; bnPubKey.SetHex("0x5F1DF16B2B704C8A578D0BBAF74D385CDE12C11EE50455F3C438EF4C3FBCF649B6DE611FEAE06279A60939E028A8D65C10B73071A6F16719274855FEB0FD8A6704"); txNew.vout[0].scriptPubKey = CScript() << bnPubKey << OP_CHECKSIG; diff --git a/main.h b/main.h index 4027f87e..452ce85b 100644 --- a/main.h +++ b/main.h @@ -1384,6 +1384,9 @@ public: CPrivKey vchPrivKey; int64 nTimeCreated; int64 nTimeExpires; + string strComment; + //// todo: add something to note what created it (user, getnewaddress, change) + //// maybe should have a map property map CWalletKey(int64 nTimeExpiresIn=0) { @@ -1398,6 +1401,7 @@ public: READWRITE(vchPrivKey); READWRITE(nTimeCreated); READWRITE(nTimeExpires); + READWRITE(strComment); ) }; diff --git a/rpc.cpp b/rpc.cpp index f4eef37f..7f3e71db 100644 --- a/rpc.cpp +++ b/rpc.cpp @@ -66,6 +66,17 @@ Value getblocknumber(const Array& params) } +Value getconnectioncount(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getconnectioncount (no parameters)\n" + "Returns the number of connections to other nodes."); + + return (int)vNodes.size(); +} + + Value getdifficulty(const Array& params) { if (params.size() != 0) @@ -85,6 +96,71 @@ Value getdifficulty(const Array& params) } +Value getbalance(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getbalance (no parameters)\n" + "Returns the server's available balance."); + + return ((double)GetBalance() / (double)COIN); +} + + +Value getgenerate(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getgenerate (no parameters)\n" + "Returns true or false."); + + return (bool)fGenerateBitcoins; +} + + +Value setgenerate(const Array& params) +{ + if (params.size() < 1 || params.size() > 2) + throw runtime_error( + "setgenerate [genproclimit]\n" + " is true or false to turn generation on or off.\n" + "Generation is limited to [genproclimit] processors, -1 is unlimited."); + + bool fGenerate = true; + if (params.size() > 0) + fGenerate = params[0].get_bool(); + + if (params.size() > 1) + { + int nGenProcLimit = params[1].get_int(); + fLimitProcessors = (nGenProcLimit != -1); + CWalletDB().WriteSetting("fLimitProcessors", fLimitProcessors); + if (nGenProcLimit != -1) + CWalletDB().WriteSetting("nLimitProcessors", nLimitProcessors = nGenProcLimit); + } + + GenerateBitcoins(fGenerate); + return Value::null; +} + + +Value getinfo(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getinfo (no parameters)"); + + Object obj; + obj.push_back(Pair("balance", (double)GetBalance() / (double)COIN)); + obj.push_back(Pair("blocks", (int)nBestHeight + 1)); + obj.push_back(Pair("connections", (int)vNodes.size())); + obj.push_back(Pair("proxy", (fUseProxy ? addrProxy.ToStringIPPort() : string()))); + obj.push_back(Pair("generate", (bool)fGenerateBitcoins)); + obj.push_back(Pair("genproclimit", (int)(fLimitProcessors ? nLimitProcessors : -1))); + return obj; +} + + Value getnewaddress(const Array& params) { if (params.size() > 1) @@ -102,8 +178,7 @@ Value getnewaddress(const Array& params) // Generate a new key that is added to wallet string strAddress = PubKeyToAddress(GenerateNewKey()); - if (params.size() > 0) - SetAddressBookName(strAddress, strLabel); + SetAddressBookName(strAddress, strLabel); return strAddress; } @@ -214,10 +289,10 @@ Value getallreceived(const Array& params) "getallreceived [minconf=1]\n" "[minconf] is the minimum number of confirmations before payments are included.\n" "Returns an array of objects containing:\n" - " \"address\" : bitcoin address\n" + " \"address\" : receiving address\n" " \"amount\" : total amount received by the address\n" - " \"conf\" : number of confirmations\n" - " \"label\" : the label set for this address when it was created by getnewaddress"); + " \"confirmations\" : number of confirmations of the most recent transaction included\n" + " \"label\" : the label of the receiving address"); // Minimum confirmations int nMinDepth = 1; @@ -235,18 +310,26 @@ Value getallreceived(const Array& params) continue; int nDepth = wtx.GetDepthInMainChain(); - if (nDepth >= nMinDepth) + if (nDepth < nMinDepth) + continue; + + // Filter out debits and payments to self, which may have change return + // we don't want to count. + int64 nCredit = wtx.GetCredit(true); + int64 nDebit = wtx.GetDebit(); + int64 nNet = nCredit - nDebit; + if (nNet <= 0) + continue; + + foreach(const CTxOut& txout, wtx.vout) { - foreach(const CTxOut& txout, wtx.vout) - { - uint160 hash160 = txout.scriptPubKey.GetBitcoinAddressHash160(); - if (hash160 == 0 || !mapPubKeys.count(hash160)) - continue; - - tallyitem& item = mapTally[hash160]; - item.nAmount += txout.nValue; - item.nConf = min(item.nConf, nDepth); - } + uint160 hash160 = txout.scriptPubKey.GetBitcoinAddressHash160(); + if (hash160 == 0 || !mapPubKeys.count(hash160)) + continue; + + tallyitem& item = mapTally[hash160]; + item.nAmount += txout.nValue; + item.nConf = min(item.nConf, nDepth); } } } @@ -264,10 +347,10 @@ Value getallreceived(const Array& params) strLabel = (*mi).second; Object obj; - obj.push_back(Pair("address", strAddress)); - obj.push_back(Pair("amount", (double)(*it).second.nAmount / (double)COIN)); - obj.push_back(Pair("conf", (*it).second.nConf)); - obj.push_back(Pair("label", strLabel)); + obj.push_back(Pair("address", strAddress)); + obj.push_back(Pair("amount", (double)(*it).second.nAmount / (double)COIN)); + obj.push_back(Pair("confirmations", (*it).second.nConf)); + obj.push_back(Pair("label", strLabel)); ret.push_back(obj); } } @@ -290,7 +373,12 @@ pair pCallTable[] = make_pair("stop", &stop), make_pair("getblockcount", &getblockcount), make_pair("getblocknumber", &getblocknumber), + make_pair("getconnectioncount", &getconnectioncount), make_pair("getdifficulty", &getdifficulty), + make_pair("getbalance", &getbalance), + make_pair("getgenerate", &getgenerate), + make_pair("setgenerate", &setgenerate), + make_pair("getinfo", &getinfo), make_pair("getnewaddress", &getnewaddress), make_pair("sendtoaddress", &sendtoaddress), make_pair("listtransactions", &listtransactions), @@ -568,9 +656,13 @@ int CommandLineRPC(int argc, char *argv[]) Array params; for (int i = 2; i < argc; i++) params.push_back(argv[i]); + int n = params.size(); + // // Special case other types - int n = params.size(); + // + if (strMethod == "setgenerate" && n > 0) ConvertTo(params[0]); + if (strMethod == "setgenerate" && n > 1) ConvertTo(params[1]); if (strMethod == "sendtoaddress" && n > 1) ConvertTo(params[1]); if (strMethod == "listtransactions" && n > 0) ConvertTo(params[0]); if (strMethod == "listtransactions" && n > 1) ConvertTo(params[1]); diff --git a/serialize.h b/serialize.h index fb06f881..8fb72a0a 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 205; +static const int VERSION = 206; static const char* pszSubVer = ".0"; diff --git a/ui.cpp b/ui.cpp index 68d57b6c..748b4dad 100644 --- a/ui.cpp +++ b/ui.cpp @@ -26,6 +26,20 @@ int fMinimizeOnClose = true; +int MyMessageBox(const wxString& message, const wxString& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1) +{ + if (fDaemon) + { + printf("wxMessageBox %s: %s\n", string(caption).c_str(), string(message).c_str()); + fprintf(stderr, "%s: %s\n", string(caption).c_str(), string(message).c_str()); + return wxOK; + } + return wxMessageBox(message, caption, style, parent, x, y); +} +#define wxMessageBox MyMessageBox + + + @@ -209,16 +223,10 @@ void CalledMessageBox(const string& message, const string& caption, int style, w int ThreadSafeMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) { - if (fDaemon) - { - printf("wxMessageBox %s: %s\n", caption.c_str(), message.c_str()); - return wxOK; - } - #ifdef __WXMSW__ return wxMessageBox(message, caption, style, parent, x, y); #else - if (wxThread::IsMain()) + if (wxThread::IsMain() || fDaemon) { return wxMessageBox(message, caption, style, parent, x, y); } @@ -563,7 +571,7 @@ string SingleLine(const string& strIn) bool CMainFrame::InsertTransaction(const CWalletTx& wtx, bool fNew, int nIndex) { int64 nTime = wtx.nTimeDisplayed = wtx.GetTxTime(); - int64 nCredit = wtx.GetCredit(); + int64 nCredit = wtx.GetCredit(true); int64 nDebit = wtx.GetDebit(); int64 nNet = nCredit - nDebit; uint256 hash = wtx.GetHash(); @@ -2571,6 +2579,9 @@ public: bool OnInit2(); int OnExit(); + // Hook Initialize so we can start without GUI + virtual bool Initialize(int& argc, wxChar** argv); + // 2nd-level exception handling: we get all the exceptions occurring in any // event handler here virtual bool OnExceptionInMainLoop(); @@ -2586,6 +2597,64 @@ public: IMPLEMENT_APP(CMyApp) +bool CMyApp::Initialize(int& argc, wxChar** argv) +{ + if (argc > 1 && argv[1][0] != '-' && (!fWindows || argv[1][0] != '/') && + wxString(argv[1]) != "start") + { + fCommandLine = true; + } + else + { + // wxApp::Initialize will remove environment-specific parameters, + // so it's too early to call ParseParameters yet + for (int i = 1; i < argc; i++) + { + wxString str = argv[i]; + #ifdef __WXMSW__ + if (str.size() >= 1 && str[0] == '/') + str[0] = '-'; + str = str.MakeLower(); + #endif + // haven't decided which argument to use for this yet + if (str == "-daemon" || str == "-d" || str == "start") + fDaemon = true; + } + } + +#ifdef __WXGTK__ + if (fDaemon || fCommandLine) + { + // Call the original Initialize while suppressing error messages + // and ignoring failure. If unable to initialize GTK, it fails + // near the end so hopefully the last few things don't matter. + { + wxLogNull logNo; + wxApp::Initialize(argc, argv); + } + + if (fDaemon) + { + fprintf(stdout, "bitcoin server starting\n"); + + // Daemonize + pid_t pid = fork(); + if (pid < 0) + { + fprintf(stderr, "Error: fork() returned %d errno %d\n", pid, errno); + return false; + } + if (pid > 0) + pthread_exit((void*)0); + } + + return true; + } +#endif + + return wxApp::Initialize(argc, argv); +} + bool CMyApp::OnInit() { bool fRet = false; @@ -2650,7 +2719,7 @@ bool CMyApp::OnInit2() // // Parameters // - if (argc > 1 && argv[1][0] != '-' && argv[1][0] != '/') + if (fCommandLine) { int ret = CommandLineRPC(argc, argv); exit(ret); @@ -2696,13 +2765,6 @@ bool CMyApp::OnInit2() if (mapArgs.count("-printtodebugger")) fPrintToDebugger = true; - if (mapArgs.count("-daemon") || mapArgs.count("-d")) - { - fDaemon = true; - /// todo: need to fork - /// should it fork after the bind/single instance stuff? - } - if (!fDebug && !pszSetDataDir[0]) ShrinkDebugFile(); printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); diff --git a/util.cpp b/util.cpp index 40f7af34..6ea735d1 100644 --- a/util.cpp +++ b/util.cpp @@ -13,6 +13,7 @@ bool fPrintToDebugger = false; char pszSetDataDir[MAX_PATH] = ""; bool fShutdown = false; bool fDaemon = false; +bool fCommandLine = false; @@ -500,7 +501,8 @@ void PrintException(std::exception* pex, const char* pszThread) char pszMessage[1000]; FormatException(pszMessage, pex, pszThread); printf("\n\n************************\n%s\n", pszMessage); - if (wxTheApp) + fprintf(stderr, "\n\n************************\n%s\n", pszMessage); + if (wxTheApp && !fDaemon) wxMessageBox(pszMessage, "Error", wxOK | wxICON_ERROR); throw; //DebugBreak(); diff --git a/util.h b/util.h index 8aed902f..75f0956f 100644 --- a/util.h +++ b/util.h @@ -121,6 +121,7 @@ extern bool fPrintToDebugger; extern char pszSetDataDir[MAX_PATH]; extern bool fShutdown; extern bool fDaemon; +extern bool fCommandLine; void RandAddSeed(); void RandAddSeedPerfmon(); From c2430126d75318508836a24f90d3524320190465 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Thu, 25 Feb 2010 21:55:27 +0000 Subject: [PATCH 059/133] GUI-less build target bitcoind that links to wxBase and shouldn't need GTK, split init and shutdown from ui.cpp into init.cpp, support wxUSE_GUI=0 -- version 0.2.7 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@71 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 7 +- init.cpp | 625 +++++++++++++++++++++++++++++++++++++++++++++ init.h | 7 + main.cpp | 4 +- main.h | 2 + makefile.mingw | 62 ++--- makefile.unix | 61 +++-- makefile.vc | 87 ++++--- rpc.cpp | 10 +- serialize.h | 2 +- ui.cpp | 669 +------------------------------------------------ ui.h | 49 +++- util.cpp | 5 +- 13 files changed, 817 insertions(+), 773 deletions(-) create mode 100644 init.cpp create mode 100644 init.h diff --git a/headers.h b/headers.h index 7fb227ec..3c8d32ca 100644 --- a/headers.h +++ b/headers.h @@ -21,9 +21,11 @@ #include #include #include -#include #include #include +#if wxUSE_GUI +#include +#endif #include #include #include @@ -100,8 +102,11 @@ using namespace boost; #include "irc.h" #include "main.h" #include "rpc.h" +#if wxUSE_GUI #include "uibase.h" +#endif #include "ui.h" +#include "init.h" #include "xpm/addressbook16.xpm" #include "xpm/addressbook20.xpm" diff --git a/init.cpp b/init.cpp new file mode 100644 index 00000000..06d8f4c9 --- /dev/null +++ b/init.cpp @@ -0,0 +1,625 @@ +// Copyright (c) 2009-2010 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +#include "headers.h" + + + + +void ExitTimeout(void* parg) +{ +#ifdef __WXMSW__ + Sleep(5000); + ExitProcess(0); +#endif +} + +void Shutdown(void* parg) +{ + static CCriticalSection cs_Shutdown; + static bool fTaken; + bool fFirstThread; + CRITICAL_BLOCK(cs_Shutdown) + { + fFirstThread = !fTaken; + fTaken = true; + } + static bool fExit; + if (fFirstThread) + { + fShutdown = true; + nTransactionsUpdated++; + DBFlush(false); + StopNode(); + DBFlush(true); + CreateThread(ExitTimeout, NULL); + Sleep(50); + printf("Bitcoin exiting\n\n"); + fExit = true; + exit(0); + } + else + { + while (!fExit) + Sleep(500); + Sleep(100); + ExitThread(0); + } +} + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// Startup folder +// + +#ifdef __WXMSW__ +typedef WINSHELLAPI BOOL (WINAPI *PSHGETSPECIALFOLDERPATHA)(HWND hwndOwner, LPSTR lpszPath, int nFolder, BOOL fCreate); + +string MyGetSpecialFolderPath(int nFolder, bool fCreate) +{ + char pszPath[MAX_PATH+100] = ""; + + // SHGetSpecialFolderPath is not usually available on NT 4.0 + HMODULE hShell32 = LoadLibraryA("shell32.dll"); + if (hShell32) + { + PSHGETSPECIALFOLDERPATHA pSHGetSpecialFolderPath = + (PSHGETSPECIALFOLDERPATHA)GetProcAddress(hShell32, "SHGetSpecialFolderPathA"); + if (pSHGetSpecialFolderPath) + (*pSHGetSpecialFolderPath)(NULL, pszPath, nFolder, fCreate); + FreeModule(hShell32); + } + + // Backup option + if (pszPath[0] == '\0') + { + if (nFolder == CSIDL_STARTUP) + { + strcpy(pszPath, getenv("USERPROFILE")); + strcat(pszPath, "\\Start Menu\\Programs\\Startup"); + } + else if (nFolder == CSIDL_APPDATA) + { + strcpy(pszPath, getenv("APPDATA")); + } + } + + return pszPath; +} + +string StartupShortcutPath() +{ + return MyGetSpecialFolderPath(CSIDL_STARTUP, true) + "\\Bitcoin.lnk"; +} + +bool GetStartOnSystemStartup() +{ + return wxFileExists(StartupShortcutPath()); +} + +void SetStartOnSystemStartup(bool fAutoStart) +{ + // If the shortcut exists already, remove it for updating + remove(StartupShortcutPath().c_str()); + + if (fAutoStart) + { + CoInitialize(NULL); + + // Get a pointer to the IShellLink interface. + IShellLink* psl = NULL; + HRESULT hres = CoCreateInstance(CLSID_ShellLink, NULL, + CLSCTX_INPROC_SERVER, IID_IShellLink, + reinterpret_cast(&psl)); + + if (SUCCEEDED(hres)) + { + // Get the current executable path + TCHAR pszExePath[MAX_PATH]; + GetModuleFileName(NULL, pszExePath, sizeof(pszExePath)); + + // Set the path to the shortcut target + psl->SetPath(pszExePath); + PathRemoveFileSpec(pszExePath); + psl->SetWorkingDirectory(pszExePath); + psl->SetShowCmd(SW_SHOWMINNOACTIVE); + + // Query IShellLink for the IPersistFile interface for + // saving the shortcut in persistent storage. + IPersistFile* ppf = NULL; + hres = psl->QueryInterface(IID_IPersistFile, + reinterpret_cast(&ppf)); + if (SUCCEEDED(hres)) + { + WCHAR pwsz[MAX_PATH]; + // Ensure that the string is ANSI. + MultiByteToWideChar(CP_ACP, 0, StartupShortcutPath().c_str(), -1, pwsz, MAX_PATH); + // Save the link by calling IPersistFile::Save. + hres = ppf->Save(pwsz, TRUE); + ppf->Release(); + } + psl->Release(); + } + CoUninitialize(); + } +} +#else +bool GetStartOnSystemStartup() { return false; } +void SetStartOnSystemStartup(bool fAutoStart) { } +#endif + + + + + + + +////////////////////////////////////////////////////////////////////////////// +// +// CMyApp +// + +// Define a new application +class CMyApp: public wxApp +{ +public: + wxLocale m_locale; + + CMyApp(){}; + ~CMyApp(){}; + bool OnInit(); + bool OnInit2(); + int OnExit(); + + // Hook Initialize so we can start without GUI + virtual bool Initialize(int& argc, wxChar** argv); + + // 2nd-level exception handling: we get all the exceptions occurring in any + // event handler here + virtual bool OnExceptionInMainLoop(); + + // 3rd, and final, level exception handling: whenever an unhandled + // exception is caught, this function is called + virtual void OnUnhandledException(); + + // and now for something different: this function is called in case of a + // crash (e.g. dereferencing null pointer, division by 0, ...) + virtual void OnFatalException(); +}; + +IMPLEMENT_APP(CMyApp) + +bool CMyApp::Initialize(int& argc, wxChar** argv) +{ + if (argc > 1 && argv[1][0] != '-' && (!fWindows || argv[1][0] != '/') && + wxString(argv[1]) != "start") + { + fCommandLine = true; + } + else if (!fGUI) + { + fDaemon = true; + } + else + { + // wxApp::Initialize will remove environment-specific parameters, + // so it's too early to call ParseParameters yet + for (int i = 1; i < argc; i++) + { + wxString str = argv[i]; + #ifdef __WXMSW__ + if (str.size() >= 1 && str[0] == '/') + str[0] = '-'; + str = str.MakeLower(); + #endif + // haven't decided which argument to use for this yet + if (str == "-daemon" || str == "-d" || str == "start") + fDaemon = true; + } + } + + if (fDaemon) + fprintf(stdout, "bitcoin server starting\n"); + +#ifdef __WXGTK__ + if (fDaemon || fCommandLine) + { + // Call the original Initialize while suppressing error messages + // and ignoring failure. If unable to initialize GTK, it fails + // near the end so hopefully the last few things don't matter. + { + wxLogNull logNo; + wxApp::Initialize(argc, argv); + } + + if (fDaemon) + { + // Daemonize + pid_t pid = fork(); + if (pid < 0) + { + fprintf(stderr, "Error: fork() returned %d errno %d\n", pid, errno); + return false; + } + if (pid > 0) + pthread_exit((void*)0); + } + + return true; + } +#endif + + return wxApp::Initialize(argc, argv); +} + +bool CMyApp::OnInit() +{ + bool fRet = false; + try + { + fRet = OnInit2(); + } + catch (std::exception& e) { + PrintException(&e, "OnInit()"); + } catch (...) { + PrintException(NULL, "OnInit()"); + } + if (!fRet) + Shutdown(NULL); + return fRet; +} + +extern int g_isPainting; + +bool CMyApp::OnInit2() +{ +#ifdef _MSC_VER + // Turn off microsoft heap dump noise + _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE); + _CrtSetReportFile(_CRT_WARN, CreateFileA("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); +#endif +#if defined(__WXMSW__) && defined(__WXDEBUG__) && wxUSE_GUI + // Disable malfunctioning wxWidgets debug assertion + g_isPainting = 10000; +#endif +#if wxUSE_GUI + wxImage::AddHandler(new wxPNGHandler); +#endif +#ifdef __WXMSW__ + SetAppName("Bitcoin"); +#else + SetAppName("bitcoin"); + umask(077); +#endif +#ifdef __WXMSW__ +#if wxUSE_UNICODE + // Hack to set wxConvLibc codepage to UTF-8 on Windows, + // may break if wxMBConv_win32 implementation in strconv.cpp changes. + class wxMBConv_win32 : public wxMBConv + { + public: + long m_CodePage; + size_t m_minMBCharWidth; + }; + if (((wxMBConv_win32*)&wxConvLibc)->m_CodePage == CP_ACP) + ((wxMBConv_win32*)&wxConvLibc)->m_CodePage = CP_UTF8; +#endif +#endif + + // Load locale//LC_MESSAGES/bitcoin.mo language file + m_locale.Init(wxLANGUAGE_DEFAULT, 0); + m_locale.AddCatalogLookupPathPrefix("locale"); + if (!fWindows) + { + m_locale.AddCatalogLookupPathPrefix("/usr/share/locale"); + m_locale.AddCatalogLookupPathPrefix("/usr/local/share/locale"); + } + m_locale.AddCatalog("wxstd"); // wxWidgets standard translations, if any + m_locale.AddCatalog("bitcoin"); + + // + // Parameters + // + if (fCommandLine) + { + int ret = CommandLineRPC(argc, argv); + exit(ret); + } + + ParseParameters(argc, argv); + if (mapArgs.count("-?") || mapArgs.count("--help")) + { + wxString strUsage = string() + + _("Usage: bitcoin [options]") + "\t\t\t\t\t\t\n" + + _("Options:\n") + + " -gen \t\t " + _("Generate coins\n") + + " -gen=0 \t\t " + _("Don't generate coins\n") + + " -min \t\t " + _("Start minimized\n") + + " -datadir= \t " + _("Specify data directory\n") + + " -proxy=\t " + _("Connect through socks4 proxy\n") + + " -addnode= \t " + _("Add a node to connect to\n") + + " -connect= \t " + _("Connect only to the specified node\n") + + " -? \t\t " + _("This help message\n"); + + if (fWindows && fGUI) + { + // Remove spaces, the tabs make the columns line up in the message box + for (int i = 0; i < 50; i++) + strUsage.Replace(" \t", "\t"); + wxMessageBox(strUsage, "Bitcoin", wxOK); + } + else + { + // Remove tabs + strUsage.Replace("\t", ""); + fprintf(stderr, "%s", ((string)strUsage).c_str()); + } + return false; + } + + if (mapArgs.count("-datadir")) + strlcpy(pszSetDataDir, mapArgs["-datadir"].c_str(), sizeof(pszSetDataDir)); + + if (mapArgs.count("-debug")) + fDebug = true; + + if (mapArgs.count("-printtodebugger")) + fPrintToDebugger = true; + + if (!fDebug && !pszSetDataDir[0]) + ShrinkDebugFile(); + printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); + printf("Bitcoin version 0.%d.%d%s beta, OS version %s\n", VERSION/100, VERSION%100, pszSubVer, ((string)wxGetOsDescription()).c_str()); + printf("System default language is %d %s\n", m_locale.GetSystemLanguage(), ((string)m_locale.GetSysName()).c_str()); + printf("Language file %s (%s)\n", (string("locale/") + (string)m_locale.GetCanonicalName() + "/LC_MESSAGES/bitcoin.mo").c_str(), ((string)m_locale.GetLocale()).c_str()); + + if (mapArgs.count("-loadblockindextest")) + { + CTxDB txdb("r"); + txdb.LoadBlockIndex(); + PrintBlockTree(); + return false; + } + + // + // Limit to single instance per user + // Required to protect the database files if we're going to keep deleting log.* + // +#ifdef __WXMSW__ + // todo: wxSingleInstanceChecker wasn't working on Linux, never deleted its lock file + // maybe should go by whether successfully bind port 8333 instead + wxString strMutexName = wxString("bitcoin_running.") + getenv("HOMEPATH"); + for (int i = 0; i < strMutexName.size(); i++) + if (!isalnum(strMutexName[i])) + strMutexName[i] = '.'; + wxSingleInstanceChecker* psingleinstancechecker = new wxSingleInstanceChecker(strMutexName); + if (psingleinstancechecker->IsAnotherRunning()) + { + printf("Existing instance found\n"); + unsigned int nStart = GetTime(); + loop + { + // TODO: find out how to do this in Linux, or replace with wxWidgets commands + // Show the previous instance and exit + HWND hwndPrev = FindWindowA("wxWindowClassNR", "Bitcoin"); + if (hwndPrev) + { + if (IsIconic(hwndPrev)) + ShowWindow(hwndPrev, SW_RESTORE); + SetForegroundWindow(hwndPrev); + return false; + } + + if (GetTime() > nStart + 60) + return false; + + // Resume this instance if the other exits + delete psingleinstancechecker; + Sleep(1000); + psingleinstancechecker = new wxSingleInstanceChecker(strMutexName); + if (!psingleinstancechecker->IsAnotherRunning()) + break; + } + } +#endif + + // Bind to the port early so we can tell if another instance is already running. + // This is a backup to wxSingleInstanceChecker, which doesn't work on Linux. + string strErrors; + if (!BindListenPort(strErrors)) + { + wxMessageBox(strErrors, "Bitcoin"); + return false; + } + + // + // Load data files + // + bool fFirstRun; + strErrors = ""; + int64 nStart; + + printf("Loading addresses...\n"); + nStart = GetTimeMillis(); + if (!LoadAddresses()) + strErrors += _("Error loading addr.dat \n"); + printf(" addresses %15"PRI64d"ms\n", GetTimeMillis() - nStart); + + printf("Loading block index...\n"); + nStart = GetTimeMillis(); + if (!LoadBlockIndex()) + strErrors += _("Error loading blkindex.dat \n"); + printf(" block index %15"PRI64d"ms\n", GetTimeMillis() - nStart); + + printf("Loading wallet...\n"); + nStart = GetTimeMillis(); + if (!LoadWallet(fFirstRun)) + strErrors += _("Error loading wallet.dat \n"); + printf(" wallet %15"PRI64d"ms\n", GetTimeMillis() - nStart); + + printf("Done loading\n"); + + //// debug print + printf("mapBlockIndex.size() = %d\n", mapBlockIndex.size()); + printf("nBestHeight = %d\n", nBestHeight); + printf("mapKeys.size() = %d\n", mapKeys.size()); + printf("mapPubKeys.size() = %d\n", mapPubKeys.size()); + printf("mapWallet.size() = %d\n", mapWallet.size()); + printf("mapAddressBook.size() = %d\n", mapAddressBook.size()); + + if (!strErrors.empty()) + { + wxMessageBox(strErrors, "Bitcoin"); + return false; + } + + // Add wallet transactions that aren't already in a block to mapTransactions + ReacceptWalletTransactions(); + + // + // Parameters + // + if (mapArgs.count("-printblockindex") || mapArgs.count("-printblocktree")) + { + PrintBlockTree(); + return false; + } + + if (mapArgs.count("-printblock")) + { + string strMatch = mapArgs["-printblock"]; + int nFound = 0; + for (map::iterator mi = mapBlockIndex.begin(); mi != mapBlockIndex.end(); ++mi) + { + uint256 hash = (*mi).first; + if (strncmp(hash.ToString().c_str(), strMatch.c_str(), strMatch.size()) == 0) + { + CBlockIndex* pindex = (*mi).second; + CBlock block; + block.ReadFromDisk(pindex); + block.BuildMerkleTree(); + block.print(); + printf("\n"); + nFound++; + } + } + if (nFound == 0) + printf("No blocks matching %s were found\n", strMatch.c_str()); + return false; + } + + if (mapArgs.count("-gen")) + { + if (mapArgs["-gen"].empty()) + fGenerateBitcoins = true; + else + fGenerateBitcoins = (atoi(mapArgs["-gen"].c_str()) != 0); + } + + if (mapArgs.count("-proxy")) + { + fUseProxy = true; + addrProxy = CAddress(mapArgs["-proxy"]); + if (!addrProxy.IsValid()) + { + wxMessageBox(_("Invalid -proxy address"), "Bitcoin"); + return false; + } + } + + if (mapArgs.count("-addnode")) + { + foreach(string strAddr, mapMultiArgs["-addnode"]) + { + CAddress addr(strAddr, NODE_NETWORK); + addr.nTime = 0; // so it won't relay unless successfully connected + if (addr.IsValid()) + AddAddress(addr); + } + } + + // + // Create the main window and start the node + // + if (!fDaemon) + CreateMainWindow(); + + if (!CheckDiskSpace()) + return false; + + RandAddSeedPerfmon(); + + if (!CreateThread(StartNode, NULL)) + wxMessageBox("Error: CreateThread(StartNode) failed", "Bitcoin"); + + if (mapArgs.count("-server") || fDaemon) + CreateThread(ThreadRPCServer, NULL); + + if (fFirstRun) + SetStartOnSystemStartup(true); + + return true; +} + +int CMyApp::OnExit() +{ + Shutdown(NULL); + return wxApp::OnExit(); +} + +bool CMyApp::OnExceptionInMainLoop() +{ + try + { + throw; + } + catch (std::exception& e) + { + PrintException(&e, "CMyApp::OnExceptionInMainLoop()"); + wxLogWarning("Exception %s %s", typeid(e).name(), e.what()); + Sleep(1000); + throw; + } + catch (...) + { + PrintException(NULL, "CMyApp::OnExceptionInMainLoop()"); + wxLogWarning("Unknown exception"); + Sleep(1000); + throw; + } + return true; +} + +void CMyApp::OnUnhandledException() +{ + // this shows how we may let some exception propagate uncaught + try + { + throw; + } + catch (std::exception& e) + { + PrintException(&e, "CMyApp::OnUnhandledException()"); + wxLogWarning("Exception %s %s", typeid(e).name(), e.what()); + Sleep(1000); + throw; + } + catch (...) + { + PrintException(NULL, "CMyApp::OnUnhandledException()"); + wxLogWarning("Unknown exception"); + Sleep(1000); + throw; + } +} + +void CMyApp::OnFatalException() +{ + wxMessageBox(_("Program has crashed and will terminate. "), "Bitcoin", wxOK | wxICON_ERROR); +} diff --git a/init.h b/init.h new file mode 100644 index 00000000..bd5e3659 --- /dev/null +++ b/init.h @@ -0,0 +1,7 @@ +// Copyright (c) 2009-2010 Satoshi Nakamoto +// Distributed under the MIT/X11 software license, see the accompanying +// file license.txt or http://www.opensource.org/licenses/mit-license.php. + +void Shutdown(void* parg); +bool GetStartOnSystemStartup(); +void SetStartOnSystemStartup(bool fAutoStart); diff --git a/main.cpp b/main.cpp index bd69de92..f7416230 100644 --- a/main.cpp +++ b/main.cpp @@ -55,6 +55,8 @@ int64 nTransactionFee = 0; CAddress addrIncoming; int fLimitProcessors = false; int nLimitProcessors = 1; +int fMinimizeToTray = true; +int fMinimizeOnClose = true; @@ -2990,7 +2992,7 @@ string SendMoneyToBitcoinAddress(string strAddress, int64 nValue, CWalletTx& wtx if (nValue <= 0) return _("Invalid amount"); if (nValue + nTransactionFee > GetBalance()) - return _("You don't have enough money"); + return _("Insufficient funds"); // Parse bitcoin address CScript scriptPubKey; diff --git a/main.h b/main.h index 452ce85b..83b84699 100644 --- a/main.h +++ b/main.h @@ -45,6 +45,8 @@ extern int64 nTransactionFee; extern CAddress addrIncoming; extern int fLimitProcessors; extern int nLimitProcessors; +extern int fMinimizeToTray; +extern int fMinimizeOnClose; diff --git a/makefile.mingw b/makefile.mingw index 2644cbaa..449a9607 100644 --- a/makefile.mingw +++ b/makefile.mingw @@ -18,69 +18,59 @@ LIBPATHS= \ -L"/openssl/out" \ -L"/wxwidgets/lib/gcc_lib" +WXLIBS= \ + -l wxmsw29ud_html -l wxmsw29ud_core -l wxmsw29ud_adv -l wxbase29ud -l wxtiffd -l wxjpegd -l wxpngd -l wxzlibd + LIBS= \ -l libboost_system-mgw34-mt-d -l libboost_filesystem-mgw34-mt-d \ -l db_cxx \ -l eay32 \ - -l wxmsw29ud_html -l wxmsw29ud_core -l wxmsw29ud_adv -l wxbase29ud -l wxtiffd -l wxjpegd -l wxpngd -l wxzlibd \ -l kernel32 -l user32 -l gdi32 -l comdlg32 -l winspool -l winmm -l shell32 -l comctl32 -l ole32 -l oleaut32 -l uuid -l rpcrt4 -l advapi32 -l ws2_32 -l shlwapi WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH DEBUGFLAGS=-g -D__WXDEBUG__ CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h - +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h \ + script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h init.h sha.h all: bitcoin.exe -headers.h.gch: headers.h $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/util.o: util.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/script.o: script.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/db.o: db.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/net.o: net.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/main.o: main.cpp $(HEADERS) sha.h +headers.h.gch: headers.h $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/ui.o: ui.cpp $(HEADERS) +obj/%.o: %.cpp $(HEADERS) headers.h.gch g++ -c $(CFLAGS) -o $@ $< -obj/uibase.o: uibase.cpp uibase.h - g++ -c $(CFLAGS) -o $@ $< - -obj/sha.o: sha.cpp sha.h +obj/sha.o: sha.cpp sha.h g++ -c $(CFLAGS) -O3 -o $@ $< -obj/irc.o: irc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/rpc.o: rpc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - obj/ui_res.o: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp windres $(WXDEFS) $(INCLUDEPATHS) -o $@ -i $< +OBJS= \ + obj/util.o \ + obj/script.o \ + obj/db.o \ + obj/net.o \ + obj/irc.o \ + obj/main.o \ + obj/rpc.o \ + obj/init.o + +bitcoin.exe: $(OBJS) obj/ui.o obj/uibase.o obj/sha.o obj/ui_res.o + g++ $(CFLAGS) -mwindows -Wl,--subsystem,windows -o $@ $(LIBPATHS) $^ $(WXLIBS) $(LIBS) + +obj/nogui/%.o: %.cpp $(HEADERS) + g++ -c $(CFLAGS) -DwxUSE_GUI=0 -o $@ $< -OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o \ - obj/ui_res.o +bitcoind.exe: $(OBJS:obj/%=obj/nogui/%) obj/sha.o obj/ui_res.o + g++ $(CFLAGS) -o $@ $(LIBPATHS) $^ -l wxbase29ud $(LIBS) -bitcoin.exe: headers.h.gch $(OBJS) - -kill /f bitcoin.exe - g++ $(CFLAGS) -mwindows -Wl,--subsystem,windows -o $@ $(LIBPATHS) $(OBJS) $(LIBS) clean: -del /Q obj\* + -del /Q obj\nogui\* -del /Q headers.h.gch diff --git a/makefile.unix b/makefile.unix index a8ec9367..10c4eece 100644 --- a/makefile.unix +++ b/makefile.unix @@ -13,66 +13,61 @@ LIBPATHS= \ -L"/usr/lib" \ -L"/usr/local/lib" +WXLIBS= \ + -Wl,-Bstatic \ + -l wx_gtk2ud-2.9 \ + -Wl,-Bdynamic \ + -l gtk-x11-2.0 -l SM + LIBS= \ -Wl,-Bstatic \ -l boost_system-mt -l boost_filesystem-mt \ -l db_cxx \ - -l wx_gtk2ud-2.9 \ -Wl,-Bdynamic \ -l crypto \ - -l gtk-x11-2.0 -l gthread-2.0 -l SM + -l gthread-2.0 WXDEFS=-D__WXGTK__ -DNOPCH DEBUGFLAGS=-g -D__WXDEBUG__ CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h - +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h \ + script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h init.h sha.h all: bitcoin -headers.h.gch: headers.h $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/util.o: util.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/script.o: script.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/db.o: db.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/net.o: net.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< - -obj/main.o: main.cpp $(HEADERS) sha.h - g++ -c $(CFLAGS) -o $@ $< - -obj/ui.o: ui.cpp $(HEADERS) +headers.h.gch: headers.h $(HEADERS) g++ -c $(CFLAGS) -o $@ $< -obj/uibase.o: uibase.cpp uibase.h +obj/%.o: %.cpp $(HEADERS) headers.h.gch g++ -c $(CFLAGS) -o $@ $< -obj/sha.o: sha.cpp sha.h +obj/sha.o: sha.cpp sha.h g++ -c $(CFLAGS) -O3 -o $@ $< -obj/irc.o: irc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< +OBJS= \ + obj/util.o \ + obj/script.o \ + obj/db.o \ + obj/net.o \ + obj/irc.o \ + obj/main.o \ + obj/rpc.o \ + obj/init.o -obj/rpc.o: rpc.cpp $(HEADERS) - g++ -c $(CFLAGS) -o $@ $< +bitcoin: $(OBJS) obj/ui.o obj/uibase.o obj/sha.o + g++ $(CFLAGS) -o $@ $(LIBPATHS) $^ $(WXLIBS) $(LIBS) +obj/nogui/%.o: %.cpp $(HEADERS) + g++ -c $(CFLAGS) -DwxUSE_GUI=0 -o $@ $< -OBJS=obj/util.o obj/script.o obj/db.o obj/net.o obj/main.o \ - obj/ui.o obj/uibase.o obj/sha.o obj/irc.o obj/rpc.o +bitcoind: $(OBJS:obj/%=obj/nogui/%) obj/sha.o + g++ $(CFLAGS) -o $@ $(LIBPATHS) $^ -l wx_baseud-2.9 $(LIBS) -bitcoin: headers.h.gch $(OBJS) - g++ $(CFLAGS) -o $@ $(LIBPATHS) $(OBJS) $(LIBS) clean: -rm obj/* + -rm obj/nogui/* -rm headers.h.gch diff --git a/makefile.vc b/makefile.vc index c407be3d..41f13522 100644 --- a/makefile.vc +++ b/makefile.vc @@ -28,57 +28,80 @@ LIBS= \ WXDEFS=/DWIN32 /D__WXMSW__ /D_WINDOWS /DNOPCH DEBUGFLAGS=/Zi /Od /D__WXDEBUG__ CFLAGS=/c /nologo /Ob0 /MDd /EHsc /GR /Zm300 $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) -HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h - +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h \ + script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h init.h sha.h all: bitcoin.exe -obj\util.obj: util.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s +.cpp{obj}.obj: + cl $(CFLAGS) /Fo$@ %s + +obj\util.obj: $(HEADERS) -obj\script.obj: script.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s +obj\script.obj: $(HEADERS) -obj\db.obj: db.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s +obj\db.obj: $(HEADERS) -obj\net.obj: net.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s +obj\net.obj: $(HEADERS) -obj\main.obj: main.cpp $(HEADERS) sha.h - cl $(CFLAGS) /Fo$@ %s +obj\irc.obj: $(HEADERS) -obj\ui.obj: ui.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s +obj\main.obj: $(HEADERS) -obj\uibase.obj: uibase.cpp uibase.h - cl $(CFLAGS) /Fo$@ %s +obj\rpc.obj: $(HEADERS) -obj\sha.obj: sha.cpp sha.h - cl $(CFLAGS) /O2 /Fo$@ %s +obj\init.obj: $(HEADERS) -obj\irc.obj: irc.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s +obj\ui.obj: $(HEADERS) -obj\rpc.obj: rpc.cpp $(HEADERS) - cl $(CFLAGS) /Fo$@ %s +obj\uibase.obj: $(HEADERS) + +obj\sha.obj: sha.cpp sha.h + cl $(CFLAGS) /O2 /Fo$@ %s obj\ui.res: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp - rc $(INCLUDEPATHS) $(WXDEFS) /Fo$@ %s + rc $(INCLUDEPATHS) $(WXDEFS) /Fo$@ %s + +OBJS= \ + obj\util.obj \ + obj\script.obj \ + obj\db.obj \ + obj\net.obj \ + obj\irc.obj \ + obj\main.obj \ + obj\rpc.obj \ + obj\init.obj + +bitcoin.exe: $(OBJS) obj\ui.obj obj\uibase.obj obj\sha.obj obj\ui.res + link /nologo /DEBUG /SUBSYSTEM:WINDOWS /OUT:$@ $(LIBPATHS) $** $(LIBS) + + +.cpp{obj\nogui}.obj: + cl $(CFLAGS) /DwxUSE_GUI=0 /Fo$@ %s + +obj\nogui\util.obj: $(HEADERS) + +obj\nogui\script.obj: $(HEADERS) + +obj\nogui\db.obj: $(HEADERS) + +obj\nogui\net.obj: $(HEADERS) + +obj\nogui\irc.obj: $(HEADERS) + +obj\nogui\main.obj: $(HEADERS) +obj\nogui\rpc.obj: $(HEADERS) +obj\nogui\init.obj: $(HEADERS) -OBJS=obj\util.obj obj\script.obj obj\db.obj obj\net.obj obj\main.obj \ - obj\ui.obj obj\uibase.obj obj\sha.obj obj\irc.obj obj\rpc.obj \ - obj\ui.res +bitcoind.exe: $(OBJS:obj\=obj\nogui\) obj\sha.obj obj\ui.res + link /nologo /DEBUG /OUT:$@ $(LIBPATHS) $** $(LIBS) -bitcoin.exe: $(OBJS) - -kill /f bitcoin.exe & sleep 1 - link /nologo /DEBUG /SUBSYSTEM:WINDOWS /OUT:$@ $(LIBPATHS) $** $(LIBS) clean: - -del /Q obj\* - -del *.ilk - -del *.pdb + -del /Q obj\* + -del *.ilk + -del *.pdb diff --git a/rpc.cpp b/rpc.cpp index 7f3e71db..f298137a 100644 --- a/rpc.cpp +++ b/rpc.cpp @@ -676,18 +676,18 @@ int CommandLineRPC(int argc, char *argv[]) string strResult = (result.type() == str_type ? result.get_str() : write_string(result, true)); if (result.type() != null_type) { - if (fWindows) + if (fWindows && fGUI) // Windows GUI apps can't print to command line, - // so for now settle for a message box yuck - wxMessageBox(strResult.c_str(), "Bitcoin", wxOK); + // so settle for a message box yuck + MyMessageBox(strResult.c_str(), "Bitcoin", wxOK); else fprintf(stdout, "%s\n", strResult.c_str()); } return 0; } catch (std::exception& e) { - if (fWindows) - wxMessageBox(strprintf("error: %s\n", e.what()).c_str(), "Bitcoin", wxOK); + if (fWindows && fGUI) + MyMessageBox(strprintf("error: %s\n", e.what()).c_str(), "Bitcoin", wxOK); else fprintf(stderr, "error: %s\n", e.what()); } catch (...) { diff --git a/serialize.h b/serialize.h index 8fb72a0a..5399c23d 100644 --- a/serialize.h +++ b/serialize.h @@ -19,7 +19,7 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 206; +static const int VERSION = 207; static const char* pszSubVer = ".0"; diff --git a/ui.cpp b/ui.cpp index 748b4dad..efb05385 100644 --- a/ui.cpp +++ b/ui.cpp @@ -7,38 +7,18 @@ #include #endif -bool GetStartOnSystemStartup(); -void SetStartOnSystemStartup(bool fAutoStart); - DEFINE_EVENT_TYPE(wxEVT_UITHREADCALL) CMainFrame* pframeMain = NULL; CMyTaskBarIcon* ptaskbaricon = NULL; -extern int g_isPainting; bool fClosedToTray = false; -// Settings -int fMinimizeToTray = true; -int fMinimizeOnClose = true; -int MyMessageBox(const wxString& message, const wxString& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1) -{ - if (fDaemon) - { - printf("wxMessageBox %s: %s\n", string(caption).c_str(), string(message).c_str()); - fprintf(stderr, "%s: %s\n", string(caption).c_str(), string(message).c_str()); - return wxOK; - } - return wxMessageBox(message, caption, style, parent, x, y); -} -#define wxMessageBox MyMessageBox - - @@ -48,47 +28,6 @@ int MyMessageBox(const wxString& message, const wxString& caption="Message", int // Util // -void ExitTimeout(void* parg) -{ -#ifdef __WXMSW__ - Sleep(5000); - ExitProcess(0); -#endif -} - -void Shutdown(void* parg) -{ - static CCriticalSection cs_Shutdown; - static bool fTaken; - bool fFirstThread; - CRITICAL_BLOCK(cs_Shutdown) - { - fFirstThread = !fTaken; - fTaken = true; - } - static bool fExit; - if (fFirstThread) - { - fShutdown = true; - nTransactionsUpdated++; - DBFlush(false); - StopNode(); - DBFlush(true); - CreateThread(ExitTimeout, NULL); - Sleep(50); - printf("Bitcoin exiting\n\n"); - fExit = true; - exit(0); - } - else - { - while (!fExit) - Sleep(500); - Sleep(100); - ExitThread(0); - } -} - void HandleCtrlA(wxKeyEvent& event) { // Ctrl-a select all @@ -312,7 +251,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_staticTextBalance->SetSize(140, 17); // resize to fit ubuntu's huge default font dResize = 1.22; - SetSize(dResize * GetSize().GetWidth(), 1.09 * GetSize().GetHeight()); + SetSize(dResize * GetSize().GetWidth(), 1.15 * GetSize().GetHeight()); #endif m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); @@ -1630,7 +1569,6 @@ void COptionsDialog::OnButtonApply(wxCommandEvent& event) - ////////////////////////////////////////////////////////////////////////////// // // CAboutDialog @@ -2014,7 +1952,7 @@ void CSendingDialog::StartTransfer() // Make sure we have enough money if (nPrice + nTransactionFee > GetBalance()) { - Error(_("You don't have enough money")); + Error(_("Insufficient funds")); return; } @@ -2081,7 +2019,7 @@ void CSendingDialog::OnReply2(CDataStream& vRecv) return; if (nPrice + nTransactionFee > GetBalance()) { - Error(_("You don't have enough money")); + Error(_("Insufficient funds")); return; } CKey key; @@ -2562,602 +2500,21 @@ wxMenu* CMyTaskBarIcon::CreatePopupMenu() -////////////////////////////////////////////////////////////////////////////// -// -// CMyApp -// - -// Define a new application -class CMyApp: public wxApp -{ -public: - wxLocale m_locale; - - CMyApp(){}; - ~CMyApp(){}; - bool OnInit(); - bool OnInit2(); - int OnExit(); - - // Hook Initialize so we can start without GUI - virtual bool Initialize(int& argc, wxChar** argv); - - // 2nd-level exception handling: we get all the exceptions occurring in any - // event handler here - virtual bool OnExceptionInMainLoop(); - - // 3rd, and final, level exception handling: whenever an unhandled - // exception is caught, this function is called - virtual void OnUnhandledException(); - - // and now for something different: this function is called in case of a - // crash (e.g. dereferencing null pointer, division by 0, ...) - virtual void OnFatalException(); -}; - -IMPLEMENT_APP(CMyApp) - -bool CMyApp::Initialize(int& argc, wxChar** argv) -{ - if (argc > 1 && argv[1][0] != '-' && (!fWindows || argv[1][0] != '/') && - wxString(argv[1]) != "start") - { - fCommandLine = true; - } - else - { - // wxApp::Initialize will remove environment-specific parameters, - // so it's too early to call ParseParameters yet - for (int i = 1; i < argc; i++) - { - wxString str = argv[i]; - #ifdef __WXMSW__ - if (str.size() >= 1 && str[0] == '/') - str[0] = '-'; - str = str.MakeLower(); - #endif - // haven't decided which argument to use for this yet - if (str == "-daemon" || str == "-d" || str == "start") - fDaemon = true; - } - } - -#ifdef __WXGTK__ - if (fDaemon || fCommandLine) - { - // Call the original Initialize while suppressing error messages - // and ignoring failure. If unable to initialize GTK, it fails - // near the end so hopefully the last few things don't matter. - { - wxLogNull logNo; - wxApp::Initialize(argc, argv); - } - - if (fDaemon) - { - fprintf(stdout, "bitcoin server starting\n"); - - // Daemonize - pid_t pid = fork(); - if (pid < 0) - { - fprintf(stderr, "Error: fork() returned %d errno %d\n", pid, errno); - return false; - } - if (pid > 0) - pthread_exit((void*)0); - } - - return true; - } -#endif - - return wxApp::Initialize(argc, argv); -} - -bool CMyApp::OnInit() -{ - bool fRet = false; - try - { - fRet = OnInit2(); - } - catch (std::exception& e) { - PrintException(&e, "OnInit()"); - } catch (...) { - PrintException(NULL, "OnInit()"); - } - if (!fRet) - Shutdown(NULL); - return fRet; -} - -bool CMyApp::OnInit2() -{ -#ifdef _MSC_VER - // Turn off microsoft heap dump noise - _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE); - _CrtSetReportFile(_CRT_WARN, CreateFileA("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); -#endif -#if defined(__WXMSW__) && defined(__WXDEBUG__) - // Disable malfunctioning wxWidgets debug assertion - g_isPainting = 10000; -#endif - wxImage::AddHandler(new wxPNGHandler); -#ifdef __WXMSW__ - SetAppName("Bitcoin"); -#else - SetAppName("bitcoin"); - umask(077); -#endif -#ifdef __WXMSW__ -#if wxUSE_UNICODE - // Hack to set wxConvLibc codepage to UTF-8 on Windows, - // may break if wxMBConv_win32 implementation in strconv.cpp changes. - class wxMBConv_win32 : public wxMBConv - { - public: - long m_CodePage; - size_t m_minMBCharWidth; - }; - if (((wxMBConv_win32*)&wxConvLibc)->m_CodePage == CP_ACP) - ((wxMBConv_win32*)&wxConvLibc)->m_CodePage = CP_UTF8; -#endif -#endif - - // Load locale//LC_MESSAGES/bitcoin.mo language file - m_locale.Init(wxLANGUAGE_DEFAULT, 0); - m_locale.AddCatalogLookupPathPrefix("locale"); - if (!fWindows) - { - m_locale.AddCatalogLookupPathPrefix("/usr/share/locale"); - m_locale.AddCatalogLookupPathPrefix("/usr/local/share/locale"); - } - m_locale.AddCatalog("wxstd"); // wxWidgets standard translations, if any - m_locale.AddCatalog("bitcoin"); - - // - // Parameters - // - if (fCommandLine) - { - int ret = CommandLineRPC(argc, argv); - exit(ret); - } - - ParseParameters(argc, argv); - if (mapArgs.count("-?") || mapArgs.count("--help")) - { - wxString strUsage = string() + - _("Usage: bitcoin [options]") + "\t\t\t\t\t\t\n" + - _("Options:\n") + - " -gen \t\t " + _("Generate coins\n") + - " -gen=0 \t\t " + _("Don't generate coins\n") + - " -min \t\t " + _("Start minimized\n") + - " -datadir= \t " + _("Specify data directory\n") + - " -proxy=\t " + _("Connect through socks4 proxy\n") + - " -addnode= \t " + _("Add a node to connect to\n") + - " -connect= \t " + _("Connect only to the specified node\n") + - " -? \t\t " + _("This help message\n"); - - if (fWindows) - { - // Remove spaces, the tabs make the columns line up in the message box - for (int i = 0; i < 50; i++) - strUsage.Replace(" \t", "\t"); - wxMessageBox(strUsage, "Bitcoin", wxOK); - } - else - { - // Remove tabs - strUsage.Replace("\t", ""); - fprintf(stderr, "%s", ((string)strUsage).c_str()); - } - return false; - } - - if (mapArgs.count("-datadir")) - strlcpy(pszSetDataDir, mapArgs["-datadir"].c_str(), sizeof(pszSetDataDir)); - - if (mapArgs.count("-debug")) - fDebug = true; - - if (mapArgs.count("-printtodebugger")) - fPrintToDebugger = true; - - if (!fDebug && !pszSetDataDir[0]) - ShrinkDebugFile(); - printf("\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n"); - printf("Bitcoin version 0.%d.%d%s beta, OS version %s\n", VERSION/100, VERSION%100, pszSubVer, ((string)wxGetOsDescription()).c_str()); - printf("System default language is %d %s\n", m_locale.GetSystemLanguage(), ((string)m_locale.GetSysName()).c_str()); - printf("Language file %s (%s)\n", (string("locale/") + (string)m_locale.GetCanonicalName() + "/LC_MESSAGES/bitcoin.mo").c_str(), ((string)m_locale.GetLocale()).c_str()); - - if (mapArgs.count("-loadblockindextest")) - { - CTxDB txdb("r"); - txdb.LoadBlockIndex(); - PrintBlockTree(); - return false; - } - - // - // Limit to single instance per user - // Required to protect the database files if we're going to keep deleting log.* - // -#ifdef __WXMSW__ - // todo: wxSingleInstanceChecker wasn't working on Linux, never deleted its lock file - // maybe should go by whether successfully bind port 8333 instead - wxString strMutexName = wxString("bitcoin_running.") + getenv("HOMEPATH"); - for (int i = 0; i < strMutexName.size(); i++) - if (!isalnum(strMutexName[i])) - strMutexName[i] = '.'; - wxSingleInstanceChecker* psingleinstancechecker = new wxSingleInstanceChecker(strMutexName); - if (psingleinstancechecker->IsAnotherRunning()) - { - printf("Existing instance found\n"); - unsigned int nStart = GetTime(); - loop - { - // TODO: find out how to do this in Linux, or replace with wxWidgets commands - // Show the previous instance and exit - HWND hwndPrev = FindWindowA("wxWindowClassNR", "Bitcoin"); - if (hwndPrev) - { - if (IsIconic(hwndPrev)) - ShowWindow(hwndPrev, SW_RESTORE); - SetForegroundWindow(hwndPrev); - return false; - } - - if (GetTime() > nStart + 60) - return false; - - // Resume this instance if the other exits - delete psingleinstancechecker; - Sleep(1000); - psingleinstancechecker = new wxSingleInstanceChecker(strMutexName); - if (!psingleinstancechecker->IsAnotherRunning()) - break; - } - } -#endif - - // Bind to the port early so we can tell if another instance is already running. - // This is a backup to wxSingleInstanceChecker, which doesn't work on Linux. - string strErrors; - if (!BindListenPort(strErrors)) - { - wxMessageBox(strErrors, "Bitcoin"); - return false; - } - - // - // Load data files - // - bool fFirstRun; - strErrors = ""; - int64 nStart; - - printf("Loading addresses...\n"); - nStart = GetTimeMillis(); - if (!LoadAddresses()) - strErrors += _("Error loading addr.dat \n"); - printf(" addresses %15"PRI64d"ms\n", GetTimeMillis() - nStart); - - printf("Loading block index...\n"); - nStart = GetTimeMillis(); - if (!LoadBlockIndex()) - strErrors += _("Error loading blkindex.dat \n"); - printf(" block index %15"PRI64d"ms\n", GetTimeMillis() - nStart); - - printf("Loading wallet...\n"); - nStart = GetTimeMillis(); - if (!LoadWallet(fFirstRun)) - strErrors += _("Error loading wallet.dat \n"); - printf(" wallet %15"PRI64d"ms\n", GetTimeMillis() - nStart); - - printf("Done loading\n"); - - //// debug print - printf("mapBlockIndex.size() = %d\n", mapBlockIndex.size()); - printf("nBestHeight = %d\n", nBestHeight); - printf("mapKeys.size() = %d\n", mapKeys.size()); - printf("mapPubKeys.size() = %d\n", mapPubKeys.size()); - printf("mapWallet.size() = %d\n", mapWallet.size()); - printf("mapAddressBook.size() = %d\n", mapAddressBook.size()); - - if (!strErrors.empty()) - { - wxMessageBox(strErrors, "Bitcoin"); - return false; - } - - // Add wallet transactions that aren't already in a block to mapTransactions - ReacceptWalletTransactions(); - - // - // Parameters - // - if (mapArgs.count("-printblockindex") || mapArgs.count("-printblocktree")) - { - PrintBlockTree(); - return false; - } - - if (mapArgs.count("-printblock")) - { - string strMatch = mapArgs["-printblock"]; - int nFound = 0; - for (map::iterator mi = mapBlockIndex.begin(); mi != mapBlockIndex.end(); ++mi) - { - uint256 hash = (*mi).first; - if (strncmp(hash.ToString().c_str(), strMatch.c_str(), strMatch.size()) == 0) - { - CBlockIndex* pindex = (*mi).second; - CBlock block; - block.ReadFromDisk(pindex); - block.BuildMerkleTree(); - block.print(); - printf("\n"); - nFound++; - } - } - if (nFound == 0) - printf("No blocks matching %s were found\n", strMatch.c_str()); - return false; - } - - if (mapArgs.count("-gen")) - { - if (mapArgs["-gen"].empty()) - fGenerateBitcoins = true; - else - fGenerateBitcoins = (atoi(mapArgs["-gen"].c_str()) != 0); - } - - if (mapArgs.count("-proxy")) - { - fUseProxy = true; - addrProxy = CAddress(mapArgs["-proxy"]); - if (!addrProxy.IsValid()) - { - wxMessageBox(_("Invalid -proxy address"), "Bitcoin"); - return false; - } - } - - if (mapArgs.count("-addnode")) - { - foreach(string strAddr, mapMultiArgs["-addnode"]) - { - CAddress addr(strAddr, NODE_NETWORK); - addr.nTime = 0; // so it won't relay unless successfully connected - if (addr.IsValid()) - AddAddress(addr); - } - } - - // - // Create the main frame window - // - if (!fDaemon) - { - pframeMain = new CMainFrame(NULL); - if (mapArgs.count("-min")) - pframeMain->Iconize(true); - pframeMain->Show(true); // have to show first to get taskbar button to hide - if (fMinimizeToTray && pframeMain->IsIconized()) - fClosedToTray = true; - pframeMain->Show(!fClosedToTray); - ptaskbaricon->Show(fMinimizeToTray || fClosedToTray); - - CreateThread(ThreadDelayedRepaint, NULL); - } - - if (!CheckDiskSpace()) - return false; - - RandAddSeedPerfmon(); - - if (!CreateThread(StartNode, NULL)) - wxMessageBox("Error: CreateThread(StartNode) failed", "Bitcoin"); - - if (mapArgs.count("-server") || fDaemon) - CreateThread(ThreadRPCServer, NULL); - - if (fFirstRun) - SetStartOnSystemStartup(true); - - - // - // Tests - // -#ifdef __WXMSW__ - if (argc >= 2 && stricmp(argv[1], "-send") == 0) -#else - if (argc >= 2 && strcmp(argv[1], "-send") == 0) -#endif - { - int64 nValue = 1; - if (argc >= 3) - ParseMoney(argv[2], nValue); - - string strAddress; - if (argc >= 4) - strAddress = argv[3]; - CAddress addr(strAddress); - - CWalletTx wtx; - wtx.mapValue["to"] = strAddress; - wtx.mapValue["from"] = addrLocalHost.ToString(); - wtx.mapValue["message"] = "command line send"; - - // Send to IP address - CSendingDialog* pdialog = new CSendingDialog(pframeMain, addr, nValue, wtx); - if (!pdialog->ShowModal()) - return false; - } - - return true; -} - -int CMyApp::OnExit() -{ - Shutdown(NULL); - return wxApp::OnExit(); -} - -bool CMyApp::OnExceptionInMainLoop() -{ - try - { - throw; - } - catch (std::exception& e) - { - PrintException(&e, "CMyApp::OnExceptionInMainLoop()"); - wxLogWarning("Exception %s %s", typeid(e).name(), e.what()); - Sleep(1000); - throw; - } - catch (...) - { - PrintException(NULL, "CMyApp::OnExceptionInMainLoop()"); - wxLogWarning("Unknown exception"); - Sleep(1000); - throw; - } - return true; -} - -void CMyApp::OnUnhandledException() -{ - // this shows how we may let some exception propagate uncaught - try - { - throw; - } - catch (std::exception& e) - { - PrintException(&e, "CMyApp::OnUnhandledException()"); - wxLogWarning("Exception %s %s", typeid(e).name(), e.what()); - Sleep(1000); - throw; - } - catch (...) - { - PrintException(NULL, "CMyApp::OnUnhandledException()"); - wxLogWarning("Unknown exception"); - Sleep(1000); - throw; - } -} - -void CMyApp::OnFatalException() -{ - wxMessageBox(_("Program has crashed and will terminate. "), "Bitcoin", wxOK | wxICON_ERROR); -} - - - - - -#ifdef __WXMSW__ -typedef WINSHELLAPI BOOL (WINAPI *PSHGETSPECIALFOLDERPATHA)(HWND hwndOwner, LPSTR lpszPath, int nFolder, BOOL fCreate); - -string MyGetSpecialFolderPath(int nFolder, bool fCreate) -{ - char pszPath[MAX_PATH+100] = ""; - - // SHGetSpecialFolderPath is not usually available on NT 4.0 - HMODULE hShell32 = LoadLibraryA("shell32.dll"); - if (hShell32) - { - PSHGETSPECIALFOLDERPATHA pSHGetSpecialFolderPath = - (PSHGETSPECIALFOLDERPATHA)GetProcAddress(hShell32, "SHGetSpecialFolderPathA"); - if (pSHGetSpecialFolderPath) - (*pSHGetSpecialFolderPath)(NULL, pszPath, nFolder, fCreate); - FreeModule(hShell32); - } - - // Backup option - if (pszPath[0] == '\0') - { - if (nFolder == CSIDL_STARTUP) - { - strcpy(pszPath, getenv("USERPROFILE")); - strcat(pszPath, "\\Start Menu\\Programs\\Startup"); - } - else if (nFolder == CSIDL_APPDATA) - { - strcpy(pszPath, getenv("APPDATA")); - } - } - - return pszPath; -} -string StartupShortcutPath() -{ - return MyGetSpecialFolderPath(CSIDL_STARTUP, true) + "\\Bitcoin.lnk"; -} -bool GetStartOnSystemStartup() -{ - return wxFileExists(StartupShortcutPath()); -} -void SetStartOnSystemStartup(bool fAutoStart) +void CreateMainWindow() { - // If the shortcut exists already, remove it for updating - remove(StartupShortcutPath().c_str()); - - if (fAutoStart) - { - CoInitialize(NULL); - - // Get a pointer to the IShellLink interface. - IShellLink* psl = NULL; - HRESULT hres = CoCreateInstance(CLSID_ShellLink, NULL, - CLSCTX_INPROC_SERVER, IID_IShellLink, - reinterpret_cast(&psl)); - - if (SUCCEEDED(hres)) - { - // Get the current executable path - TCHAR pszExePath[MAX_PATH]; - GetModuleFileName(NULL, pszExePath, sizeof(pszExePath)); - - // Set the path to the shortcut target - psl->SetPath(pszExePath); - PathRemoveFileSpec(pszExePath); - psl->SetWorkingDirectory(pszExePath); - psl->SetShowCmd(SW_SHOWMINNOACTIVE); - - // Query IShellLink for the IPersistFile interface for - // saving the shortcut in persistent storage. - IPersistFile* ppf = NULL; - hres = psl->QueryInterface(IID_IPersistFile, - reinterpret_cast(&ppf)); - if (SUCCEEDED(hres)) - { - WCHAR pwsz[MAX_PATH]; - // Ensure that the string is ANSI. - MultiByteToWideChar(CP_ACP, 0, StartupShortcutPath().c_str(), -1, pwsz, MAX_PATH); - // Save the link by calling IPersistFile::Save. - hres = ppf->Save(pwsz, TRUE); - ppf->Release(); - } - psl->Release(); - } - CoUninitialize(); - } + pframeMain = new CMainFrame(NULL); + if (mapArgs.count("-min")) + pframeMain->Iconize(true); + pframeMain->Show(true); // have to show first to get taskbar button to hide + if (fMinimizeToTray && pframeMain->IsIconized()) + fClosedToTray = true; + pframeMain->Show(!fClosedToTray); + ptaskbaricon->Show(fMinimizeToTray || fClosedToTray); + CreateThread(ThreadDelayedRepaint, NULL); } -#else -bool GetStartOnSystemStartup() { return false; } -void SetStartOnSystemStartup(bool fAutoStart) { } -#endif - diff --git a/ui.h b/ui.h index acabd551..e944eb93 100644 --- a/ui.h +++ b/ui.h @@ -4,26 +4,61 @@ DECLARE_EVENT_TYPE(wxEVT_UITHREADCALL, -1) +#if wxUSE_GUI +static const bool fGUI=true; +#else +static const bool fGUI=false; +#endif -extern map mapArgs; +inline int MyMessageBox(const wxString& message, const wxString& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1) +{ +#if wxUSE_GUI + if (!fDaemon) + return wxMessageBox(message, caption, style, parent, x, y); +#endif + printf("wxMessageBox %s: %s\n", string(caption).c_str(), string(message).c_str()); + fprintf(stderr, "%s: %s\n", string(caption).c_str(), string(message).c_str()); + return wxOK; +} +#define wxMessageBox MyMessageBox -// Settings -extern int fMinimizeToTray; -extern int fMinimizeOnClose; void HandleCtrlA(wxKeyEvent& event); string FormatTxStatus(const CWalletTx& wtx); void UIThreadCall(boost::function0); -void MainFrameRepaint(); -void Shutdown(void* parg); int ThreadSafeMessageBox(const string& message, const string& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1); bool ThreadSafeAskFee(int64 nFeeRequired, const string& strCaption, wxWindow* parent); +void MainFrameRepaint(); +void CreateMainWindow(); + +#if !wxUSE_GUI +inline int ThreadSafeMessageBox(const string& message, const string& caption, int style, wxWindow* parent, int x, int y) +{ + return MyMessageBox(message, caption, style, parent, x, y); +} + +inline bool ThreadSafeAskFee(int64 nFeeRequired, const string& strCaption, wxWindow* parent) +{ + return true; +} + +inline void MainFrameRepaint() +{ +} + +inline void CreateMainWindow() +{ +} +#else // wxUSE_GUI + + + class CMainFrame : public CMainFrameBase @@ -331,3 +366,5 @@ public: DECLARE_EVENT_TABLE() }; + +#endif // wxUSE_GUI diff --git a/util.cpp b/util.cpp index 6ea735d1..d90cdc48 100644 --- a/util.cpp +++ b/util.cpp @@ -459,6 +459,7 @@ const char* wxGetTranslation(const char* pszEnglish) mapCache[pszEnglish] = pszCached; return pszCached; } + return NULL; } @@ -502,8 +503,8 @@ void PrintException(std::exception* pex, const char* pszThread) FormatException(pszMessage, pex, pszThread); printf("\n\n************************\n%s\n", pszMessage); fprintf(stderr, "\n\n************************\n%s\n", pszMessage); - if (wxTheApp && !fDaemon) - wxMessageBox(pszMessage, "Error", wxOK | wxICON_ERROR); + if (wxTheApp && !fDaemon && fGUI) + MyMessageBox(pszMessage, "Error", wxOK | wxICON_ERROR); throw; //DebugBreak(); } From 6557910ccf36e098b1a0de4841be7c04c4ffd58a Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 26 Feb 2010 16:09:43 +0000 Subject: [PATCH 060/133] wx/utils.h and wx/clipbrd.h not needed on bitcoind wxbase build git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@72 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- headers.h | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/headers.h b/headers.h index 3c8d32ca..44a00deb 100644 --- a/headers.h +++ b/headers.h @@ -19,11 +19,11 @@ #define WIN32_LEAN_AND_MEAN 1 #define __STDC_LIMIT_MACROS // to enable UINT64_MAX from stdint.h #include -#include -#include #include -#include +#include #if wxUSE_GUI +#include +#include #include #endif #include From d7d80a74d58152453cfb0c71a08f6b424d2493c9 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Sun, 28 Feb 2010 15:00:32 +0000 Subject: [PATCH 061/133] fix unsafe string handling in wxGetTranslation git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@73 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- makefile.vc | 18 +++++++++--------- serialize.h | 2 +- util.cpp | 8 ++++---- 3 files changed, 14 insertions(+), 14 deletions(-) diff --git a/makefile.vc b/makefile.vc index 41f13522..9f821771 100644 --- a/makefile.vc +++ b/makefile.vc @@ -36,7 +36,7 @@ all: bitcoin.exe .cpp{obj}.obj: - cl $(CFLAGS) /Fo$@ %s + cl $(CFLAGS) /Fo$@ %s obj\util.obj: $(HEADERS) @@ -59,10 +59,10 @@ obj\ui.obj: $(HEADERS) obj\uibase.obj: $(HEADERS) obj\sha.obj: sha.cpp sha.h - cl $(CFLAGS) /O2 /Fo$@ %s + cl $(CFLAGS) /O2 /Fo$@ %s obj\ui.res: ui.rc rc/bitcoin.ico rc/check.ico rc/send16.bmp rc/send16mask.bmp rc/send16masknoshadow.bmp rc/send20.bmp rc/send20mask.bmp rc/addressbook16.bmp rc/addressbook16mask.bmp rc/addressbook20.bmp rc/addressbook20mask.bmp - rc $(INCLUDEPATHS) $(WXDEFS) /Fo$@ %s + rc $(INCLUDEPATHS) $(WXDEFS) /Fo$@ %s OBJS= \ obj\util.obj \ @@ -75,11 +75,11 @@ OBJS= \ obj\init.obj bitcoin.exe: $(OBJS) obj\ui.obj obj\uibase.obj obj\sha.obj obj\ui.res - link /nologo /DEBUG /SUBSYSTEM:WINDOWS /OUT:$@ $(LIBPATHS) $** $(LIBS) + link /nologo /DEBUG /SUBSYSTEM:WINDOWS /OUT:$@ $(LIBPATHS) $** $(LIBS) .cpp{obj\nogui}.obj: - cl $(CFLAGS) /DwxUSE_GUI=0 /Fo$@ %s + cl $(CFLAGS) /DwxUSE_GUI=0 /Fo$@ %s obj\nogui\util.obj: $(HEADERS) @@ -98,10 +98,10 @@ obj\nogui\rpc.obj: $(HEADERS) obj\nogui\init.obj: $(HEADERS) bitcoind.exe: $(OBJS:obj\=obj\nogui\) obj\sha.obj obj\ui.res - link /nologo /DEBUG /OUT:$@ $(LIBPATHS) $** $(LIBS) + link /nologo /DEBUG /OUT:$@ $(LIBPATHS) $** $(LIBS) clean: - -del /Q obj\* - -del *.ilk - -del *.pdb + -del /Q obj\* + -del *.ilk + -del *.pdb diff --git a/serialize.h b/serialize.h index 5399c23d..77dfa95b 100644 --- a/serialize.h +++ b/serialize.h @@ -20,7 +20,7 @@ class CDataStream; class CAutoFile; static const int VERSION = 207; -static const char* pszSubVer = ".0"; +static const char* pszSubVer = ".1"; diff --git a/util.cpp b/util.cpp index d90cdc48..09de89b1 100644 --- a/util.cpp +++ b/util.cpp @@ -445,17 +445,17 @@ const char* wxGetTranslation(const char* pszEnglish) return (*mi).second; // wxWidgets translation - const char* pszTranslated = wxGetTranslation(wxString(pszEnglish, wxConvUTF8)).utf8_str(); + wxString strTranslated = wxGetTranslation(wxString(pszEnglish, wxConvUTF8)); // We don't cache unknown strings because caller might be passing in a // dynamic string and we would keep allocating memory for each variation. - if (strcmp(pszEnglish, pszTranslated) == 0) + if (strcmp(pszEnglish, strTranslated.utf8_str()) == 0) return pszEnglish; // Add to cache, memory doesn't need to be freed. We only cache because // we must pass back a pointer to permanently allocated memory. - char* pszCached = new char[strlen(pszTranslated)+1]; - strcpy(pszCached, pszTranslated); + char* pszCached = new char[strlen(strTranslated.utf8_str())+1]; + strcpy(pszCached, strTranslated.utf8_str()); mapCache[pszEnglish] = pszCached; return pszCached; } From 2cffa7ce315d9b98d35192d16927b44d21b9e1a7 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 5 Mar 2010 01:13:27 +0000 Subject: [PATCH 062/133] fixed runaway memory alloc bug on 64-bit in ParseString found by sirius-m git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@74 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- util.cpp | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/util.cpp b/util.cpp index 09de89b1..acfbcd6c 100644 --- a/util.cpp +++ b/util.cpp @@ -282,15 +282,21 @@ bool error(const char* format, ...) void ParseString(const string& str, char c, vector& v) { - unsigned int i1 = 0; - unsigned int i2; - do + if (str.empty()) + return; + string::size_type i1 = 0; + string::size_type i2; + loop { i2 = str.find(c, i1); + if (i2 == str.npos) + { + v.push_back(str.substr(i1)); + return; + } v.push_back(str.substr(i1, i2-i1)); i1 = i2+1; } - while (i2 != str.npos); } From 966cca4bd4a6d869583fb47f5f66ed4672c007ff Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Fri, 5 Mar 2010 02:55:09 +0000 Subject: [PATCH 063/133] misc git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@75 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- irc.cpp | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/irc.cpp b/irc.cpp index 9d563cce..de0b997a 100644 --- a/irc.cpp +++ b/irc.cpp @@ -76,6 +76,8 @@ bool RecvLine(SOCKET hSocket, string& strLine) if (c == '\r') return true; strLine += c; + if (strLine.size() >= 9000) + return true; } else if (nBytes <= 0) { @@ -109,7 +111,7 @@ bool RecvLineIRC(SOCKET hSocket, string& strLine) return false; vector vWords; ParseString(strLine, ' ', vWords); - if (vWords[0] == "PING") + if (vWords.size() >= 1 && vWords[0] == "PING") { strLine[1] = 'O'; strLine += '\r'; @@ -156,6 +158,7 @@ bool Wait(int nSeconds) void ThreadIRCSeed(void* parg) { + printf("ThreadIRCSeed started\n"); SetThreadPriority(THREAD_PRIORITY_NORMAL); int nErrorWait = 10; int nRetryWait = 10; From 2d98de1b3ae2ade8b1e5493bc63c0c1d776deeb1 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 19 May 2010 00:26:56 +0000 Subject: [PATCH 064/133] Mac OS build fixes by laszlo -- version 0.2.8 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@76 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- init.cpp | 8 +++++++- main.cpp | 6 +++--- net.cpp | 4 +--- net.h | 43 +++++++++++++++++++++++++++---------------- rpc.cpp | 23 ++++++++++++++--------- serialize.h | 4 ++-- ui.cpp | 15 ++++++++++++--- uibase.cpp | 14 +++++++------- uibase.h | 45 ++++++++++++++++++++++----------------------- uiproject.fbp | 14 +++++++------- util.cpp | 6 ++++++ util.h | 1 + 12 files changed, 109 insertions(+), 74 deletions(-) diff --git a/init.cpp b/init.cpp index 06d8f4c9..db8886e1 100644 --- a/init.cpp +++ b/init.cpp @@ -284,6 +284,10 @@ bool CMyApp::OnInit2() _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE); _CrtSetReportFile(_CRT_WARN, CreateFileA("NUL", GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, 0)); #endif +#if _MSC_VER >= 1400 + // Disable confusing "helpful" text message on abort, ctrl-c + _set_abort_behavior(0, _WRITE_ABORT_MSG | _CALL_REPORTFAULT); +#endif #if defined(__WXMSW__) && defined(__WXDEBUG__) && wxUSE_GUI // Disable malfunctioning wxWidgets debug assertion g_isPainting = 10000; @@ -291,10 +295,12 @@ bool CMyApp::OnInit2() #if wxUSE_GUI wxImage::AddHandler(new wxPNGHandler); #endif -#ifdef __WXMSW__ +#if defined(__WXMSW__ ) || defined(__WXMAC__) SetAppName("Bitcoin"); #else SetAppName("bitcoin"); +#endif +#ifndef __WXMSW__ umask(077); #endif #ifdef __WXMSW__ diff --git a/main.cpp b/main.cpp index f7416230..ad0208a2 100644 --- a/main.cpp +++ b/main.cpp @@ -1338,7 +1338,7 @@ bool CBlock::AcceptBlock() // Don't relay old inventory during initial block download. // Please keep this number updated to a few thousand below current block count. - if (hashBestChain == hash && nBestHeight > 40000) + if (hashBestChain == hash && nBestHeight > 55000) RelayInventory(CInv(MSG_BLOCK, hash)); // // Add atoms to user reviews for coins created @@ -2255,9 +2255,9 @@ bool SendMessages(CNode* pto) // Delay tx inv messages to protect privacy, // trickle them out to a few nodes at a time. bool fSendTxInv = false; - if (GetTimeMillis() - pto->nLastSentTxInv > 1800 + GetRand(200)) + if (GetTimeMillis() > pto->nNextSendTxInv) { - pto->nLastSentTxInv = GetTimeMillis(); + pto->nNextSendTxInv = GetTimeMillis() + 3000 + GetRand(2000); fSendTxInv = true; } diff --git a/net.cpp b/net.cpp index 8e8127d7..fbfa7462 100644 --- a/net.cpp +++ b/net.cpp @@ -927,9 +927,7 @@ void ThreadOpenConnections2(void* parg) continue; // Only try the old stuff if we don't have enough connections - if (vNodes.size() >= 2 && nSinceLastSeen > 7 * 24 * 60 * 60) - continue; - if (vNodes.size() >= 5 && nSinceLastSeen > 24 * 60 * 60) + if (vNodes.size() >= 8 && nSinceLastSeen > 24 * 60 * 60) continue; // If multiple addresses are ready, prioritize by time since diff --git a/net.h b/net.h index c7d15460..1593a975 100644 --- a/net.h +++ b/net.h @@ -45,6 +45,7 @@ bool StopNode(); // (4) message start // (12) command // (4) size +// (4) checksum // The message start string is designed to be unlikely to occur in normal data. // The characters are rarely used upper ascii, not valid as UTF-8, and produce @@ -58,6 +59,7 @@ public: char pchMessageStart[sizeof(::pchMessageStart)]; char pchCommand[COMMAND_SIZE]; unsigned int nMessageSize; + //unsigned int nChecksum; CMessageHeader() { @@ -65,6 +67,7 @@ public: memset(pchCommand, 0, sizeof(pchCommand)); pchCommand[1] = 1; nMessageSize = -1; + //nChecksum = 0; } CMessageHeader(const char* pszCommand, unsigned int nMessageSizeIn) @@ -79,6 +82,8 @@ public: READWRITE(FLATDATA(pchMessageStart)); READWRITE(FLATDATA(pchCommand)); READWRITE(nMessageSize); + //if (nVersion >= 209 && GetCommand() != "version") + // READWRITE(nChecksum); ) string GetCommand() @@ -484,7 +489,8 @@ public: int64 nLastRecv; int64 nLastSendEmpty; int64 nTimeConnected; - unsigned int nPushPos; + unsigned int nHeaderStart; + unsigned int nMessageStart; CAddress addr; int nVersion; bool fClient; @@ -512,7 +518,7 @@ public: vector vInventoryToSend; CCriticalSection cs_inventory; multimap mapAskFor; - int64 nLastSentTxInv; + int64 nNextSendTxInv; // publish and subscription vector vfSubscribe; @@ -528,7 +534,8 @@ public: nLastRecv = 0; nLastSendEmpty = GetTime(); nTimeConnected = GetTime(); - nPushPos = -1; + nHeaderStart = -1; + nMessageStart = -1; addr = addrIn; nVersion = 0; fClient = false; // set by version message @@ -542,6 +549,7 @@ public: pindexLastGetBlocksBegin = 0; hashLastGetBlocksEnd = 0; fGetAddr = false; + nNextSendTxInv = 0; vfSubscribe.assign(256, false); // Push a version message @@ -639,10 +647,11 @@ public: void BeginMessage(const char* pszCommand) { cs_vSend.Enter(); - if (nPushPos != -1) + if (nHeaderStart != -1) AbortMessage(); - nPushPos = vSend.size(); + nHeaderStart = vSend.size(); vSend << CMessageHeader(pszCommand, 0); + nMessageStart = vSend.size(); if (fDebug) printf("%s ", DateTimeStrFormat("%x %H:%M:%S", GetTime()).c_str()); printf("sending: %s ", pszCommand); @@ -650,10 +659,11 @@ public: void AbortMessage() { - if (nPushPos == -1) + if (nHeaderStart == -1) return; - vSend.resize(nPushPos); - nPushPos = -1; + vSend.resize(nHeaderStart); + nHeaderStart = -1; + nMessageStart = -1; cs_vSend.Leave(); printf("(aborted)\n"); } @@ -667,25 +677,26 @@ public: return; } - if (nPushPos == -1) + if (nHeaderStart == -1) return; // Patch in the size - unsigned int nSize = vSend.size() - nPushPos - sizeof(CMessageHeader); - memcpy((char*)&vSend[nPushPos] + offsetof(CMessageHeader, nMessageSize), &nSize, sizeof(nSize)); + unsigned int nSize = vSend.size() - nMessageStart; + memcpy((char*)&vSend[nHeaderStart] + offsetof(CMessageHeader, nMessageSize), &nSize, sizeof(nSize)); printf("(%d bytes) ", nSize); printf("\n"); - nPushPos = -1; + nHeaderStart = -1; + nMessageStart = -1; cs_vSend.Leave(); } void EndMessageAbortIfEmpty() { - if (nPushPos == -1) + if (nHeaderStart == -1) return; - int nSize = vSend.size() - nPushPos - sizeof(CMessageHeader); + int nSize = vSend.size() - nMessageStart; if (nSize > 0) EndMessage(); else @@ -694,9 +705,9 @@ public: const char* GetMessageCommand() const { - if (nPushPos == -1) + if (nHeaderStart == -1) return ""; - return &vSend[nPushPos] + offsetof(CMessageHeader, pchCommand); + return &vSend[nHeaderStart] + offsetof(CMessageHeader, pchCommand); } diff --git a/rpc.cpp b/rpc.cpp index f298137a..c56ce263 100644 --- a/rpc.cpp +++ b/rpc.cpp @@ -77,24 +77,28 @@ Value getconnectioncount(const Array& params) } -Value getdifficulty(const Array& params) +double GetDifficulty() { - if (params.size() != 0) - throw runtime_error( - "getdifficulty (no parameters)\n" - "Returns the proof-of-work difficulty as a multiple of the minimum difficulty."); - - if (pindexBest == NULL) - throw runtime_error("block chain not loaded"); - // Floating point number that is a multiple of the minimum difficulty, // minimum difficulty = 1.0. + if (pindexBest == NULL) + return 1.0; int nShift = 256 - 32 - 31; // to fit in a uint double dMinimum = (CBigNum().SetCompact(bnProofOfWorkLimit.GetCompact()) >> nShift).getuint(); double dCurrently = (CBigNum().SetCompact(pindexBest->nBits) >> nShift).getuint(); return dMinimum / dCurrently; } +Value getdifficulty(const Array& params) +{ + if (params.size() != 0) + throw runtime_error( + "getdifficulty (no parameters)\n" + "Returns the proof-of-work difficulty as a multiple of the minimum difficulty."); + + return GetDifficulty(); +} + Value getbalance(const Array& params) { @@ -157,6 +161,7 @@ Value getinfo(const Array& params) obj.push_back(Pair("proxy", (fUseProxy ? addrProxy.ToStringIPPort() : string()))); obj.push_back(Pair("generate", (bool)fGenerateBitcoins)); obj.push_back(Pair("genproclimit", (int)(fLimitProcessors ? nLimitProcessors : -1))); + obj.push_back(Pair("difficulty", (double)GetDifficulty())); return obj; } diff --git a/serialize.h b/serialize.h index 77dfa95b..c47dddc8 100644 --- a/serialize.h +++ b/serialize.h @@ -19,8 +19,8 @@ class CScript; class CDataStream; class CAutoFile; -static const int VERSION = 207; -static const char* pszSubVer = ".1"; +static const int VERSION = 208; +static const char* pszSubVer = ".0"; diff --git a/ui.cpp b/ui.cpp index efb05385..5c7311f9 100644 --- a/ui.cpp +++ b/ui.cpp @@ -256,11 +256,20 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) m_staticTextBalance->SetLabel(FormatMoney(GetBalance()) + " "); m_listCtrl->SetFocus(); ptaskbaricon = new CMyTaskBarIcon(); +#ifdef __WXMAC__ + // Mac automatically moves wxID_EXIT, wxID_PREFERENCES and wxID_ABOUT + // to their standard places, leaving these menus empty. + GetMenuBar()->Remove(2); // remove Help menu + GetMenuBar()->Remove(0); // remove File menu +#endif // Init column headers int nDateWidth = DateTimeStr(1229413914).size() * 6 + 8; if (!strstr(DateTimeStr(1229413914).c_str(), "2008")) nDateWidth += 12; +#ifdef __WXMAC__ + nDateWidth += 2; +#endif wxListCtrl* pplistCtrl[] = {m_listCtrlAll, m_listCtrlSentReceived, m_listCtrlSent, m_listCtrlReceived}; foreach(wxListCtrl* p, pplistCtrl) { @@ -274,7 +283,7 @@ CMainFrame::CMainFrame(wxWindow* parent) : CMainFrameBase(parent) } // Init status bar - int pnWidths[3] = { -100, 88, 290 }; + int pnWidths[3] = { -100, 88, 300 }; #ifndef __WXMSW__ pnWidths[1] = pnWidths[1] * 1.1 * dResize; pnWidths[2] = pnWidths[2] * 1.1 * dResize; @@ -2157,7 +2166,7 @@ CAddressBookDialog::CAddressBookDialog(wxWindow* parent, const wxString& strInit bool fMine = (AddressToHash160(strAddress, hash160) && mapPubKeys.count(hash160)); wxListCtrl* plistCtrl = fMine ? m_listCtrlReceiving : m_listCtrlSending; int nIndex = InsertLine(plistCtrl, strName, strAddress); - if (strAddress == (fMine ? strDefaultReceiving : strInitSelected)) + if (strAddress == (fMine ? strDefaultReceiving : string(strInitSelected))) plistCtrl->SetItemState(nIndex, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED, wxLIST_STATE_SELECTED|wxLIST_STATE_FOCUSED); } } @@ -2444,7 +2453,7 @@ void CMyTaskBarIcon::OnMenuRestore(wxCommandEvent& event) void CMyTaskBarIcon::OnMenuOptions(wxCommandEvent& event) { // Since it's modal, get the main window to do it - wxCommandEvent event2(wxEVT_COMMAND_MENU_SELECTED, wxID_MENUOPTIONSOPTIONS); + wxCommandEvent event2(wxEVT_COMMAND_MENU_SELECTED, wxID_PREFERENCES); pframeMain->GetEventHandler()->AddPendingEvent(event2); } diff --git a/uibase.cpp b/uibase.cpp index 03ca13db..5fd83ee4 100644 --- a/uibase.cpp +++ b/uibase.cpp @@ -24,7 +24,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_menuFile = new wxMenu(); wxMenuItem* m_menuFileExit; - m_menuFileExit = new wxMenuItem( m_menuFile, wxID_ANY, wxString( _("E&xit") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuFileExit = new wxMenuItem( m_menuFile, wxID_EXIT, wxString( _("E&xit") ) , wxEmptyString, wxITEM_NORMAL ); m_menuFile->Append( m_menuFileExit ); m_menubar->Append( m_menuFile, _("&File") ); @@ -39,14 +39,14 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& m_menuOptions->Append( m_menuOptionsChangeYourAddress ); wxMenuItem* m_menuOptionsOptions; - m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_MENUOPTIONSOPTIONS, wxString( _("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuOptionsOptions = new wxMenuItem( m_menuOptions, wxID_PREFERENCES, wxString( _("&Options...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuOptions->Append( m_menuOptionsOptions ); m_menubar->Append( m_menuOptions, _("&Settings") ); m_menuHelp = new wxMenu(); wxMenuItem* m_menuHelpAbout; - m_menuHelpAbout = new wxMenuItem( m_menuHelp, wxID_ANY, wxString( _("&About...") ) , wxEmptyString, wxITEM_NORMAL ); + m_menuHelpAbout = new wxMenuItem( m_menuHelp, wxID_ABOUT, wxString( _("&About...") ) , wxEmptyString, wxITEM_NORMAL ); m_menuHelp->Append( m_menuHelpAbout ); m_menubar->Append( m_menuHelp, _("&Help") ); @@ -133,7 +133,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& wxBoxSizer* bSizer11; bSizer11 = new wxBoxSizer( wxVERTICAL ); - m_listCtrlAll = new wxListCtrl( m_panel9, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + m_listCtrlAll = new wxListCtrl( m_panel9, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING ); bSizer11->Add( m_listCtrlAll, 1, wxEXPAND, 5 ); m_panel9->SetSizer( bSizer11 ); @@ -144,7 +144,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& wxBoxSizer* bSizer111; bSizer111 = new wxBoxSizer( wxVERTICAL ); - m_listCtrlSentReceived = new wxListCtrl( m_panel91, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + m_listCtrlSentReceived = new wxListCtrl( m_panel91, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING ); bSizer111->Add( m_listCtrlSentReceived, 1, wxEXPAND, 5 ); m_panel91->SetSizer( bSizer111 ); @@ -155,7 +155,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& wxBoxSizer* bSizer112; bSizer112 = new wxBoxSizer( wxVERTICAL ); - m_listCtrlSent = new wxListCtrl( m_panel92, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + m_listCtrlSent = new wxListCtrl( m_panel92, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING ); bSizer112->Add( m_listCtrlSent, 1, wxEXPAND, 5 ); m_panel92->SetSizer( bSizer112 ); @@ -166,7 +166,7 @@ CMainFrameBase::CMainFrameBase( wxWindow* parent, wxWindowID id, const wxString& wxBoxSizer* bSizer113; bSizer113 = new wxBoxSizer( wxVERTICAL ); - m_listCtrlReceived = new wxListCtrl( m_panel93, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING|wxVSCROLL ); + m_listCtrlReceived = new wxListCtrl( m_panel93, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_NO_SORT_HEADER|wxLC_REPORT|wxLC_SORT_DESCENDING ); bSizer113->Add( m_listCtrlReceived, 1, wxEXPAND, 5 ); m_panel93->SetSizer( bSizer113 ); diff --git a/uibase.h b/uibase.h index 1686cbfa..e2f0a1c0 100644 --- a/uibase.h +++ b/uibase.h @@ -42,29 +42,28 @@ #define wxID_MAINFRAME 1000 #define wxID_OPTIONSGENERATEBITCOINS 1001 -#define wxID_MENUOPTIONSOPTIONS 1002 -#define wxID_BUTTONSEND 1003 -#define wxID_BUTTONRECEIVE 1004 -#define wxID_TEXTCTRLADDRESS 1005 -#define wxID_BUTTONNEW 1006 -#define wxID_BUTTONCOPY 1007 -#define wxID_TRANSACTIONFEE 1008 -#define wxID_PROXYIP 1009 -#define wxID_PROXYPORT 1010 -#define wxID_TEXTCTRLPAYTO 1011 -#define wxID_BUTTONPASTE 1012 -#define wxID_BUTTONADDRESSBOOK 1013 -#define wxID_TEXTCTRLAMOUNT 1014 -#define wxID_CHOICETRANSFERTYPE 1015 -#define wxID_LISTCTRL 1016 -#define wxID_BUTTONRENAME 1017 -#define wxID_PANELSENDING 1018 -#define wxID_LISTCTRLSENDING 1019 -#define wxID_PANELRECEIVING 1020 -#define wxID_LISTCTRLRECEIVING 1021 -#define wxID_BUTTONDELETE 1022 -#define wxID_BUTTONEDIT 1023 -#define wxID_TEXTCTRL 1024 +#define wxID_BUTTONSEND 1002 +#define wxID_BUTTONRECEIVE 1003 +#define wxID_TEXTCTRLADDRESS 1004 +#define wxID_BUTTONNEW 1005 +#define wxID_BUTTONCOPY 1006 +#define wxID_TRANSACTIONFEE 1007 +#define wxID_PROXYIP 1008 +#define wxID_PROXYPORT 1009 +#define wxID_TEXTCTRLPAYTO 1010 +#define wxID_BUTTONPASTE 1011 +#define wxID_BUTTONADDRESSBOOK 1012 +#define wxID_TEXTCTRLAMOUNT 1013 +#define wxID_CHOICETRANSFERTYPE 1014 +#define wxID_LISTCTRL 1015 +#define wxID_BUTTONRENAME 1016 +#define wxID_PANELSENDING 1017 +#define wxID_LISTCTRLSENDING 1018 +#define wxID_PANELRECEIVING 1019 +#define wxID_LISTCTRLRECEIVING 1020 +#define wxID_BUTTONDELETE 1021 +#define wxID_BUTTONEDIT 1022 +#define wxID_TEXTCTRL 1023 /////////////////////////////////////////////////////////////////////////////// /// Class CMainFrameBase diff --git a/uiproject.fbp b/uiproject.fbp index 8643fbab..990af521 100644 --- a/uiproject.fbp +++ b/uiproject.fbp @@ -123,7 +123,7 @@ 0 1 - wxID_ANY + wxID_EXIT wxITEM_NORMAL E&xit m_menuFileExit @@ -173,7 +173,7 @@ 0 1 - wxID_MENUOPTIONSOPTIONS + wxID_PREFERENCES wxITEM_NORMAL &Options... m_menuOptionsOptions @@ -193,7 +193,7 @@ 0 1 - wxID_ANY + wxID_ABOUT wxITEM_NORMAL &About... m_menuHelpAbout @@ -924,7 +924,7 @@ - wxVSCROLL + @@ -1047,7 +1047,7 @@ - wxVSCROLL + @@ -1170,7 +1170,7 @@ - wxVSCROLL + @@ -1293,7 +1293,7 @@ - wxVSCROLL + diff --git a/util.cpp b/util.cpp index acfbcd6c..8f6bf38c 100644 --- a/util.cpp +++ b/util.cpp @@ -314,6 +314,12 @@ string FormatMoney(int64 n, bool fPlus) return str; } + +bool ParseMoney(const string& str, int64& nRet) +{ + return ParseMoney(str.c_str(), nRet); +} + bool ParseMoney(const char* pszIn, int64& nRet) { string strWhole; diff --git a/util.h b/util.h index 75f0956f..1a25c7dd 100644 --- a/util.h +++ b/util.h @@ -133,6 +133,7 @@ void PrintException(std::exception* pex, const char* pszThread); void LogException(std::exception* pex, const char* pszThread); void ParseString(const string& str, char c, vector& v); string FormatMoney(int64 n, bool fPlus=false); +bool ParseMoney(const string& str, int64& nRet); bool ParseMoney(const char* pszIn, int64& nRet); vector ParseHex(const char* psz); vector ParseHex(const std::string& str); From 124baa4ccbf3ecb78eab7ecdd40105fb54ee0369 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Tue, 25 May 2010 23:58:27 +0000 Subject: [PATCH 065/133] build-osx.txt build instructions and makefile.osx from Laszlo git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@77 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- build-osx.txt | 209 ++++++++++++++++++++++++++++++++++++++++++++++++++ makefile.osx | 65 ++++++++++++++++ 2 files changed, 274 insertions(+) create mode 100644 build-osx.txt create mode 100644 makefile.osx diff --git a/build-osx.txt b/build-osx.txt new file mode 100644 index 00000000..c7e3a218 --- /dev/null +++ b/build-osx.txt @@ -0,0 +1,209 @@ +Mac OS X build instructions +Laszlo Hanyecz (solar@heliacal.net) + + +Tested on 10.5 and 10.6 intel. PPC is not supported because it's big-endian. + +All of the commands should be executed in Terminal.app.. it's in +/Applications/Utilities + +You need to install XCode with all the options checked so that the compiler +and everything is available in /usr not just /Developer +I think it comes on the DVD but you can get the current version from +http://developer.apple.com + + +1. Pick a directory to work inside.. something like ~/bitcoin works. The +structure I use looks like this: +(~ is your home directory) + +~/bitcoin +~/bitcoin/trunk # source code +~/bitcoin/deps # dependencies.. like libraries and headers needed to compile +~/bitcoin/Bitcoin.app # the application bundle where you can run the app + +Just execute: mkdir ~/bitcoin +This will create the top dir for you.. + +WARNING: do not use the ~ notation with the configure scripts.. use the full +name of the directory, for example /Users/james/bitcoin/deps for a user named +'james'. In my examples I am using 'macosuser' so make sure you change that. + +2. Check out the trunk version of the bitcoin code from subversion: + +cd ~/bitcoin +svn checkout https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk + +This will make ~/bitcoin/trunk for you with all the files from subversion. + +3. Get and build the dependencies + + +Boost +----- + +Download from http://www.boost.org/users/download/ +I'm assuming it ended up in ~/Downloads.. + +mkdir ~/bitcoin/deps +cd ~/bitcoin/deps +tar xvjf ~/Downloads/boost_1_42_0.tar.bz2 +cd boost_1_42_0 +./bootstrap.sh +./bjam architecture=combined address-model=32_64 macosx-version=10.6 macosx-version-min=10.5 link=static runtime-link=static --toolset=darwin --prefix=/Users/macosuser/bitcoin/deps install + +This part takes a while.. use your judgement and fix it if something doesn't +build for some reason. + +Change the prefix to whatever your directory is (my username in this example +is macosuser). I'm also running on 10.6 so i have macosx-version=10.6 change +to 10.5 if you're using leopard. + +This is what my output looked like at the end: +...failed updating 2 targets... +...skipped 144 targets... +...updated 8074 targets... + + +OpenSSL +------- + +Download from http://www.openssl.org/source/ + +We would like to build this as a 32 bit/64 bit library so we actually build it +2 times and join it together here.. If you downloaded with safari it already +uncompressed it so it will just be a tar not a tar.gz + +cd ~/bitcoin/deps +tar xvf ~/Downloads/openssl-1.0.0.tar +mv openssl-1.0.0 openssl-1.0.0-i386 +tar xvf ~/Downloads/openssl-1.0.0.tar +mv openssl-1.0.0 openssl-1.0.0-x86_64 +# build i386 (32 bit intel) binary +cd openssl-1.0.0-i386 +./Configure --prefix=/Users/macosuser/bitcoin/deps --openssldir=/Users/macosuser/deps/openssl darwin-i386-cc && make +make install # only do this on one of the architectures, to install the headers +cd .. +# build x86_64 (64 bit intel) binary +cd openssl-1.0.0-x86_64 +./Configure --prefix=/Users/macosuser/bitcoin/deps --openssldir=/Users/macosuser/deps/openssl darwin64-x86_64-cc && make +cd .. + +# combine the libs +cd ~/bitcoin/deps +lipo -arch i386 openssl-1.0.0-i386/libcrypto.a -arch x86_64 openssl-1.0.0-x86_64/libcrypto.a -o lib/libcrypto.a -create +lipo -arch i386 openssl-1.0.0-i386/libssl.a -arch x86_64 openssl-1.0.0-x86_64/libssl.a -o lib/libssl.a -create + +Verify your binaries + +file lib/libcrypto.a + +output should look like this: + +ib/libcrypto.a: Mach-O universal binary with 2 architectures +lib/libcrypto.a (for architecture i386): current ar archive random library +lib/libcrypto.a (for architecture x86_64): current ar archive random library + + +Berkeley DB +----------- + +Download from http://freshmeat.net/projects/berkeleydb/ + +cd ~/bitcoin/deps +tar xvf ~/Downloads/db-4.8.26.tar +cd db-4.8.26/build_unix +../dist/configure --prefix=/Users/macosuser/bitcoin/deps --enable-cxx && make && make install + + +wxWidgets +--------- + +This is the big one.. + +Check it out from svn + +cd ~/bitcoin/deps +svn checkout http://svn.wxwidgets.org/svn/wx/wxWidgets/trunk wxWidgets-trunk + +This will make a wxWidgets-trunk directory in deps. + +Use this script snippet, change your prefix to whatever your dir is: + +PREFIX=~/bitcoin/deps +SRCDIR="$PREFIX/wxWidgets-trunk" +BUILDDIR="$SRCDIR/macbuild" + +cd "$PREFIX" && +#svn checkout http://svn.wxwidgets.org/svn/wx/wxWidgets/trunk wxWidgets-trunk && +cd "$SRCDIR" && + +[ -f include/wx/hashmap.h.orig ] || cp include/wx/hashmap.h include/wx/hashmap.h.orig && +sed 's/if wxUSE_STL/if 0 \&\& wxUSE_STL/g' < include/wx/hashmap.h.orig > include/wx/hashmap.h && + +[ -f include/wx/hashset.h.orig ] || cp include/wx/hashset.h include/wx/hashset.h.orig && +sed 's/if wxUSE_STL/if 0 \&\& wxUSE_STL/g' < include/wx/hashset.h.orig > include/wx/hashset.h && + + + +rm -vrf "$BUILDDIR" && +mkdir "$BUILDDIR" && +cd "$BUILDDIR" && + +../configure --prefix="$PREFIX" \ +--with-osx_cocoa \ +--disable-shared \ +--disable-debug_flag \ +--with-macosx-version-min=10.5 \ +--enable-stl \ +--enable-utf8 \ +--enable-universal_binary \ +--with-libjpeg=builtin \ +--with-libpng=builtin \ +--with-regex=builtin \ +--with-libtiff=builtin \ +--with-zlib=builtin \ +--with-expat=builtin \ +--with-macosx-sdk=/Developer/SDKs/MacOSX10.5.sdk && + + +find . -name Makefile | +while read i; do + echo $i; + sed 's/-arch i386/-arch i386 -arch x86_64/g' < "$i" > "$i".new && + mv "$i" "$i".old && + mv "$i".new "$i"; +done + + + +make && +make install + + + +Now you should be able to build bitcoin + +cd ~/bitcoin/trunk +make -f makefile.osx bitcoin + +Before you can run it, you need to create an application bundle for Mac OS. +Create the directories in terminal using mkdir and copy the files into place. +They are available at http://heliacal.net/~solar/bitcoin/mac-build/ +You need the Info.plist and the .ins file. The Contents/MacOS/bitcoin file is +the output of the build. +Your directory structure should look like this: + +Bitcoin.app +Bitcoin.app/Contents +Bitcoin.app/Contents/Info.plist +Bitcoin.app/Contents/MacOS +Bitcoin.app/Contents/MacOS/bitcoin +Bitcoin.app/Contents/Resources +Bitcoin.app/Contents/Resources/BitcoinAppIcon.icns + +To run it you can just click the Bitcoin.app in Finder, or just do open +~/bitcoin/Bitcoin.app +If you want to run it with arguments you can just run it without backgrounding +by specifying the full name in terminal: +~/bitcoin/Bitcoin.app/Contents/MacOS/bitcoin -addnode=192.75.207.66 diff --git a/makefile.osx b/makefile.osx new file mode 100644 index 00000000..264f0b16 --- /dev/null +++ b/makefile.osx @@ -0,0 +1,65 @@ +# Copyright (c) 2009-2010 Satoshi Nakamoto +# Distributed under the MIT/X11 software license, see the accompanying +# file license.txt or http://www.opensource.org/licenses/mit-license.php. + +# Mac OS X makefile for bitcoin +# Laszlo Hanyecz (solar@heliacal.net) + +DEPSDIR=/Users/macosuser/bitcoin/deps + +INCLUDEPATHS= \ + -I"$(DEPSDIR)/include" + +LIBPATHS= \ + -L"$(DEPSDIR)/lib" + +WXLIBS=$(shell $(DEPSDIR)/bin/wx-config --libs --static) + +LIBS= -dead_strip \ + $(DEPSDIR)/lib/libdb_cxx-4.8.a \ + $(DEPSDIR)/lib/libboost_system.a \ + $(DEPSDIR)/lib/libboost_filesystem.a \ + $(DEPSDIR)/lib/libcrypto.a + +WXDEFS=$(shell $(DEPSDIR)/bin/wx-config --cxxflags) -DNOPCH -DMSG_NOSIGNAL=0 + +DEBUGFLAGS=-g -DwxDEBUG_LEVEL=0 +# ppc doesn't work because we don't support big-endian +CFLAGS=-mmacosx-version-min=10.5 -arch i386 -arch x86_64 -O2 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h \ + script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h init.h sha.h + + +all: bitcoin + + +obj/%.o: %.cpp $(HEADERS) + g++ -c $(CFLAGS) -o $@ $< + +obj/sha.o: sha.cpp sha.h + g++ -c $(CFLAGS) -O3 -o $@ $< + +OBJS= \ + obj/util.o \ + obj/script.o \ + obj/db.o \ + obj/net.o \ + obj/irc.o \ + obj/main.o \ + obj/rpc.o \ + obj/init.o + +bitcoin: $(OBJS) obj/ui.o obj/uibase.o obj/sha.o + g++ $(CFLAGS) -o $@ $(LIBPATHS) $^ $(WXLIBS) $(LIBS) + + +obj/nogui/%.o: %.cpp $(HEADERS) + g++ -c $(CFLAGS) -DwxUSE_GUI=0 -o $@ $< + +bitcoind: $(OBJS:obj/%=obj/nogui/%) obj/sha.o + g++ $(CFLAGS) -o $@ $(LIBPATHS) $^ $(WXLIBS) $(LIBS) + + +clean: + -rm -f obj/*.o + -rm -f obj/nogui/*.o From 42605ce8bcc9bd01b86491c74fee14de77960868 Mon Sep 17 00:00:00 2001 From: s_nakamoto Date: Wed, 26 May 2010 00:05:26 +0000 Subject: [PATCH 066/133] better prevention of inventory relaying during initial download, message checksum between nodes with 0.2.9 or higher, optimization level up from -O0 to -O2, rpc functions: setlabel, getlabel, getaddressesbylabel, getreceivedbyaddress, getreceivedbylabel, listreceivedbyaddress, listreceivedbylabel -- version 0.2.9 git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@78 1a98c847-1fd6-4fd8-948a-caf3550aa51b --- db.h | 2 + init.cpp | 7 +- main.cpp | 76 +++++++----- makefile.mingw | 2 +- makefile.unix | 2 +- net.h | 30 ++++- rpc.cpp | 305 ++++++++++++++++++++++++++++++++++++++++--------- serialize.h | 14 ++- ui.cpp | 9 +- ui.h | 9 ++ uibase.h | 2 +- uiproject.fbp | 2 +- util.cpp | 5 - 13 files changed, 364 insertions(+), 101 deletions(-) diff --git a/db.h b/db.h index fb4e9268..0b778d5f 100644 --- a/db.h +++ b/db.h @@ -328,6 +328,8 @@ public: bool EraseName(const string& strAddress) { + // This should only be used for sending addresses, never for receiving addresses, + // receiving addresses must always have an address book entry if they're not change return. CRITICAL_BLOCK(cs_mapAddressBook) mapAddressBook.erase(strAddress); nWalletDBUpdated++; diff --git a/init.cpp b/init.cpp index db8886e1..833a8a3f 100644 --- a/init.cpp +++ b/init.cpp @@ -224,9 +224,6 @@ bool CMyApp::Initialize(int& argc, wxChar** argv) } } - if (fDaemon) - fprintf(stdout, "bitcoin server starting\n"); - #ifdef __WXGTK__ if (fDaemon || fCommandLine) { @@ -447,7 +444,8 @@ bool CMyApp::OnInit2() // // Load data files // - bool fFirstRun; + if (fDaemon) + fprintf(stdout, "bitcoin server starting\n"); strErrors = ""; int64 nStart; @@ -465,6 +463,7 @@ bool CMyApp::OnInit2() printf("Loading wallet...\n"); nStart = GetTimeMillis(); + bool fFirstRun; if (!LoadWallet(fFirstRun)) strErrors += _("Error loading wallet.dat \n"); printf(" wallet %15"PRI64d"ms\n", GetTimeMillis() - nStart); diff --git a/main.cpp b/main.cpp index ad0208a2..05a182c1 100644 --- a/main.cpp +++ b/main.cpp @@ -1336,19 +1336,12 @@ bool CBlock::AcceptBlock() if (!AddToBlockIndex(nFile, nBlockPos)) return error("AcceptBlock() : AddToBlockIndex failed"); - // Don't relay old inventory during initial block download. - // Please keep this number updated to a few thousand below current block count. - if (hashBestChain == hash && nBestHeight > 55000) - RelayInventory(CInv(MSG_BLOCK, hash)); - - // // Add atoms to user reviews for coins created - // vector vchPubKey; - // if (ExtractPubKey(vtx[0].vout[0].scriptPubKey, false, vchPubKey)) - // { - // unsigned short nAtom = GetRand(USHRT_MAX - 100) + 100; - // vector vAtoms(1, nAtom); - // AddAtomsAndPropagate(Hash(vchPubKey.begin(), vchPubKey.end()), vAtoms, true); - // } + // Relay inventory, but don't relay old inventory during initial block download + if (hashBestChain == hash) + CRITICAL_BLOCK(cs_vNodes) + foreach(CNode* pnode, vNodes) + if (nBestHeight > (pnode->nStartingHeight != -1 ? pnode->nStartingHeight - 2000 : 55000)) + pnode->PushInventory(CInv(MSG_BLOCK, hash)); return true; } @@ -1721,6 +1714,7 @@ bool ProcessMessages(CNode* pfrom) // (4) message start // (12) command // (4) size + // (4) checksum // (x) data // @@ -1728,12 +1722,13 @@ bool ProcessMessages(CNode* pfrom) { // Scan for message start CDataStream::iterator pstart = search(vRecv.begin(), vRecv.end(), BEGIN(pchMessageStart), END(pchMessageStart)); - if (vRecv.end() - pstart < sizeof(CMessageHeader)) + int nHeaderSize = vRecv.GetSerializeSize(CMessageHeader()); + if (vRecv.end() - pstart < nHeaderSize) { - if (vRecv.size() > sizeof(CMessageHeader)) + if (vRecv.size() > nHeaderSize) { printf("\n\nPROCESSMESSAGE MESSAGESTART NOT FOUND\n\n"); - vRecv.erase(vRecv.begin(), vRecv.end() - sizeof(CMessageHeader)); + vRecv.erase(vRecv.begin(), vRecv.end() - nHeaderSize); } break; } @@ -1742,6 +1737,7 @@ bool ProcessMessages(CNode* pfrom) vRecv.erase(vRecv.begin(), pstart); // Read header + vector vHeaderSave(vRecv.begin(), vRecv.begin() + nHeaderSize); CMessageHeader hdr; vRecv >> hdr; if (!hdr.IsValid()) @@ -1757,10 +1753,9 @@ bool ProcessMessages(CNode* pfrom) { // Rewind and wait for rest of message ///// need a mechanism to give up waiting for overlong message size error - //if (fDebug) - // printf("message-break\n"); - vRecv.insert(vRecv.begin(), BEGIN(hdr), END(hdr)); - Sleep(100); + if (fDebug) + printf("message-break\n"); + vRecv.insert(vRecv.begin(), vHeaderSave.begin(), vHeaderSave.end()); break; } @@ -1768,6 +1763,20 @@ bool ProcessMessages(CNode* pfrom) CDataStream vMsg(vRecv.begin(), vRecv.begin() + nMessageSize, vRecv.nType, vRecv.nVersion); vRecv.ignore(nMessageSize); + // Checksum + if (vRecv.GetVersion() >= 209) + { + uint256 hash = Hash(vMsg.begin(), vMsg.end()); + unsigned int nChecksum = 0; + memcpy(&nChecksum, &hash, sizeof(nChecksum)); + if (nChecksum != hdr.nChecksum) + { + printf("ProcessMessage(%s, %d bytes) : CHECKSUM ERROR nChecksum=%08x hdr.nChecksum=%08x\n", + strCommand.c_str(), nMessageSize, nChecksum, hdr.nChecksum); + continue; + } + } + // Process message bool fRet = false; try @@ -1844,6 +1853,9 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vRecv >> addrFrom >> nNonce; if (pfrom->nVersion >= 106 && !vRecv.empty()) vRecv >> strSubVer; + if (pfrom->nVersion >= 209 && !vRecv.empty()) + vRecv >> pfrom->nStartingHeight; + if (pfrom->nVersion == 0) return false; @@ -1854,9 +1866,6 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) return true; } - pfrom->vSend.SetVersion(min(pfrom->nVersion, VERSION)); - pfrom->vRecv.SetVersion(min(pfrom->nVersion, VERSION)); - pfrom->fClient = !(pfrom->nServices & NODE_NETWORK); if (pfrom->fClient) { @@ -1866,6 +1875,13 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) AddTimeData(pfrom->addr.ip, nTime); + // Change version + if (pfrom->nVersion >= 209) + pfrom->PushMessage("verack"); + pfrom->vSend.SetVersion(min(pfrom->nVersion, VERSION)); + if (pfrom->nVersion < 209) + pfrom->vRecv.SetVersion(min(pfrom->nVersion, VERSION)); + // Ask the first connected node for block updates static bool fAskedForBlocks; if (!fAskedForBlocks && !pfrom->fClient) @@ -1876,7 +1892,7 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) pfrom->fSuccessfullyConnected = true; - printf("version message: version %d\n", pfrom->nVersion); + printf("version message: version %d, blocks=%d\n", pfrom->nVersion, pfrom->nStartingHeight); } @@ -1887,6 +1903,12 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) } + else if (strCommand == "verack") + { + pfrom->vRecv.SetVersion(min(pfrom->nVersion, VERSION)); + } + + else if (strCommand == "addr") { vector vAddr; @@ -2101,9 +2123,8 @@ bool ProcessMessage(CNode* pfrom, string strCommand, CDataStream& vRecv) vRecv >> *pblock; //// debug print - // printf("received block:\n"); - // pblock->print(); printf("received block %s\n", pblock->GetHash().ToString().substr(0,16).c_str()); + // pblock->print(); CInv inv(MSG_BLOCK, pblock->GetHash()); pfrom->AddInventoryKnown(inv); @@ -2388,8 +2409,11 @@ void GenerateBitcoins(bool fGenerate) int nAddThreads = nProcessors - vnThreadsRunning[3]; printf("Starting %d BitcoinMiner threads\n", nAddThreads); for (int i = 0; i < nAddThreads; i++) + { if (!CreateThread(ThreadBitcoinMiner, NULL)) printf("Error: CreateThread(ThreadBitcoinMiner) failed\n"); + Sleep(10); + } } } diff --git a/makefile.mingw b/makefile.mingw index 449a9607..ee63d11e 100644 --- a/makefile.mingw +++ b/makefile.mingw @@ -29,7 +29,7 @@ LIBS= \ WXDEFS=-DWIN32 -D__WXMSW__ -D_WINDOWS -DNOPCH DEBUGFLAGS=-g -D__WXDEBUG__ -CFLAGS=-mthreads -O0 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +CFLAGS=-mthreads -O2 -w -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h \ script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h init.h sha.h diff --git a/makefile.unix b/makefile.unix index 10c4eece..e7729adc 100644 --- a/makefile.unix +++ b/makefile.unix @@ -29,7 +29,7 @@ LIBS= \ WXDEFS=-D__WXGTK__ -DNOPCH DEBUGFLAGS=-g -D__WXDEBUG__ -CFLAGS=-O0 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) +CFLAGS=-O2 -Wno-invalid-offsetof -Wformat $(DEBUGFLAGS) $(WXDEFS) $(INCLUDEPATHS) HEADERS=headers.h strlcpy.h serialize.h uint256.h util.h key.h bignum.h base58.h \ script.h db.h net.h irc.h main.h rpc.h uibase.h ui.h init.h sha.h diff --git a/net.h b/net.h index 1593a975..46344ed9 100644 --- a/net.h +++ b/net.h @@ -8,6 +8,7 @@ class CInv; class CRequestTracker; class CNode; class CBlockIndex; +extern int nBestHeight; @@ -59,7 +60,7 @@ public: char pchMessageStart[sizeof(::pchMessageStart)]; char pchCommand[COMMAND_SIZE]; unsigned int nMessageSize; - //unsigned int nChecksum; + unsigned int nChecksum; CMessageHeader() { @@ -67,7 +68,7 @@ public: memset(pchCommand, 0, sizeof(pchCommand)); pchCommand[1] = 1; nMessageSize = -1; - //nChecksum = 0; + nChecksum = 0; } CMessageHeader(const char* pszCommand, unsigned int nMessageSizeIn) @@ -75,6 +76,7 @@ public: memcpy(pchMessageStart, ::pchMessageStart, sizeof(pchMessageStart)); strncpy(pchCommand, pszCommand, COMMAND_SIZE); nMessageSize = nMessageSizeIn; + nChecksum = 0; } IMPLEMENT_SERIALIZE @@ -82,8 +84,8 @@ public: READWRITE(FLATDATA(pchMessageStart)); READWRITE(FLATDATA(pchCommand)); READWRITE(nMessageSize); - //if (nVersion >= 209 && GetCommand() != "version") - // READWRITE(nChecksum); + if (nVersion >= 209) + READWRITE(nChecksum); ) string GetCommand() @@ -475,6 +477,7 @@ extern CAddress addrProxy; + class CNode { public: @@ -507,6 +510,7 @@ public: uint256 hashContinue; CBlockIndex* pindexLastGetBlocksBegin; uint256 hashLastGetBlocksEnd; + int nStartingHeight; // flood vector vAddrToSend; @@ -529,7 +533,9 @@ public: nServices = 0; hSocket = hSocketIn; vSend.SetType(SER_NETWORK); + vSend.SetVersion(0); vRecv.SetType(SER_NETWORK); + vRecv.SetVersion(0); nLastSend = 0; nLastRecv = 0; nLastSendEmpty = GetTime(); @@ -548,6 +554,7 @@ public: hashContinue = 0; pindexLastGetBlocksBegin = 0; hashLastGetBlocksEnd = 0; + nStartingHeight = -1; fGetAddr = false; nNextSendTxInv = 0; vfSubscribe.assign(256, false); @@ -558,7 +565,8 @@ public: CAddress addrYou = (fUseProxy ? CAddress("0.0.0.0") : addr); CAddress addrMe = (fUseProxy ? CAddress("0.0.0.0") : addrLocalHost); RAND_bytes((unsigned char*)&nLocalHostNonce, sizeof(nLocalHostNonce)); - PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, nLocalHostNonce, string(pszSubVer)); + PushMessage("version", VERSION, nLocalServices, nTime, addrYou, addrMe, + nLocalHostNonce, string(pszSubVer), nBestHeight); } ~CNode() @@ -680,10 +688,20 @@ public: if (nHeaderStart == -1) return; - // Patch in the size + // Set the size unsigned int nSize = vSend.size() - nMessageStart; memcpy((char*)&vSend[nHeaderStart] + offsetof(CMessageHeader, nMessageSize), &nSize, sizeof(nSize)); + // Set the checksum + if (vSend.GetVersion() >= 209) + { + uint256 hash = Hash(vSend.begin() + nMessageStart, vSend.end()); + unsigned int nChecksum = 0; + memcpy(&nChecksum, &hash, sizeof(nChecksum)); + assert(nMessageStart - nHeaderStart >= offsetof(CMessageHeader, nChecksum) + sizeof(nChecksum)); + memcpy((char*)&vSend[nHeaderStart] + offsetof(CMessageHeader, nChecksum), &nChecksum, sizeof(nChecksum)); + } + printf("(%d bytes) ", nSize); printf("\n"); diff --git a/rpc.cpp b/rpc.cpp index c56ce263..d988786e 100644 --- a/rpc.cpp +++ b/rpc.cpp @@ -26,7 +26,7 @@ void ThreadRPCServer2(void* parg); /// -/// Note: I'm not finished designing this interface, it's still subject to change. +/// Note: This interface may still be subject to change. /// @@ -188,6 +188,73 @@ Value getnewaddress(const Array& params) } +Value setlabel(const Array& params) +{ + if (params.size() < 1 || params.size() > 2) + throw runtime_error( + "setlabel