2013-11-04 16:20:43 +01:00
// Copyright (c) 2011-2013 The Bitcoin developers
// Distributed under the MIT/X11 software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2013-01-23 21:51:02 +01:00
2011-05-12 14:49:42 +02:00
# include "bitcoingui.h"
2013-04-13 07:13:08 +02:00
2011-05-22 17:19:43 +02:00
# include "clientmodel.h"
2012-04-13 17:10:50 +02:00
# include "guiconstants.h"
2013-04-13 07:13:08 +02:00
# include "guiutil.h"
# include "intro.h"
# include "optionsmodel.h"
2013-02-12 00:52:30 +01:00
# include "paymentserver.h"
2013-04-14 11:35:37 +02:00
# include "splashscreen.h"
2013-04-13 07:13:08 +02:00
# include "walletmodel.h"
# include "init.h"
# include "main.h"
# include "ui_interface.h"
# include "util.h"
2013-12-13 16:14:48 +01:00
# include "wallet.h"
2011-05-07 22:13:39 +02:00
2013-04-13 07:13:08 +02:00
# include <stdint.h>
# include <boost/filesystem/operations.hpp>
2013-07-23 08:52:24 +02:00
# include <QApplication>
2013-04-13 07:13:08 +02:00
# include <QLibraryInfo>
# include <QLocale>
2011-06-05 16:03:29 +02:00
# include <QMessageBox>
2013-04-13 07:13:08 +02:00
# include <QSettings>
# include <QTimer>
# include <QTranslator>
2013-05-31 14:02:24 +02:00
# if QT_VERSION < 0x050000
2011-08-28 14:12:26 +02:00
# include <QTextCodec>
2013-05-31 14:02:24 +02:00
# endif
2011-06-05 16:03:29 +02:00
2012-03-24 16:52:43 +01:00
# if defined(BITCOIN_NEED_QT_PLUGINS) && !defined(_BITCOIN_QT_PLUGINS_INCLUDED)
# define _BITCOIN_QT_PLUGINS_INCLUDED
# define __INSURE__
# include <QtPlugin>
Q_IMPORT_PLUGIN ( qcncodecs )
Q_IMPORT_PLUGIN ( qjpcodecs )
Q_IMPORT_PLUGIN ( qtwcodecs )
Q_IMPORT_PLUGIN ( qkrcodecs )
Q_IMPORT_PLUGIN ( qtaccessiblewidgets )
# endif
2013-04-14 18:50:40 +02:00
// Declare meta types used for QMetaObject::invokeMethod
Q_DECLARE_METATYPE ( bool * )
2011-06-05 16:24:23 +02:00
// Need a global reference for the notifications to find the GUI
2012-03-24 18:48:18 +01:00
static BitcoinGUI * guiref ;
2013-04-14 11:35:37 +02:00
static SplashScreen * splashref ;
2011-06-05 16:03:29 +02:00
2013-02-16 17:58:45 +01:00
static bool ThreadSafeMessageBox ( const std : : string & message , const std : : string & caption , unsigned int style )
2011-06-05 16:03:29 +02:00
{
if ( guiref )
{
2012-05-19 09:35:26 +02:00
bool modal = ( style & CClientUIInterface : : MODAL ) ;
2013-02-16 17:58:45 +01:00
bool ret = false ;
2012-11-05 08:04:21 +01:00
// In case of modal message, use blocking connection to wait for user to click a button
QMetaObject : : invokeMethod ( guiref , " message " ,
2012-03-25 20:47:33 +02:00
modal ? GUIUtil : : blockingGUIThreadConnection ( ) : Qt : : QueuedConnection ,
2011-06-05 16:03:29 +02:00
Q_ARG ( QString , QString : : fromStdString ( caption ) ) ,
2012-03-25 20:47:33 +02:00
Q_ARG ( QString , QString : : fromStdString ( message ) ) ,
2013-02-16 17:58:45 +01:00
Q_ARG ( unsigned int , style ) ,
Q_ARG ( bool * , & ret ) ) ;
return ret ;
2011-06-05 16:03:29 +02:00
}
else
{
2013-09-18 12:38:08 +02:00
LogPrintf ( " %s: %s \n " , caption . c_str ( ) , message . c_str ( ) ) ;
2011-06-05 16:03:29 +02:00
fprintf ( stderr , " %s: %s \n " , caption . c_str ( ) , message . c_str ( ) ) ;
2013-02-16 17:58:45 +01:00
return false ;
2011-06-05 16:03:29 +02:00
}
}
2012-05-06 19:40:58 +02:00
static void InitMessage ( const std : : string & message )
2011-08-02 21:48:59 +02:00
{
if ( splashref )
{
2013-04-14 11:35:37 +02:00
splashref - > showMessage ( QString : : fromStdString ( message ) , Qt : : AlignBottom | Qt : : AlignHCenter , QColor ( 55 , 55 , 55 ) ) ;
2013-04-02 17:30:14 +02:00
qApp - > processEvents ( ) ;
2011-08-02 21:48:59 +02:00
}
2013-09-18 12:38:08 +02:00
LogPrintf ( " init message: %s \n " , message . c_str ( ) ) ;
2011-08-02 21:48:59 +02:00
}
2011-06-13 16:56:37 +02:00
/*
Translate string to current locale using Qt .
*/
2012-05-06 19:40:58 +02:00
static std : : string Translate ( const char * psz )
2011-06-13 16:56:37 +02:00
{
return QCoreApplication : : translate ( " bitcoin-core " , psz ) . toStdString ( ) ;
}
2012-04-14 09:41:05 +02:00
/* Handle runaway exceptions. Shows a message box with the problem and quits the program.
*/
static void handleRunawayException ( std : : exception * e )
{
PrintExceptionContinue ( e , " Runaway exception " ) ;
2012-07-23 05:59:02 +02:00
QMessageBox : : critical ( 0 , " Runaway exception " , BitcoinGUI : : tr ( " A fatal error occurred. Bitcoin can no longer continue safely and will quit. " ) + QString ( " \n \n " ) + QString : : fromStdString ( strMiscWarning ) ) ;
2012-04-14 09:41:05 +02:00
exit ( 1 ) ;
}
2013-05-19 17:36:01 +02:00
/** Set up translations */
static void initTranslations ( QTranslator & qtTranslatorBase , QTranslator & qtTranslator , QTranslator & translatorBase , QTranslator & translator )
{
QSettings settings ;
// Get desired locale (e.g. "de_DE")
// 1) System default language
QString lang_territory = QLocale : : system ( ) . name ( ) ;
// 2) Language from QSettings
QString lang_territory_qsettings = settings . value ( " language " , " " ) . toString ( ) ;
if ( ! lang_territory_qsettings . isEmpty ( ) )
lang_territory = lang_territory_qsettings ;
// 3) -lang command line argument
lang_territory = QString : : fromStdString ( GetArg ( " -lang " , lang_territory . toStdString ( ) ) ) ;
// Convert to "de" only by truncating "_DE"
QString lang = lang_territory ;
lang . truncate ( lang_territory . lastIndexOf ( ' _ ' ) ) ;
// Load language files for configured locale:
// - First load the translator for the base language, without territory
// - Then load the more specific locale translator
// Load e.g. qt_de.qm
if ( qtTranslatorBase . load ( " qt_ " + lang , QLibraryInfo : : location ( QLibraryInfo : : TranslationsPath ) ) )
QApplication : : installTranslator ( & qtTranslatorBase ) ;
// Load e.g. qt_de_DE.qm
if ( qtTranslator . load ( " qt_ " + lang_territory , QLibraryInfo : : location ( QLibraryInfo : : TranslationsPath ) ) )
QApplication : : installTranslator ( & qtTranslator ) ;
// Load e.g. bitcoin_de.qm (shortcut "de" needs to be defined in bitcoin.qrc)
if ( translatorBase . load ( lang , " :/translations/ " ) )
QApplication : : installTranslator ( & translatorBase ) ;
// Load e.g. bitcoin_de_DE.qm (shortcut "de_DE" needs to be defined in bitcoin.qrc)
if ( translator . load ( lang_territory , " :/translations/ " ) )
QApplication : : installTranslator ( & translator ) ;
}
2013-07-23 09:01:08 +02:00
/* qDebug() message handler --> debug.log */
# if QT_VERSION < 0x050000
2013-12-03 09:25:24 +01:00
void DebugMessageHandler ( QtMsgType type , const char * msg )
2013-07-23 09:01:08 +02:00
{
2013-09-28 19:29:44 +02:00
Q_UNUSED ( type ) ;
2013-12-13 07:39:56 +01:00
LogPrint ( " qt " , " GUI: %s \n " , msg ) ;
2013-07-23 09:01:08 +02:00
}
# else
void DebugMessageHandler ( QtMsgType type , const QMessageLogContext & context , const QString & msg )
{
2013-09-28 19:29:44 +02:00
Q_UNUSED ( type ) ;
Q_UNUSED ( context ) ;
2013-12-13 07:39:56 +01:00
LogPrint ( " qt " , " GUI: %s \n " , qPrintable ( msg ) ) ;
2013-07-23 09:01:08 +02:00
}
# endif
2012-02-10 23:25:36 +01:00
# ifndef BITCOIN_QT_TEST
2011-05-07 22:13:39 +02:00
int main ( int argc , char * argv [ ] )
{
2013-06-23 18:04:44 +02:00
bool fSelParFromCLFailed = false ;
2014-01-05 12:37:51 +01:00
/// 1. Parse command-line options. These take precedence over anything else.
2013-01-03 22:06:18 +01:00
// Command-line options take precedence:
ParseParameters ( argc , argv ) ;
2013-06-23 18:04:44 +02:00
// Check for -testnet or -regtest parameter (TestNet() calls are only valid after this clause)
if ( ! SelectParamsFromCommandLine ( ) ) {
fSelParFromCLFailed = true ;
}
2014-01-05 12:37:51 +01:00
// Parse URIs on command line -- this can affect TestNet() / RegTest() mode
if ( ! PaymentServer : : ipcParseCommandLine ( argc , argv ) )
exit ( 0 ) ;
bool isaTestNet = TestNet ( ) | | RegTest ( ) ;
// Do not refer to data directory yet, this can be overridden by Intro::pickDataDirectory
2013-01-03 22:06:18 +01:00
2014-01-05 12:37:51 +01:00
/// 2. Basic Qt initialization (not dependent on parameters or configuration)
2013-05-31 14:02:24 +02:00
# if QT_VERSION < 0x050000
2011-11-12 12:20:34 +01:00
// Internal string conversion is all UTF-8
2011-08-28 14:12:26 +02:00
QTextCodec : : setCodecForTr ( QTextCodec : : codecForName ( " UTF-8 " ) ) ;
QTextCodec : : setCodecForCStrings ( QTextCodec : : codecForTr ( ) ) ;
2013-05-31 14:02:24 +02:00
# endif
2011-08-28 14:12:26 +02:00
2011-06-18 14:25:24 +02:00
Q_INIT_RESOURCE ( bitcoin ) ;
2011-05-07 22:13:39 +02:00
QApplication app ( argc , argv ) ;
2013-11-21 17:59:31 +01:00
# if QT_VERSION > 0x050100
// Generate high-dpi pixmaps
QApplication : : setAttribute ( Qt : : AA_UseHighDpiPixmaps ) ;
# endif
# ifdef Q_OS_MAC
QApplication : : setAttribute ( Qt : : AA_DontShowIconsInMenus ) ;
# endif
2011-06-23 22:35:30 +02:00
2013-04-14 18:50:40 +02:00
// Register meta types used for QMetaObject::invokeMethod
qRegisterMetaType < bool * > ( ) ;
2014-01-05 12:37:51 +01:00
/// 3. Application identification
// must be set before OptionsModel is initialized or translations are loaded,
// as it is used to locate QSettings
2013-05-19 17:36:01 +02:00
QApplication : : setOrganizationName ( " Bitcoin " ) ;
QApplication : : setOrganizationDomain ( " bitcoin.org " ) ;
2013-12-18 09:45:36 +01:00
if ( isaTestNet ) // Separate UI settings for testnets
2013-05-19 17:36:01 +02:00
QApplication : : setApplicationName ( " Bitcoin-Qt-testnet " ) ;
else
QApplication : : setApplicationName ( " Bitcoin-Qt " ) ;
2014-01-05 12:37:51 +01:00
/// 4. Initialization of translations, so that intro dialog is in user's language
2013-05-19 17:36:01 +02:00
// Now that QSettings are accessible, initialize translations
QTranslator qtTranslatorBase , qtTranslator , translatorBase , translator ;
initTranslations ( qtTranslatorBase , qtTranslator , translatorBase , translator ) ;
2014-01-05 12:37:51 +01:00
// Show help message immediately after parsing command-line options (for "-lang") and setting locale,
// but before showing splash screen.
if ( mapArgs . count ( " -? " ) | | mapArgs . count ( " --help " ) )
{
GUIUtil : : HelpMessageBox help ;
help . showOrPrint ( ) ;
2013-06-23 18:04:44 +02:00
return 1 ;
}
2014-01-05 12:37:51 +01:00
// Now that translations are initialized, check for earlier errors and show a translatable error message
if ( fSelParFromCLFailed ) {
2013-06-23 18:04:44 +02:00
QMessageBox : : critical ( 0 , QObject : : tr ( " Bitcoin " ) , QObject : : tr ( " Error: Invalid combination of -regtest and -testnet. " ) ) ;
return 1 ;
}
2014-01-05 12:37:51 +01:00
/// 5. Now that settings and translations are available, ask user for data directory
// User language is set up: pick a data directory
Intro : : pickDataDirectory ( isaTestNet ) ;
/// 6. Determine availability of data directory and parse bitcoin.conf
if ( ! boost : : filesystem : : is_directory ( GetDataDir ( false ) ) )
{
QMessageBox : : critical ( 0 , QObject : : tr ( " Bitcoin " ) ,
QObject : : tr ( " Error: Specified data directory \" %1 \" does not exist. " ) . arg ( QString : : fromStdString ( mapArgs [ " -datadir " ] ) ) ) ;
return 1 ;
}
ReadConfigFile ( mapArgs , mapMultiArgs ) ;
/// 7. URI IPC sending
// - Do this early as we don't want to bother initializing if we are just calling IPC
// - Do this *after* setting up the data directory, as the data directory hash is used in the name
// of the server.
// - Do this after creating app and setting up translations, so errors are
// translated properly.
if ( PaymentServer : : ipcSendCommandLine ( ) )
exit ( 0 ) ;
2013-07-18 08:50:17 +02:00
// Start up the payment server early, too, so impatient users that click on
// bitcoin: links repeatedly have their payment requests routed to this process:
PaymentServer * paymentServer = new PaymentServer ( & app ) ;
2014-01-05 12:37:51 +01:00
/// 8. Main GUI initialization
2012-04-13 17:10:50 +02:00
// Install global event filter that makes sure that long tooltips can be word-wrapped
app . installEventFilter ( new GUIUtil : : ToolTipToRichTextFilter ( TOOLTIP_WRAP_THRESHOLD , & app ) ) ;
2013-08-31 15:32:35 +02:00
// Install qDebug() message handler to route to debug.log
# if QT_VERSION < 0x050000
qInstallMsgHandler ( DebugMessageHandler ) ;
# else
qInstallMessageHandler ( DebugMessageHandler ) ;
# endif
2014-01-05 12:37:51 +01:00
// Load GUI settings from QSettings
2012-02-17 03:09:41 +01:00
OptionsModel optionsModel ;
2012-05-06 19:40:58 +02:00
// Subscribe to global signals from core
uiInterface . ThreadSafeMessageBox . connect ( ThreadSafeMessageBox ) ;
uiInterface . InitMessage . connect ( InitMessage ) ;
uiInterface . Translate . connect ( Translate ) ;
2014-01-05 12:37:51 +01:00
// Show splash screen if appropriate
2013-12-18 09:45:36 +01:00
SplashScreen splash ( QPixmap ( ) , 0 , isaTestNet ) ;
2013-04-28 17:37:50 +02:00
if ( GetBoolArg ( " -splash " , true ) & & ! GetBoolArg ( " -min " , false ) )
2012-02-25 03:54:18 +01:00
{
splash . show ( ) ;
splash . setAutoFillBackground ( true ) ;
splashref = & splash ;
}
2011-08-02 21:48:59 +02:00
app . processEvents ( ) ;
2011-06-05 14:19:57 +02:00
app . setQuitOnLastWindowClosed ( false ) ;
2011-05-07 22:13:39 +02:00
2011-06-07 18:59:01 +02:00
try
{
2013-05-03 15:18:28 +02:00
# ifndef Q_OS_MAC
2012-05-01 18:44:11 +02:00
// Regenerate startup link, to fix links to old versions
2013-05-03 15:18:28 +02:00
// OSX: makes no sense on mac and might also scan/mount external (and sleeping) volumes (can take up some secs)
2012-05-01 18:44:11 +02:00
if ( GUIUtil : : GetStartOnSystemStartup ( ) )
GUIUtil : : SetStartOnSystemStartup ( true ) ;
2013-05-03 15:18:28 +02:00
# endif
2012-05-01 18:44:11 +02:00
2013-03-07 04:16:05 +01:00
boost : : thread_group threadGroup ;
2013-03-23 23:14:12 +01:00
2013-12-18 09:45:36 +01:00
BitcoinGUI window ( isaTestNet , 0 ) ;
2012-03-25 20:47:33 +02:00
guiref = & window ;
2013-03-23 23:14:12 +01:00
QTimer * pollShutdownTimer = new QTimer ( guiref ) ;
QObject : : connect ( pollShutdownTimer , SIGNAL ( timeout ( ) ) , guiref , SLOT ( detectShutdown ( ) ) ) ;
pollShutdownTimer - > start ( 200 ) ;
2013-11-04 14:36:49 +01:00
if ( AppInit2 ( threadGroup , false ) )
2011-05-22 17:19:43 +02:00
{
2011-07-03 20:53:56 +02:00
{
2012-03-25 20:47:33 +02:00
// Put this in a block, so that the Model objects are cleaned up before
// calling Shutdown().
2012-02-17 03:09:41 +01:00
optionsModel . Upgrade ( ) ; // Must be done after AppInit2
2013-07-22 08:50:39 +02:00
PaymentServer : : LoadRootCAs ( ) ;
2013-08-24 15:07:17 +02:00
paymentServer - > setOptionsModel ( & optionsModel ) ;
2013-07-22 08:50:39 +02:00
2012-02-25 03:54:18 +01:00
if ( splashref )
splash . finish ( & window ) ;
2012-02-17 03:09:41 +01:00
2011-07-29 14:36:35 +02:00
ClientModel clientModel ( & optionsModel ) ;
2013-12-03 09:25:24 +01:00
window . setClientModel ( & clientModel ) ;
2013-11-12 14:54:43 +01:00
WalletModel * walletModel = 0 ;
if ( pwalletMain )
walletModel = new WalletModel ( pwalletMain , & optionsModel ) ;
2011-07-09 15:26:57 +02:00
2013-11-12 14:54:43 +01:00
if ( walletModel )
{
window . addWallet ( " ~Default " , walletModel ) ;
window . setCurrentWallet ( " ~Default " ) ;
}
2011-07-03 20:53:56 +02:00
2011-12-03 17:57:30 +01:00
// If -min option passed, start window minimized.
2013-04-28 17:37:50 +02:00
if ( GetBoolArg ( " -min " , false ) )
2011-12-03 17:57:30 +01:00
{
window . showMinimized ( ) ;
}
else
{
window . show ( ) ;
}
2011-07-03 20:53:56 +02:00
2013-02-12 00:52:30 +01:00
// Now that initialization/startup is done, process any command-line
2013-07-22 08:50:39 +02:00
// bitcoin: URIs or payment requests:
QObject : : connect ( paymentServer , SIGNAL ( receivedPaymentRequest ( SendCoinsRecipient ) ) ,
& window , SLOT ( handlePaymentRequest ( SendCoinsRecipient ) ) ) ;
2013-11-06 15:08:56 +01:00
QObject : : connect ( & window , SIGNAL ( receivedURI ( QString ) ) ,
paymentServer , SLOT ( handleURIOrFile ( QString ) ) ) ;
2013-11-12 14:54:43 +01:00
if ( walletModel )
{
QObject : : connect ( walletModel , SIGNAL ( coinsSent ( CWallet * , SendCoinsRecipient , QByteArray ) ) ,
paymentServer , SLOT ( fetchPaymentACK ( CWallet * , const SendCoinsRecipient & , QByteArray ) ) ) ;
}
2013-10-18 11:44:05 +02:00
QObject : : connect ( paymentServer , SIGNAL ( message ( QString , QString , unsigned int ) ) ,
guiref , SLOT ( message ( QString , QString , unsigned int ) ) ) ;
2013-02-12 00:52:30 +01:00
QTimer : : singleShot ( 100 , paymentServer , SLOT ( uiReady ( ) ) ) ;
2012-03-26 18:18:24 +02:00
2011-07-03 20:53:56 +02:00
app . exec ( ) ;
2012-05-06 08:14:19 +02:00
window . hide ( ) ;
2012-03-25 20:47:33 +02:00
window . setClientModel ( 0 ) ;
2013-03-22 18:32:49 +01:00
window . removeAllWallets ( ) ;
2011-07-03 20:53:56 +02:00
guiref = 0 ;
2013-11-12 14:54:43 +01:00
delete walletModel ;
2011-07-03 20:53:56 +02:00
}
2013-12-13 07:39:56 +01:00
// Shutdown the core and its threads, but don't exit the GUI here
2013-03-07 04:16:05 +01:00
threadGroup . interrupt_all ( ) ;
threadGroup . join_all ( ) ;
2013-03-09 18:02:57 +01:00
Shutdown ( ) ;
2011-05-22 17:19:43 +02:00
}
else
{
2013-05-02 18:26:33 +02:00
threadGroup . interrupt_all ( ) ;
threadGroup . join_all ( ) ;
Shutdown ( ) ;
2011-05-22 17:19:43 +02:00
return 1 ;
}
} catch ( std : : exception & e ) {
2012-04-14 09:41:05 +02:00
handleRunawayException ( & e ) ;
2011-05-22 17:19:43 +02:00
} catch ( . . . ) {
2012-04-14 09:41:05 +02:00
handleRunawayException ( NULL ) ;
2011-05-22 17:19:43 +02:00
}
2011-07-03 20:53:56 +02:00
return 0 ;
2011-05-07 22:13:39 +02:00
}
2012-02-10 23:25:36 +01:00
# endif // BITCOIN_QT_TEST