summaryrefslogblamecommitdiff
path: root/src/main.cpp
blob: 2d6b08a8fd51b013fc473879a89df9b90f89870b (plain) (tree)
1
  

















                                                                             
   

                 
 
                   

                   
                   
                 

                      
                                     
                                    


                          


                      
             
                 

                     


                                    

                          
                           
                 
                     
                         

                        
                
                      


                           

                  
                            
                           

                      
                          
                            
                           
                         

                             
                          
                             
 
                                  
                           
                             
                              
                        

                                            
                                      



                                      
 
                            

                                
                            
                                      
 
                       
                          
                           
 
                                              
 
                   
 
                    
                         
 
            
           
 
                                                                   
                                                                     
                                                
                        
 


                                          














                                                                        
   









                                                                            
                            
                          



                     
                      
                    
                         
                         
                          
                           
                           
                         




                           

                        
   







                                                                  
                                                                             




































                                                                              

                                                                          
   
                  
 


                                                


                                                   

                                                


                                                       

                                                                               
     
                                                 
                                                                               
                              
      
     
                            
                                                                        
                               

                
 
 


                            
                                              
 
                                       
                                                 




                                                                  
                                  
                                                              
                                 
     
                                 
      





                                       
                                   


                                                                          
                                       
                                          
 

                                                                            
                      
                                                

                           
                                             
 
                                             

                                              
                          
                                                            
                                                  
     

                                                          
                                                    
            


                                


   
                               
   
                                       

                     
                                       








                                                                               
 
                                                                                    
 








                                                                    


                                                      



                                                                 
                                          












                                                                            
                                                      
      
 
















                                                                                                              
                 

                                                         

                                                     
 
                                  
                                                                 




                                  

                                                                            
                

                                                                
 
                                        

                                                                         


                                                                            
                

     

                             
 


                                      
                            
                                                           
 
                              
         

                                               
         


                                                                 
                                         

                             
                            
                           
                                        
     


                          


                       

                  


                                                                        
                   
 

                    
 

                       
 

                     
 
                           

                        
                    
 
                                      

 

                



                                                 
                                                   
                                                       
                                                                

                                                        

                                                       
                                                         

                                                         

 









                                                                           

                                                           
                                                   

                                          
                                                    
                                                    
                                                    
                                                    

                                                    

                                                    
                                                    
                                                    
                                                    
             




                                                                            
                         
                  





                                         


                                            


                                          


                                          
                     
                                          



                                          





                                                         

                                           
                      


                                            


                                            



         
   

                                                                           


                  
                                                                         




                                                                       


                                         

                                                                


     
 
           



                                                 
     


                                    
 


                                                   
     


                                  
 


                                                 
     




                            
 
                                                
                                       
 
                                                               
 
                                         
                                        
 
                                             
                                             

                       



                                                      
 


                                                                    

                                                                              

                            



                                                         

 
                                          
 
                                                               
 
                                        
 
                                             
                                             
 




                                                                

 




                                                                
                                                                 


 



                                                

                                                                    


                                                                        



                                             
                                                                  

 





















































                                                                  
                                                































                                                   





                                                                             








                                                                    






                                                

                                                     
           

                                
       
     












                                          
 
              
            
                                                                           
      



                                                

      

                               
 




                                           




                                                                            
 
                              
                      
                                                           
     
                                                            

      
                                   
 
                            
 



                                      
 
                                                                          
                      
                                                                   
                                      
      
 
                                                             
 


                                                                       
                                                                                       





                                                                          
                                                         


                                                
 
                                              






                                                                     
         
 

                                                            
 



                                                       
 
                                                                         
 



                                         
                                         
             
                                   
                 





                                                                
                         







                                                                               
                         



                                           
             
 

                         
 

                                 

                                                                             
                                                                  
                                                

                                     
                                                 
                 
             
 









                                                                                

                                     
 








                                                           
             





                                                          
 



                                                            
             

                                        
             
 



                                                              
                                  

                                                                       


                                                                               



















                                                                               

                                                                          




                                                                                    


                                                                          








                                                                              

                                                                    


                                      






                                                                            
                                                     

                                                                     



                                                    








                                                                               











                                                                             

                                          
                                      






                                                 






                                                                             












                                                                           

                                                           













                                                                       
                                                                        

                                                                 
                                                                  

                                                                        


                                                                              
                         
 

                              







                                                                            
                                                                       





                                                                           








                                                                              

                                                                           

                                                                   

                              


                                                                 




                                                              
                                                                             
                                                                                  
                                          




                                                                                 







                                                                             











                                                                            
                                                                                                 









                                                                                 
 
                      

                                            
      






                                                   
                                                                      












                                                                
 
                                                


                                                           


                                                         













                                                   
             
         



                            



                                 
     
 





                                              
 



                                   
 
                            
                  
                    
                  
 
             
 
/*
 *  The Mana World
 *  Copyright 2004 The Mana World Development Team
 *
 *  This file is part of The Mana World.
 *
 *  The Mana World is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  any later version.
 *
 *  The Mana World is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with The Mana World; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

#include "main.h"

#include <getopt.h>
#include <iostream>
#include <physfs.h>
#include <unistd.h>
#include <vector>
#include <SDL_image.h>

#include <guichan/actionlistener.hpp>
#include <guichan/widgets/label.hpp>

#include <libxml/parser.h>

#ifdef WIN32
#include <SDL_syswm.h>
#endif
#ifndef WIN32
#include <cerrno>
#include <sys/stat.h>
#endif
#if defined __APPLE__
#include <CoreFoundation/CFBundle.h>
#endif

#include "configuration.h"
#include "keyboardconfig.h"
#include "game.h"
#include "graphics.h"
#include "itemshortcut.h"
#include "lockedarray.h"
#include "localplayer.h"
#include "log.h"
#include "logindata.h"
#ifdef USE_OPENGL
#include "openglgraphics.h"
#endif
#include "sound.h"

#include "gui/char_select.h"
#include "gui/connection.h"
#include "gui/gui.h"
#include "gui/login.h"
#include "gui/ok_dialog.h"
#include "gui/progressbar.h"
#include "gui/quitdialog.h"
#include "gui/register.h"
#include "gui/sdlinput.h"
#include "gui/serverdialog.h"
#include "gui/textfield.h"
#include "gui/updatewindow.h"

#include "net/charserverhandler.h"
#include "net/connection.h"
#include "net/loginhandler.h"
#include "net/logouthandler.h"
#include "net/network.h"

#include "net/accountserver/accountserver.h"
#include "net/accountserver/account.h"

#include "net/chatserver/chatserver.h"

#include "net/gameserver/gameserver.h"

#include "resources/image.h"
#include "resources/itemdb.h"
#include "resources/monsterdb.h"
#include "resources/npcdb.h"
#include "resources/resourcemanager.h"

#include "utils/dtor.h"
#include "utils/gettext.h"
#include "utils/tostring.h"

std::string token; //used to store magic_token

Graphics *graphics;

unsigned char state;
std::string errorMessage;

Sound sound;
Music *bgm;

Configuration config;         /**< XML file configuration reader */
Configuration branding;       /**< XML branding information reader */
Logger *logger;               /**< Log object */
KeyboardConfig keyboard;

Net::Connection *gameServerConnection = 0;
Net::Connection *chatServerConnection = 0;

CharServerHandler charServerHandler;
LoginData loginData;
LoginHandler loginHandler;
LogoutHandler logoutHandler;
LockedArray<LocalPlayer*> charInfo(maxSlot + 1);

// This anonymous namespace hides whatever is inside from other modules.
namespace {

Net::Connection *accountServerConnection = 0;

std::string homeDir;
std::string updateHost;
std::string updatesDir;

/**
 * A structure holding the values of various options that can be passed from
 * the command line.
 */
struct Options
{
    /**
     * Constructor.
     */
    Options():
        printHelp(false),
        printVersion(false),
        skipUpdate(false),
        serverPort(0)
    {};

    bool printHelp;
    bool printVersion;
    bool skipUpdate;
    std::string username;
    std::string password;
    std::string character;
    std::string configPath;
    std::string updateHost;
    std::string dataPath;

    std::string serverName;
    short serverPort;
};

} // anonymous namespace

/**
 * Parse the update host and determine the updates directory
 * Then verify that the directory exists (creating if needed).
 */
void setUpdatesDir()
{
    // If updatesHost is currently empty, fill it from config file
    if (updateHost.empty()) {
        updateHost =
            config.getValue("updatehost", "http://updates.themanaworld.org");
    }

    // Remove any trailing slash at the end of the update host
    if (updateHost.at(updateHost.size() - 1) == '/')
        updateHost.resize(updateHost.size() - 1);

    // Parse out any "http://" or "ftp://", and set the updates directory
    size_t pos;
    pos = updateHost.find("://");
    if (pos != updateHost.npos) {
        if (pos + 3 < updateHost.length()) {
            updatesDir =
                "updates/" + updateHost.substr(pos + 3);
        } else {
            logger->log("Error: Invalid update host: %s", updateHost.c_str());
            errorMessage = "Invalid update host: " + updateHost;
            state = STATE_ERROR;
        }
    } else {
        logger->log("Warning: no protocol was specified for the update host");
        updatesDir = "updates/" + updateHost;
    }

    ResourceManager *resman = ResourceManager::getInstance();

    // Verify that the updates directory exists. Create if necessary.
    if (!resman->isDirectory("/" + updatesDir)) {
        if (!resman->mkdir("/" + updatesDir)) {
            logger->log("Error: %s/%s can't be made, but doesn't exist!",
                         homeDir.c_str(), updatesDir.c_str());
            errorMessage = "Error creating updates directory!";
            state = STATE_ERROR;
        }
    }
}

/**
 * Initializes the home directory. On UNIX and FreeBSD, ~/.tmw is used. On
 * Windows and other systems we use the current working directory.
 */
void initHomeDir()
{
    homeDir = std::string(PHYSFS_getUserDir()) +
        "/." +
        branding.getValue("appShort", "tmw");
#if defined WIN32
    if (!CreateDirectory(homeDir.c_str(), 0) &&
            GetLastError() != ERROR_ALREADY_EXISTS)
#elif defined __APPLE__
    // Use Application Directory instead of .tmw
    homeDir = std::string(PHYSFS_getUserDir()) +
        "/Library/Application Support/" +
        branding.getValue("appName", "The Mana World");
    if ((mkdir(homeDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) != 0) &&
            (errno != EEXIST))
#else
    // Checking if /home/user/.tmw folder exists.
    if ((mkdir(homeDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) != 0) &&
            (errno != EEXIST))
#endif
    {
        std::cout << homeDir
                  << " can't be created, but it doesn't exist! Exiting."
                  << std::endl;
        exit(1);
    }
}

/**
 * Initialize configuration.
 */
void initConfiguration(const Options &options)
{
    // Fill configuration with defaults
    logger->log("Initializing configuration...");
    std::string defaultHost = branding.getValue("defaultServer",
        "server.themanaworld.org");
    config.setValue("host", defaultHost);
    int defaultPort = (int)branding.getValue("defaultPort", 9601);
    config.setValue("port", defaultPort);
    config.setValue("hwaccel", 0);
#if (defined __APPLE__ || defined WIN32) && defined USE_OPENGL
    config.setValue("opengl", 1);
#else
    config.setValue("opengl", 0);
#endif
    config.setValue("screen", 0);
    config.setValue("sound", 1);
    config.setValue("guialpha", 0.8f);
    config.setValue("remember", 1);
    config.setValue("sfxVolume", 100);
    config.setValue("musicVolume", 60);
    config.setValue("fpslimit", 0);
    std::string defaultUpdateHost = branding.getValue("defaultUpdateHost",
        "http://updates.themanaworld.org");
    config.setValue("updatehost", defaultUpdateHost);
    config.setValue("customcursor", 1);
    config.setValue("ChatLogLength", 128);

    // Checking if the configuration file exists... otherwise create it with
    // default options.
    FILE *tmwFile = 0;
    std::string configPath = options.configPath;

    if (configPath.empty())
        configPath = homeDir + "/config.xml";

    tmwFile = fopen(configPath.c_str(), "r");

    // If we can't read it, it doesn't exist !
    if (tmwFile == NULL) {
        // We reopen the file in write mode and we create it
        tmwFile = fopen(configPath.c_str(), "wt");
    }
    if (tmwFile == NULL) {
        std::cout << "Can't create " << configPath << ". "
                  << "Using Defaults." << std::endl;
    } else {
        fclose(tmwFile);
        config.init(configPath);
    }
}

/**
 * Do all initialization stuff.
 */
void initEngine(const Options &options)
{
    // Initialize SDL
    logger->log("Initializing SDL...");
    if (SDL_Init(SDL_INIT_VIDEO | SDL_INIT_TIMER) < 0) {
        std::cerr << "Could not initialize SDL: " <<
            SDL_GetError() << std::endl;
        exit(1);
    }
    atexit(SDL_Quit);

    SDL_EnableUNICODE(1);
    SDL_EnableKeyRepeat(SDL_DEFAULT_REPEAT_DELAY, SDL_DEFAULT_REPEAT_INTERVAL);

    SDL_WM_SetCaption(branding.getValue("appName", "The Mana World").c_str(), NULL);

    ResourceManager *resman = ResourceManager::getInstance();

    if (!resman->setWriteDir(homeDir)) {
        std::cout << homeDir
                  << " couldn't be set as home directory! Exitting."
                  << std::endl;
        exit(1);
    }

    // Add the user's homedir to PhysicsFS search path
    resman->addToSearchPath(homeDir, false);

    // Add the main data directories to our PhysicsFS search path
    if (!options.dataPath.empty()) {
        resman->addToSearchPath(options.dataPath, true);
    }
    resman->addToSearchPath("data", true);
#if defined __APPLE__
    CFBundleRef mainBundle = CFBundleGetMainBundle();
    CFURLRef resourcesURL = CFBundleCopyResourcesDirectoryURL(mainBundle);
    char path[PATH_MAX];
    if (!CFURLGetFileSystemRepresentation(resourcesURL, TRUE, (UInt8 *)path,
                                          PATH_MAX))
    {
        fprintf(stderr, "Can't find Resources directory\n");
    }
    CFRelease(resourcesURL);
    strncat(path, "/data", PATH_MAX - 1);
    resman->addToSearchPath(path, true);
#else
    resman->addToSearchPath(TMW_DATADIR "data", true);
#endif

#ifdef WIN32
    static SDL_SysWMinfo pInfo;
    SDL_GetWMInfo(&pInfo);
    HICON icon = LoadIcon(GetModuleHandle(NULL), "A");
    if (icon)
    {
        SetClassLong(pInfo.window, GCL_HICON, (LONG) icon);
    }
#else
    SDL_Surface *icon = IMG_Load(resman->getPath(branding.getValue("appIcon", "data/icons/tmw.png")).c_str());
    if (icon)
    {
        SDL_SetAlpha(icon, SDL_SRCALPHA, SDL_ALPHA_OPAQUE);
        SDL_WM_SetIcon(icon, NULL);
    }
#endif

#ifdef USE_OPENGL
    bool useOpenGL = (config.getValue("opengl", 0) == 1);

    // Setup image loading for the right image format
    Image::setLoadAsOpenGL(useOpenGL);

    // Create the graphics context
    graphics = useOpenGL ? new OpenGLGraphics() : new Graphics();
#else
    // Create the graphics context
    graphics = new Graphics();
#endif

    int width = (int) config.getValue("screenwidth", defaultScreenWidth);
    int height = (int) config.getValue("screenheight", defaultScreenHeight);
    int bpp = 0;
    bool fullscreen = ((int) config.getValue("screen", 0) == 1);
    bool hwaccel = ((int) config.getValue("hwaccel", 0) == 1);

    // Try to set the desired video mode
    if (!graphics->setVideoMode(width, height, bpp, fullscreen, hwaccel))
    {
        std::cerr << "Couldn't set "
                  << width << "x" << height << "x" << bpp << " video mode: "
                  << SDL_GetError() << std::endl;
        exit(1);
    }

    // Initialize for drawing
    graphics->_beginDraw();

    // Initialize the item shortcuts.
    itemShortcut = new ItemShortcut();

    gui = new Gui(graphics);
    state = STATE_CHOOSE_SERVER; /**< Initial game state */

    // Initialize sound engine
    try {
        if (config.getValue("sound", 0) == 1) {
            sound.init();
        }
        sound.setSfxVolume((int) config.getValue("sfxVolume",
                    defaultSfxVolume));
        sound.setMusicVolume((int) config.getValue("musicVolume",
                    defaultMusicVolume));
    }
    catch (const char *err) {
        state = STATE_ERROR;
        errorMessage = err;
        logger->log("Warning: %s", err);
    }

    // Initialize keyboard
    keyboard.init();
}

/** Clear the engine */
void exit_engine()
{
    // Before config.write() since it writes the shortcuts to the config
    delete itemShortcut;

    config.write();

    delete gui;
    delete graphics;

    // Shutdown libxml
    xmlCleanupParser();

    // Shutdown sound
    sound.close();

    // Unload XML databases
    ItemDB::unload();
    MonsterDB::unload();
    NPCDB::unload();

    ResourceManager::deleteInstance();
}

void printHelp()
{
    std::cout <<
        "tmw\n\n"
        "Options:\n"
        "  -h --help       : Display this help\n"
        "  -v --version    : Display the version\n"
        "  -u --skipupdate : Skip the update process\n"
        "  -d --data       : Directory to load game data from\n"
        "  -U --username   : Login with this username\n"
        "  -P --password   : Login with this password\n"
        "  -s --server     : Login Server name or IP\n"
        "  -o --port       : Login Server Port\n"
        "  -c --character  : Login with this character\n"
        "  -C --configfile : Configuration file to use\n"
        "  -H --updatehost : Use this update host\n";
}

void printVersion()
{
#ifdef PACKAGE_VERSION
    std::cout << "The Mana World version " << PACKAGE_VERSION << std::endl;
#else
    std::cout << "The Mana World version " <<
             "(local build?, PACKAGE_VERSION is not defined)" << std::endl;
#endif
}

void parseOptions(int argc, char *argv[], Options &options)
{
    const char *optstring = "hvud:U:P:Dc:s:o:C:H:";

    const struct option long_options[] = {
        { "help",       no_argument,       0, 'h' },
        { "version",    no_argument,       0, 'v' },
        { "skipupdate", no_argument,       0, 'u' },
        { "data",       required_argument, 0, 'd' },
        { "username",   required_argument, 0, 'U' },
        { "password",   required_argument, 0, 'P' },
        { "server",     required_argument, 0, 's' },
        { "port",       required_argument, 0, 'o' },
        { "character",  required_argument, 0, 'c' },
        { "configfile", required_argument, 0, 'C' },
        { "updatehost", required_argument, 0, 'H' },
        { 0 }
    };

    while (optind < argc) {
        int result = getopt_long(argc, argv, optstring, long_options, NULL);

        if (result == -1)
            break;

        switch (result) {
            default: // Unknown option
            case 'h':
                options.printHelp = true;
                break;
            case 'v':
                options.printVersion = true;
                break;
            case 'u':
                options.skipUpdate = true;
                break;
            case 'd':
                options.dataPath = optarg;
                break;
            case 'U':
                options.username = optarg;
                break;
            case 'P':
                options.password = optarg;
                break;
            case 's':
                options.serverName = optarg;
                break;
            case 'o':
                options.serverPort = (short)atoi(optarg);
                break;
            case 'c':
                options.character = optarg;
                break;
            case 'C':
                options.configPath = optarg;
                break;
            case 'H':
                options.updateHost = optarg;
                break;
        }
    }
}

/**
 * Reads the file "{Updates Directory}/resources2.txt" and attempts to load
 * each update mentioned in it.
 */
void loadUpdates()
{
    const std::string updatesFile = "/" + updatesDir + "/resources2.txt";
    ResourceManager *resman = ResourceManager::getInstance();
    std::vector<std::string> lines = resman->loadTextFile(updatesFile);

    for (unsigned int i = 0; i < lines.size(); ++i)
    {
        std::stringstream line(lines[i]);
        std::string filename;
        line >> filename;
        resman->addToSearchPath(homeDir + "/" + updatesDir + "/"
                                + filename, false);
    }
}


namespace {

struct ErrorListener : public gcn::ActionListener
{
    void action(const gcn::ActionEvent &event)
    {
        state = STATE_CHOOSE_SERVER;
    }
} errorListener;

struct AccountListener : public gcn::ActionListener
{
    void action(const gcn::ActionEvent &event)
    {
        state = STATE_CHAR_SELECT;
    }
} accountListener;

struct LoginListener : public gcn::ActionListener
{
    void action(const gcn::ActionEvent &event)
    {
        state = STATE_LOGIN;
    }
} loginListener;

} // anonymous namespace

// TODO Find some nice place for these functions
void accountLogin(LoginData *loginData)
{
    logger->log("Username is %s", loginData->username.c_str());

    loginHandler.setLoginData(loginData);
    Net::registerHandler(&loginHandler);

    charServerHandler.setCharInfo(&charInfo);
    Net::registerHandler(&charServerHandler);

    // Send login infos
    Net::AccountServer::login(accountServerConnection,
            0,  // client version
            loginData->username,
            loginData->password);

    // Clear the password, avoids auto login when returning to login
    loginData->password = "";

    // TODO This is not the best place to save the config, but at least better
    // than the login gui window
    if (loginData->remember)
    {
        config.setValue("host", loginData->hostname);
        config.setValue("username", loginData->username);
    }
    config.setValue("remember", loginData->remember);
}

void accountRegister(LoginData *loginData)
{
    logger->log("Username is %s", loginData->username.c_str());

    Net::registerHandler(&loginHandler);

    charServerHandler.setCharInfo(&charInfo);
    Net::registerHandler(&charServerHandler);

    Net::AccountServer::registerAccount(accountServerConnection,
            0, // client version
            loginData->username,
            loginData->password,
            loginData->email);
}

void accountUnRegister(LoginData *loginData)
{
    Net::registerHandler(&logoutHandler);

    Net::AccountServer::Account::unregister(loginData->username,
                                            loginData->password);

}

void accountChangePassword(LoginData *loginData)
{
    Net::registerHandler(&loginHandler);

    Net::AccountServer::Account::changePassword(loginData->username,
                                                loginData->password,
                                                loginData->newPassword);
}

void accountChangeEmail(LoginData *loginData)
{
    Net::registerHandler(&loginHandler);

    Net::AccountServer::Account::changeEmail(loginData->newEmail);
}

void switchCharacter(std::string* passToken)
{
    Net::registerHandler(&logoutHandler);

    logoutHandler.reset();
    logoutHandler.setScenario(LOGOUT_SWITCH_CHARACTER, passToken);

    Net::GameServer::logout(true);
    Net::ChatServer::logout();
}

void switchAccountServer()
{
    Net::registerHandler(&logoutHandler);

    logoutHandler.reset();
    logoutHandler.setScenario(LOGOUT_SWITCH_ACCOUNTSERVER);

    //Can't logout if we were not logged in ...
    if (accountServerConnection->isConnected())
    {
        Net::AccountServer::logout();
    }
    else
    {
        logoutHandler.setAccountLoggedOut();
    }

    if (gameServerConnection->isConnected())
    {
        Net::GameServer::logout(false);
    }
    else
    {
        logoutHandler.setGameLoggedOut();
    }

    if (chatServerConnection->isConnected())
    {
        Net::ChatServer::logout();
    }
    else
    {
        logoutHandler.setChatLoggedOut();
    }
}

void logoutThenExit()
{
    Net::registerHandler(&logoutHandler);

    logoutHandler.reset();
    logoutHandler.setScenario(LOGOUT_EXIT);

    // Can't logout if we were not logged in ...
    if (accountServerConnection->isConnected())
    {
        Net::AccountServer::logout();
    }
    else
    {
        logoutHandler.setAccountLoggedOut();
    }

    if (gameServerConnection->isConnected())
    {
        Net::GameServer::logout(false);
    }
    else
    {
        logoutHandler.setGameLoggedOut();
    }

    if (chatServerConnection->isConnected())
    {
        Net::ChatServer::logout();
    }
    else
    {
        logoutHandler.setChatLoggedOut();
    }
}

void reconnectAccount(const std::string& passToken)
{
    Net::registerHandler(&loginHandler);

    charServerHandler.setCharInfo(&charInfo);
    Net::registerHandler(&charServerHandler);

    Net::AccountServer::reconnectAccount(accountServerConnection, passToken);
}

void xmlNullLogger(void *ctx, const char *msg, ...)
{
    // Does nothing, that's the whole point of it
}

// Initialize libxml2 and check for potential ABI mismatches between
// compiled version and the shared library actually used.
void initXML()
{
    xmlInitParser();
    LIBXML_TEST_VERSION;

    // Suppress libxml2 error messages
    xmlSetGenericErrorFunc(NULL, xmlNullLogger);
}

extern "C" char const *_nl_locale_name_default(void);

/** Main */
int main(int argc, char *argv[])
{
    try
    {
        // Parse command line options
        Options options;
        parseOptions(argc, argv, options);
        if (options.printHelp)
        {
            printHelp();
            return 0;
        }
        else if (options.printVersion)
        {
            printVersion();
            return 0;
        }

#if ENABLE_NLS
#ifdef WIN32
        putenv(("LANG=" + std::string(_nl_locale_name_default())).c_str());
#endif
        setlocale(LC_MESSAGES, "");
        bindtextdomain("tmw", LOCALEDIR);
        bind_textdomain_codeset("tmw", "UTF-8");
        textdomain("tmw");
#endif

        // Initialize PhysicsFS
        PHYSFS_init(argv[0]);

        initXML();

        // load branding information
        branding.init("data/branding.xml");

        initHomeDir();
        // Configure logger
        logger = new Logger();
        logger->setLogFile(homeDir + std::string("/tmw.log"));
        logger->setLogToStandardOut(config.getValue("logToStandardOut", 0));

        // Log the tmw version
#ifdef PACKAGE_VERSION
        logger->log("The Mana World v%s", PACKAGE_VERSION);
#else
        logger->log("The Mana World - version not defined");
#endif

        initConfiguration(options);

        initEngine(options);

        Game *game = NULL;
        Window *currentDialog = NULL;
        QuitDialog* quitDialog = NULL;
        Image *login_wallpaper = NULL;

        gcn::Container *top = static_cast<gcn::Container*>(gui->getTop());
#ifdef PACKAGE_VERSION
        gcn::Label *versionLabel = new gcn::Label(PACKAGE_VERSION);
        top->add(versionLabel, 25, 2);
#endif

        sound.playMusic(branding.getValue("loginMusic", ""));

        // Server choice
        if (options.serverName.empty()) {
            loginData.hostname = config.getValue("MostUsedServerName0",
                branding.getValue("defaultServer", "server.themanaworld.org").c_str());
        }
        else {
            loginData.hostname = options.serverName;
        }
        if (options.serverPort == 0) {
            loginData.port = (short)config.getValue("MostUsedServerPort0",
                 branding.getValue("defaultPort", 9601));
        } else {
            loginData.port = options.serverPort;
        }

        loginData.username = options.username;
        if (loginData.username.empty()) {
            if (config.getValue("remember", 0)) {
                loginData.username = config.getValue("username", "");
            }
        }
        if (!options.password.empty()) {
            loginData.password = options.password;
        }

        loginData.remember = config.getValue("remember", 0);
        loginData.registerLogin = false;

        Net::initialize();
        accountServerConnection = Net::getConnection();
        gameServerConnection = Net::getConnection();
        chatServerConnection = Net::getConnection();

        unsigned int oldstate = !state; // We start with a status change.

        SDL_Event event;
        while (state != STATE_FORCE_QUIT)
        {
            // Handle SDL events
            while (SDL_PollEvent(&event))
            {
                switch (event.type)
                {
                    case SDL_QUIT:
                        state = STATE_FORCE_QUIT;
                        break;

                    case SDL_KEYDOWN:
                        if (event.key.keysym.sym == SDLK_ESCAPE)
                        {
                            if (!quitDialog)
                            {
                                quitDialog = new QuitDialog(NULL, &quitDialog);
                            }
                            else
                            {
                                quitDialog->requestMoveToTop();
                            }
                        }
                        break;
                }

                guiInput->pushInput(event);
            }

            Net::flush();
            gui->logic();

            if (!login_wallpaper)
            {
                std::string wallpaperFile = branding.getValue(
                    "loginWallpaper", "graphics/images/login_wallpaper.png");
                login_wallpaper = ResourceManager::getInstance()->
                        getImage(wallpaperFile);
                if (!login_wallpaper)
                {
                    logger->error(wallpaperFile);
                }
            }

            if (graphics->getWidth() > login_wallpaper->getWidth() ||
                    graphics->getHeight() > login_wallpaper->getHeight())
            {
                graphics->setColor(gcn::Color(64, 64, 64));
                graphics->fillRectangle(gcn::Rectangle(
                            0, 0, graphics->getWidth(), graphics->getHeight()));
            }
            graphics->drawImage(login_wallpaper,
                    (graphics->getWidth() - login_wallpaper->getWidth()) / 2,
                    (graphics->getHeight() - login_wallpaper->getHeight()) / 2);
            gui->draw();
            graphics->updateScreen();

            // TODO: Add connect timeouts
            if (state == STATE_CONNECT_ACCOUNT &&
                    accountServerConnection->isConnected())
            {
                if (options.skipUpdate) {
                    state = STATE_LOADDATA;
                } else {
                    state = STATE_UPDATE;
                }
            }
            else if (state == STATE_CONNECT_GAME &&
                    gameServerConnection->isConnected() &&
                    chatServerConnection->isConnected())
            {
                accountServerConnection->disconnect();
                Net::clearHandlers();

                state = STATE_GAME;
            }
            else if (state == STATE_RECONNECT_ACCOUNT &&
                     accountServerConnection->isConnected())
            {
                reconnectAccount(token);
                state = STATE_WAIT;
            }

            if (state != oldstate) {
                // Load updates after exiting the update state
                if (oldstate == STATE_UPDATE)
                {
                    loadUpdates();
                    // Reload the wallpaper in case that it was updated
                    login_wallpaper->decRef();
                    login_wallpaper = ResourceManager::getInstance()->getImage(
                        branding.getValue("loginWallpaper",
                            "graphics/images/login_wallpaper.png"));
                }

                oldstate = state;

                // Get rid of the dialog of the previous state
                if (currentDialog) {
                    delete currentDialog;
                    currentDialog = NULL;
                }
                // State has changed, while the quitDialog was active, it might
                // not be correct anymore
                if (quitDialog) {
                    quitDialog->scheduleDelete();
                }

                switch (state) {
                    case STATE_CHOOSE_SERVER:
                        logger->log("State: CHOOSE_SERVER");

                        // Allow changing this using a server choice dialog
                        // We show the dialog box only if the command-line
                        // options weren't set.
                        if (options.serverName.empty() && options.serverPort == 0) {
                            currentDialog = new ServerDialog(&loginData);
                        } else {
                            state = STATE_CONNECT_ACCOUNT;

                            // Reset options so that cancelling or connect
                            // timeout will show the server dialog.
                            options.serverName.clear();
                            options.serverPort = 0;
                        }
                        break;

                    case STATE_CONNECT_ACCOUNT:
                        logger->log("State: CONNECT_ACCOUNT");
                        logger->log("Trying to connect to account server...");
                        accountServerConnection->connect(loginData.hostname,
                                loginData.port);
                        currentDialog = new ConnectionDialog(
                                STATE_SWITCH_ACCOUNTSERVER_ATTEMPT);
                        break;

                    case STATE_UPDATE:
                        // Determine which source to use for the update host
                        if (!options.updateHost.empty())
                            updateHost = options.updateHost;
                        else
                            updateHost = loginData.updateHost;

                        setUpdatesDir();
                        logger->log("State: UPDATE");
                        currentDialog = new UpdaterWindow(updateHost,
                                homeDir + "/" + updatesDir);
                        break;

                    case STATE_LOGIN:
                        logger->log("State: LOGIN");
                        if (options.username.empty()
                                || options.password.empty()) {
                            currentDialog = new LoginDialog(&loginData);
                        } else {
                            state = STATE_LOGIN_ATTEMPT;
                            // Clear the password so that when login fails, the
                            // dialog will show up next time.
                            options.password.clear();
                        }
                        break;

                    case STATE_LOADDATA:
                        logger->log("State: LOADDATA");

                        // Add customdata directory
                        ResourceManager::getInstance()->searchAndAddArchives(
                            "customdata/",
                            "zip",
                            false);

                        // Load XML databases
                        ItemDB::load();
                        MonsterDB::load();
                        NPCDB::load();
                        state = STATE_LOGIN;
                        break;

                    case STATE_LOGIN_ATTEMPT:
                        accountLogin(&loginData);
                        break;

                    case STATE_LOGIN_ERROR:
                        logger->log("State: LOGIN ERROR");
                        currentDialog = new OkDialog("Error ", errorMessage);
                        currentDialog->addActionListener(&loginListener);
                        currentDialog = NULL; // OkDialog deletes itself
                        break;

                    case STATE_SWITCH_ACCOUNTSERVER:
                        logger->log("State: SWITCH_ACCOUNTSERVER");

                        gameServerConnection->disconnect();
                        chatServerConnection->disconnect();
                        accountServerConnection->disconnect();

                        state = STATE_CHOOSE_SERVER;
                        break;

                    case STATE_SWITCH_ACCOUNTSERVER_ATTEMPT:
                        logger->log("State: SWITCH_ACCOUNTSERVER_ATTEMPT");
                        switchAccountServer();

                        state = STATE_SWITCH_ACCOUNTSERVER;
                        break;

                    case STATE_REGISTER:
                        logger->log("State: REGISTER");
                        currentDialog = new RegisterDialog(&loginData);
                        break;

                    case STATE_REGISTER_ATTEMPT:
                        accountRegister(&loginData);
                        break;

                    case STATE_CHAR_SELECT:
                        logger->log("State: CHAR_SELECT");
                        currentDialog =
                            new CharSelectDialog(&charInfo, &loginData);

                        if (((CharSelectDialog*) currentDialog)->
                                selectByName(options.character)) {
                            ((CharSelectDialog*) currentDialog)->action(
                                gcn::ActionEvent(NULL, "ok"));
                        } else {
                            ((CharSelectDialog*) currentDialog)->selectByName(
                                config.getValue("lastCharacter", ""));
                        }

                        break;

                    case STATE_CHANGEEMAIL_ATTEMPT:
                        logger->log("State: CHANGE EMAIL ATTEMPT");
                        accountChangeEmail(&loginData);
                        break;

                    case STATE_CHANGEEMAIL:
                        logger->log("State: CHANGE EMAIL");
                        currentDialog = new OkDialog("Email Address change",
                                "Email Address changed successfully!");
                        currentDialog->addActionListener(&accountListener);
                        currentDialog = NULL; // OkDialog deletes itself
                        loginData.email = loginData.newEmail;
                        loginData.newEmail = "";
                        break;

                    case STATE_CHANGEPASSWORD_ATTEMPT:
                        logger->log("State: CHANGE PASSWORD ATTEMPT");
                        accountChangePassword(&loginData);
                        break;

                    case STATE_CHANGEPASSWORD:
                        logger->log("State: CHANGE PASSWORD");
                        currentDialog = new OkDialog("Password change",
                                            "Password changed successfully!");
                        currentDialog->addActionListener(&accountListener);
                        currentDialog = NULL; // OkDialog deletes itself
                        loginData.password = loginData.newPassword;
                        loginData.newPassword = "";
                        break;

                    case STATE_UNREGISTER_ATTEMPT:
                        logger->log("State: UNREGISTER ATTEMPT");
                        accountUnRegister(&loginData);
                        break;

                    case STATE_UNREGISTER:
                        logger->log("State: UNREGISTER");
                        accountServerConnection->disconnect();
                        currentDialog = new OkDialog("Unregister successful",
                                             "Farewell, come back any time ....");
                        loginData.clear();
                        //The errorlistener sets the state to STATE_CHOOSE_SERVER
                        currentDialog->addActionListener(&errorListener);
                        currentDialog = NULL; // OkDialog deletes itself
                        break;

                    case STATE_ACCOUNTCHANGE_ERROR:
                        logger->log("State: ACCOUNT CHANGE ERROR");
                        currentDialog = new OkDialog("Error ", errorMessage);
                        currentDialog->addActionListener(&accountListener);
                        currentDialog = NULL; // OkDialog deletes itself
                        break;


                    case STATE_ERROR:
                        logger->log("State: ERROR");
                        currentDialog = new OkDialog("Error", errorMessage);
                        currentDialog->addActionListener(&errorListener);
                        currentDialog = NULL; // OkDialog deletes itself
                        gameServerConnection->disconnect();
                        chatServerConnection->disconnect();
                        Net::clearHandlers();
                        break;

                    case STATE_CONNECT_GAME:
                        logger->log("State: CONNECT_GAME");
                        currentDialog = new ConnectionDialog(STATE_SWITCH_ACCOUNTSERVER_ATTEMPT);
                        break;

                    case STATE_GAME:
                        logger->log("Memorizing selected character %s",
                                player_node->getName().c_str());
                        config.setValue("lastCharacter", player_node->getName());

                        Net::GameServer::connect(gameServerConnection, token);
                        Net::ChatServer::connect(chatServerConnection, token);
                        sound.fadeOutMusic(1000);

#ifdef PACKAGE_VERSION
                        delete versionLabel;
                        versionLabel = NULL;
#endif
                        currentDialog = NULL;

                        logger->log("State: GAME");
                        game = new Game;
                        game->logic();
                        delete game;

                        // If the quitdialog didn't set the next state
                        if (state == STATE_GAME)
                        {
                            state = STATE_EXIT;
                        }
                        break;

                    case STATE_SWITCH_CHARACTER:
                        logger->log("State: SWITCH_CHARACTER");
                        switchCharacter(&token);
                        break;

                    case STATE_RECONNECT_ACCOUNT:
                        logger->log("State: RECONNECT_ACCOUNT");

                        // Done with game & chat
                        gameServerConnection->disconnect();
                        chatServerConnection->disconnect();

                        accountServerConnection->connect(
                                loginData.hostname,
                                loginData.port);
                        break;

                    case STATE_WAIT:
                        break;

                    case STATE_EXIT:
                        logger->log("State: EXIT");
                        logoutThenExit();
                        break;

                    default:
                        state = STATE_FORCE_QUIT;
                        break;
                }
            }
        }

#ifdef PACKAGE_VERSION
        delete versionLabel;
#endif
    }
    catch (...)
    {
        logger->log("Exception");
    }

    if (accountServerConnection)
        accountServerConnection->disconnect();
    if (gameServerConnection)
        gameServerConnection->disconnect();
    if (chatServerConnection)
        chatServerConnection->disconnect();

    delete accountServerConnection;
    delete gameServerConnection;
    delete chatServerConnection;
    Net::finalize();

    logger->log("Quitting");
    exit_engine();
    PHYSFS_deinit();
    delete logger;

    return 0;
}