/*
    ircprotocol - IRC Protocol

    Copyright (c) 2002      by Nick Betcher <nbetcher@kde.org>

    Kopete    (c) 2002-2004 by the Kopete developers <kopete-devel@kde.org>

    *************************************************************************
    *                                                                       *
    * This program 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     *
    * (at your option) any later version.                                   *
    *                                                                       *
    *************************************************************************
*/

#include "ircaccount.h"
#include "ircprotocol.h"
#include "ksparser.h"

#include "ircaddcontactpage.h"
#include "ircchannelcontact.h"
#include "irccontactmanager.h"

#include "networkconfig.h"
#include "channellist.h"
#include "ircguiclient.h"
#include "ircusercontact.h"
#include "irceditaccountwidget.h"
#include "irctransferhandler.h"

#include "kircengine.h"

#include "kopeteaccountmanager.h"
#include "kopetecommandhandler.h"
#include "kopeteglobal.h"
#include "kopeteonlinestatusmanager.h"
#include "kopeteonlinestatus.h"
#include "kopeteview.h"
#include "kopeteuiglobal.h"

#undef KDE_NO_COMPAT
#include <tdeaction.h>
#include <kcharsets.h>
#include <kdebug.h>
#include <kgenericfactory.h>
#include <tdeglobal.h>
#include <kinputdialog.h>
#include <kiconloader.h>
#include <tdemessagebox.h>
#include <ksimpleconfig.h>
#include <kstandarddirs.h>
#include <kuser.h>

#include <tqcheckbox.h>
#include <tqdom.h>
#include <tqfile.h>
#include <tqlineedit.h>
#include <tqpushbutton.h>
#include <tqregexp.h>
#include <tqspinbox.h>
#include <tqvalidator.h>

#include <dom/html_element.h>
#include <unistd.h>

typedef KGenericFactory<IRCProtocol> IRCProtocolFactory;
K_EXPORT_COMPONENT_FACTORY( kopete_irc, IRCProtocolFactory( "kopete_irc" )  )

IRCProtocol *IRCProtocol::s_protocol = 0L;

IRCProtocolHandler::IRCProtocolHandler() : Kopete::MimeTypeHandler( false )
{
	registerAsProtocolHandler( TQString::fromLatin1("irc") );
}

void IRCProtocolHandler::handleURL( const KURL &url ) const
{
	kdDebug(14120) << url << endl;
	if( !url.isValid() )
		return;

	unsigned short port = url.port();
	if( port == 0 )
		port = 6667;

	TQString chan = url.url().section('/',3);
	if( chan.isEmpty() )
		return;

	KUser user( getuid() );
	TQString accountId = TQString::fromLatin1("%1@%2:%3").arg(
		user.loginName(),
		url.host(),
		TQString::number(port)
	);

	kdDebug(14120) << accountId << endl;

	IRCAccount *newAccount = new IRCAccount( IRCProtocol::protocol(), accountId, chan );
	newAccount->setNickName( user.loginName() );
	newAccount->setUserName( user.loginName() );
	newAccount->connect();
}

IRCProtocol::IRCProtocol( TQObject *parent, const char *name, const TQStringList & /* args */ )
: Kopete::Protocol( IRCProtocolFactory::instance(), parent, name ),

	m_ServerStatusOnline(Kopete::OnlineStatus::Online,
			100, this, OnlineServer, TQString(), i18n("Online")),
	m_ServerStatusOffline(Kopete::OnlineStatus::Offline,
			90, this, OfflineServer, TQString(), i18n("Offline")),

	m_ChannelStatusOnline(Kopete::OnlineStatus::Online,
			80, this, OnlineChannel, TQString(), i18n("Online")),
	m_ChannelStatusOffline(Kopete::OnlineStatus::Offline,
			70, this, OfflineChannel, TQString(), i18n("Offline")),

	m_UserStatusOpVoice(Kopete::OnlineStatus::Online,
			60, this, Online | Operator | Voiced, TQStringList::split(' ',"irc_voice irc_op"), i18n("Op")),
	m_UserStatusOpVoiceAway(Kopete::OnlineStatus::Away,
			55, this, Online | Operator | Voiced | Away,
			TQStringList::split(' ',"irc_voice irc_op contact_away_overlay"), i18n("Away")),

	m_UserStatusOp(Kopete::OnlineStatus::Online,
			50, this, Online | Operator, "irc_op", i18n("Op")),
	m_UserStatusOpAway(Kopete::OnlineStatus::Away,
			45, this, Online | Operator | Away,
			TQStringList::split(' ',"irc_op contact_away_overlay"), i18n("Away")),

	m_UserStatusVoice(Kopete::OnlineStatus::Online,
			40, this, Online | Voiced, "irc_voice", i18n("Voice")),
	m_UserStatusVoiceAway(Kopete::OnlineStatus::Away,
			35, this, Online | Voiced | Away,
			TQStringList::split(' ',"irc_voice contact_away_overlay"),  i18n("Away")),

	m_UserStatusOnline(Kopete::OnlineStatus::Online,
			25, this, Online, TQString(), i18n("Online"), i18n("Online"), Kopete::OnlineStatusManager::Online),

	m_UserStatusAway(Kopete::OnlineStatus::Away,
			2, this, Online | Away, "contact_away_overlay",
			i18n("Away"), i18n("Away"), Kopete::OnlineStatusManager::Away),
	m_UserStatusConnecting(Kopete::OnlineStatus::Connecting,
			1, this, Connecting, "irc_connecting", i18n("Connecting")),
	m_UserStatusOffline(Kopete::OnlineStatus::Offline,
			0, this, Offline, TQString(), i18n("Offline"), i18n("Offline"), Kopete::OnlineStatusManager::Offline),

	m_StatusUnknown(Kopete::OnlineStatus::Unknown,
			999, this, 999, "status_unknown", i18n("Status not available")),

	propChannelTopic(TQString::fromLatin1("channelTopic"), i18n("Topic"), TQString(), false, true ),
	propChannelMembers(TQString::fromLatin1("channelMembers"), i18n("Members")),
	propHomepage(TQString::fromLatin1("homePage"), i18n("Home Page")),
	propLastSeen(Kopete::Global::Properties::self()->lastSeen()),
	propUserInfo(TQString::fromLatin1("userInfo"), i18n("IRC User")),
	propServer(TQString::fromLatin1("ircServer"), i18n("IRC Server")),
	propChannels( TQString::fromLatin1("ircChannels"), i18n("IRC Channels")),
	propHops(TQString::fromLatin1("ircHops"), i18n("IRC Hops")),
	propFullName(TQString::fromLatin1("FormattedName"), i18n("Full Name")),
	propIsIdentified(TQString::fromLatin1("identifiedUser"), i18n("User Is Authenticated"))
{
//	kdDebug(14120) << k_funcinfo << endl;

	s_protocol = this;

	//m_status = m_unknownStatus = m_Unknown;

	addAddressBookField("messaging/irc", Kopete::Plugin::MakeIndexField);

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("raw"),
		TQ_SLOT( slotRawCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /raw <text> - Sends the text in raw form to the server."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("quote"),
		TQ_SLOT( slotQuoteCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /quote <text> - Sends the text in quoted form to the server."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("ctcp"),
		TQ_SLOT( slotCtcpCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /ctcp <nick> <message> - Send the CTCP message to nick<action>."), 2 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("ping"),
		TQ_SLOT( slotPingCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /ping <nickname> - Alias for /CTCP <nickname> PING."), 1, 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("motd"),
		TQ_SLOT( slotMotdCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /motd [<server>] - Shows the message of the day for the current or the given server.") );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("list"),
		TQ_SLOT( slotListCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /list - List the public channels on the server.") );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("join"),
		TQ_SLOT( slotJoinCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /join <#channel 1> [<password>] - Joins the specified channel."), 1, 2 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("topic"),
		TQ_SLOT( slotTopicCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /topic [<topic>] - Sets and/or displays the topic for the active channel.") );

	//FIXME: Update help text
	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("whois"),
		TQ_SLOT( slotWhoisCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /whois <nickname> - Display whois info on this user."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("whowas"),
		TQ_SLOT( slotWhoWasCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /whowas <nickname> - Display whowas info on this user."), 1, 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("who"),
		TQ_SLOT( slotWhoCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /who <nickname|channel> - Display who info on this user/channel."), 1, 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("query"),
		TQ_SLOT( slotQueryCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /query <nickname> [<message>] - Open a private chat with this user."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("mode"),
		TQ_SLOT( slotModeCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /mode <channel> <modes> - Set modes on the given channel."), 2 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("nick"),
		TQ_SLOT( slotNickCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /nick <nickname> - Change your nickname to the given one."), 1, 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("me"),
		TQ_SLOT( slotMeCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /me <action> - Do something."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("ame"),
		TQ_SLOT( slotAllMeCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /ame <action> - Do something in every open chat."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("kick"),
		TQ_SLOT( slotKickCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /kick <nickname> [<reason>] - Kick someone from the channel (requires operator status).")
		, 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("ban"),
		TQ_SLOT( slotBanCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /ban <mask> - Add someone to this channel's ban list. (requires operator status)."),
		1, 1 );

	Kopete::CommandHandler::commandHandler()->registerAlias( this, TQString::fromLatin1("bannick"),
		TQString::fromLatin1("ban %1!*@*"),
		i18n("USAGE: /bannick <nickname> - Add someone to this channel's ban list. Uses the hostmask nickname!*@* (requires operator status)."), Kopete::CommandHandler::SystemAlias, 1, 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("op"),
		TQ_SLOT( slotOpCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /op <nickname 1> [<nickname 2> <...>] - Give channel operator status to someone (requires operator status)."),
		1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("deop"),
		TQ_SLOT( slotDeopCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /deop <nickname> [<nickname 2> <...>]- Remove channel operator status from someone (requires operator status)."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("voice"),
		TQ_SLOT( slotVoiceCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /voice <nickname> [<nickname 2> <...>]- Give channel voice status to someone (requires operator status)."),
		1);

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("devoice"),
		TQ_SLOT( slotDevoiceCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /devoice <nickname> [<nickname 2> <...>]- Remove channel voice status from someone (requires operator status)."), 1 );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("quit"),
		TQ_SLOT( slotQuitCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /quit [<reason>] - Disconnect from IRC, optionally leaving a message.") );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("part"),
		TQ_SLOT( slotPartCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /part [<reason>] - Part from a channel, optionally leaving a message.") );

	Kopete::CommandHandler::commandHandler()->registerCommand( this, TQString::fromLatin1("invite"),
		TQ_SLOT( slotInviteCommand( const TQString &, Kopete::ChatSession*) ),
		i18n("USAGE: /invite <nickname> [<channel>] - Invite a user to join a channel."), 1 );

	Kopete::CommandHandler::commandHandler()->registerAlias( this, TQString::fromLatin1("j"),
		TQString::fromLatin1("join %1"),
		i18n("USAGE: /j <#channel 1> [<password>] - Alias for JOIN."), Kopete::CommandHandler::SystemAlias,
		1, 2 );

	Kopete::CommandHandler::commandHandler()->registerAlias( this, TQString::fromLatin1("msg"),
		TQString::fromLatin1("query %s"),
		i18n("USAGE: /msg <nickname> [<message>] - Alias for QUERY <nickname> <message>."), Kopete::CommandHandler::SystemAlias, 1 );

	TQObject::connect( Kopete::ChatSessionManager::self(), TQ_SIGNAL(aboutToDisplay(Kopete::Message &)),
		this, TQ_SLOT(slotMessageFilter(Kopete::Message &)) );

	TQObject::connect( Kopete::ChatSessionManager::self(), TQ_SIGNAL( viewCreated( KopeteView* ) ),
		this, TQ_SLOT( slotViewCreated( KopeteView* ) ) );

	setCapabilities( Kopete::Protocol::RichBFormatting | Kopete::Protocol::RichUFormatting | Kopete::Protocol::RichColor );

	netConf = 0L;

	slotReadNetworks();

	m_protocolHandler = new IRCProtocolHandler();

	IRCTransferHandler::self(); // Initiate the transfer handling system.
}

IRCProtocol * IRCProtocol::protocol()
{
	return s_protocol;
}

IRCProtocol::~IRCProtocol()
{
	delete m_protocolHandler;
}

const Kopete::OnlineStatus IRCProtocol::statusLookup( IRCStatus status ) const
{
	kdDebug(14120) << k_funcinfo << "Looking up status for " << status << endl;

	switch( status )
	{
	case Offline:
		return m_UserStatusOffline;
	case Connecting:
		return m_UserStatusConnecting;

	// Regular user
	case Online:
		return m_UserStatusOnline;
	case Online | Away:
		return m_UserStatusAway;

	// Voiced
	case Online | Voiced:
		return m_UserStatusVoice;
	case Online | Away | Voiced:
		return m_UserStatusVoiceAway;

	// Operator
	case Online | Operator:
		return m_UserStatusOp;
	case Online | Away | Operator:
		return m_UserStatusOpAway;
	case Online | Operator | Voiced:
		return m_UserStatusOpVoice;
	case Online | Operator | Voiced | Away:
		return m_UserStatusOpVoiceAway;

	// Server
	case OnlineServer:
		return m_ServerStatusOnline;
	case OfflineServer:
		return m_ServerStatusOffline;

	// Channel
	case OnlineChannel:
		return m_ChannelStatusOnline;
	case OfflineChannel:
		return m_ChannelStatusOffline;

	default:
		return m_StatusUnknown;
	}
}

void IRCProtocol::slotViewCreated( KopeteView *view )
{
	if( view->msgManager()->protocol() == this )
		new IRCGUIClient( view->msgManager() );
}

void IRCProtocol::slotMessageFilter( Kopete::Message &msg )
{
	if( msg.from()->protocol() == this )
	{
		TQString messageText = msg.escapedBody();

		//Add right click for channels, only replace text not in HTML tags
		messageText.replace( TQRegExp( TQString::fromLatin1("(?![^<]+>)(#[^#\\s]+)(?![^<]+>)") ), TQString::fromLatin1("<span class=\"KopeteLink\" type=\"IRCChannel\">\\1</span>") );

		msg.setBody( messageText, Kopete::Message::RichText );
	}
}

TQPtrList<TDEAction> *IRCProtocol::customChatWindowPopupActions( const Kopete::Message &m, DOM::Node &n )
{
	DOM::HTMLElement e = n;

	//isNull checks that the cast was successful
	if( !e.isNull() && !m.to().isEmpty() )
	{
		activeNode = n;
		activeAccount = static_cast<IRCAccount*>( m.from()->account() );
		if( e.getAttribute( TQString::fromLatin1("type") ) == TQString::fromLatin1("IRCChannel") )
			return activeAccount->contactManager()->findChannel(
				e.innerText().string() )->customContextMenuActions();
	}

	return 0L;
}

AddContactPage *IRCProtocol::createAddContactWidget(TQWidget *parent, Kopete::Account *account)
{
	return new IRCAddContactPage(parent,static_cast<IRCAccount*>(account));
}

KopeteEditAccountWidget *IRCProtocol::createEditAccountWidget(Kopete::Account *account, TQWidget *parent)
{
	return new IRCEditAccountWidget(this, static_cast<IRCAccount*>(account),parent);
}

Kopete::Account *IRCProtocol::createNewAccount(const TQString &accountId)
{
	return new IRCAccount( this, accountId );
}

Kopete::Contact *IRCProtocol::deserializeContact( Kopete::MetaContact *metaContact, const TQMap<TQString, TQString> &serializedData,
	const TQMap<TQString, TQString> & /* addressBookData */ )
{
	kdDebug(14120) << k_funcinfo << endl;

	TQString contactId = serializedData[ "contactId" ];
	TQString displayName = serializedData[ "displayName" ];

	if( displayName.isEmpty() )
		displayName = contactId;

	TQDict<Kopete::Account> accounts = Kopete::AccountManager::self()->accounts( this );
	if( !accounts.isEmpty() )
	{
		Kopete::Account *a = accounts[ serializedData[ "accountId" ] ];
		if( a )
		{
			a->addContact( contactId, metaContact );
			return a->contacts()[contactId];
		}
		else
			kdDebug(14120) << k_funcinfo << serializedData[ "accountId" ] << " was a contact's account,"
				" but we don't have it in the accounts list" << endl;
	}
	else
		kdDebug(14120) << k_funcinfo << "No accounts loaded!" << endl;

	return 0;
}

void IRCProtocol::slotRawCommand( const TQString &args, Kopete::ChatSession *manager )
{
	IRCAccount *account = static_cast<IRCAccount*>( manager->account() );

	if (!args.isEmpty())
	{
		account->engine()->writeRawMessage(args);
	}
	else
	{
		account->appendMessage(i18n("You must enter some text to send to the server."),
				IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotQuoteCommand( const TQString &args, Kopete::ChatSession *manager )
{
	IRCAccount *account = static_cast<IRCAccount*>( manager->account() );

	if( !args.isEmpty() )
	{
		account->engine()->writeMessage( args );
	}
	else
	{
		account->appendMessage(i18n("You must enter some text to send to the server."),
				IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotCtcpCommand( const TQString &args, Kopete::ChatSession *manager )
{
	if( !args.isEmpty() )
	{
		TQString user = args.section( ' ', 0, 0 );
		TQString message = args.section( ' ', 1 );
		static_cast<IRCAccount*>( manager->account() )->engine()->writeCtcpQueryMessage( user, TQString(), message );
	}
}

void IRCProtocol::slotMotdCommand( const TQString &args, Kopete::ChatSession *manager )
{
	TQStringList argsList = Kopete::CommandHandler::parseArguments( args );
	static_cast<IRCAccount*>( manager->account() )->engine()->motd(argsList.front());
}

void IRCProtocol::slotPingCommand( const TQString &args, Kopete::ChatSession *manager )
{
	TQStringList argsList = Kopete::CommandHandler::parseArguments(args);
	static_cast<IRCAccount*>( manager->account() )->engine()->CtcpRequest_ping(argsList.front());
}

void IRCProtocol::slotListCommand( const TQString &/*args*/, Kopete::ChatSession *manager )
{
	static_cast<IRCAccount*>( manager->account() )->listChannels();
}

void IRCProtocol::slotTopicCommand( const TQString &args, Kopete::ChatSession *manager )
{
	Kopete::ContactPtrList members = manager->members();
	IRCChannelContact *chan = dynamic_cast<IRCChannelContact*>( members.first() );
	if( chan )
	{
		if( !args.isEmpty() )
			chan->setTopic( args );
		else
		{
			static_cast<IRCAccount*>(manager->account())->engine()->
				writeRawMessage(TQString::fromLatin1("TOPIC %1").arg(chan->nickName()));
		}
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("You must be in a channel to use this command."), IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotJoinCommand( const TQString &arg, Kopete::ChatSession *manager )
{
	TQStringList args = Kopete::CommandHandler::parseArguments( arg );
	if( KIRC::Entity::isChannel(args[0]) )
	{
		IRCChannelContact *chan = static_cast<IRCAccount*>( manager->account() )->contactManager()->findChannel( args[0] );
		if( args.count() == 2 )
			chan->setPassword( args[1] );
		static_cast<IRCAccount*>( manager->account() )->engine()->join(args[0], chan->password());
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("\"%1\" is an invalid channel. Channels must start with '#', '!', '+', or '&'.")
			.arg(args[0]), IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotInviteCommand( const TQString &args, Kopete::ChatSession *manager )
{
	IRCChannelContact *c = 0L;
	TQStringList argsList = Kopete::CommandHandler::parseArguments( args );

	if( argsList.count() > 1 )
	{
		if( KIRC::Entity::isChannel(argsList[1]) )
		{
			c = static_cast<IRCAccount*>( manager->account() )->contactManager()->
				findChannel( argsList[1] );
		}
		else
		{
			static_cast<IRCAccount*>( manager->account() )->appendMessage(
				i18n("\"%1\" is an invalid channel. Channels must start with '#', '!', '+', or '&'.")
				.arg(argsList[1]), IRCAccount::ErrorReply );
		}
	}
	else
	{
		Kopete::ContactPtrList members = manager->members();
		c = dynamic_cast<IRCChannelContact*>( members.first() );
	}

	if( c && c->manager()->contactOnlineStatus( manager->myself() ) == m_UserStatusOp )
	{
		static_cast<IRCAccount*>( manager->account() )->engine()->writeMessage(
			TQString::fromLatin1("INVITE %1 %2").arg( argsList[0] ).
			arg( c->nickName() )
		);
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("You must be a channel operator to perform this operation."), IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotQueryCommand( const TQString &args, Kopete::ChatSession *manager )
{
	TQString user = args.section( ' ', 0, 0 );
	TQString rest = args.section( ' ', 1 );

	if( !KIRC::Entity::isChannel(user) )
	{
		IRCUserContact *c = static_cast<IRCAccount*>( manager->account() )->
			contactManager()->findUser( user );
		c->startChat();
		if( !rest.isEmpty() )
		{
			Kopete::Message msg( c->manager()->myself(), c->manager()->members(), rest,
				Kopete::Message::Outbound, Kopete::Message::PlainText, CHAT_VIEW);
			c->manager()->sendMessage(msg);
		}
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("\"%1\" is an invalid nickname. Nicknames must not start with '#','!','+', or '&'.").arg(user),
			IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotWhoisCommand( const TQString &args, Kopete::ChatSession *manager )
{
	static_cast<IRCAccount*>( manager->account() )->engine()->whois( args );
	static_cast<IRCAccount*>( manager->account() )->setCurrentCommandSource( manager );
}

void IRCProtocol::slotWhoCommand( const TQString &args, Kopete::ChatSession *manager )
{
	TQStringList argsList = Kopete::CommandHandler::parseArguments( args );
	static_cast<IRCAccount*>( manager->account() )->engine()->writeMessage(
		TQString::fromLatin1("WHO %1").arg( argsList.first() ) );
	static_cast<IRCAccount*>( manager->account() )->setCurrentCommandSource( manager );
}

void IRCProtocol::slotWhoWasCommand( const TQString &args, Kopete::ChatSession *manager )
{
	TQStringList argsList = Kopete::CommandHandler::parseArguments( args );
	static_cast<IRCAccount*>( manager->account() )->engine()->writeMessage(
		TQString::fromLatin1("WHOWAS %1").arg( argsList.first() ) );
	static_cast<IRCAccount*>( manager->account() )->setCurrentCommandSource( manager );
}

void IRCProtocol::slotQuitCommand( const TQString &args, Kopete::ChatSession *manager )
{
	static_cast<IRCAccount*>( manager->account() )->quit( args );
}

void IRCProtocol::slotNickCommand( const TQString &args, Kopete::ChatSession *manager )
{
	TQStringList argsList = Kopete::CommandHandler::parseArguments( args );
	static_cast<IRCAccount*>( manager->account() )->engine()->nick( argsList.front() );
}

void IRCProtocol::slotModeCommand(const TQString &args, Kopete::ChatSession *manager)
{
	TQStringList argsList = Kopete::CommandHandler::parseArguments( args );
	static_cast<IRCAccount*>( manager->account() )->engine()->mode( argsList.front(),
		args.section( TQRegExp(TQString::fromLatin1("\\s+")), 1 ) );
}

void IRCProtocol::slotMeCommand(const TQString &args, Kopete::ChatSession *manager)
{
	Kopete::ContactPtrList members = manager->members();
	static_cast<IRCAccount*>( manager->account() )->engine()->CtcpRequest_action(
		static_cast<const IRCContact*>(members.first())->nickName(), args
	);
}

void IRCProtocol::slotAllMeCommand(const TQString &args, Kopete::ChatSession *)
{
	TQValueList<Kopete::ChatSession*> sessions = Kopete::ChatSessionManager::self()->sessions();

	for( TQValueList<Kopete::ChatSession*>::iterator it = sessions.begin(); it != sessions.end(); ++it )
	{
		Kopete::ChatSession *session = *it;
		if( session->protocol() == this )
			slotMeCommand(args, session);
	}
}

void IRCProtocol::slotKickCommand(const TQString &args, Kopete::ChatSession *manager)
{
	if (manager->contactOnlineStatus( manager->myself() ) == m_UserStatusOp)
	{
		TQRegExp spaces(TQString::fromLatin1("\\s+"));
		TQString nick = args.section( spaces, 0, 0);
		TQString reason = args.section( spaces, 1);
		Kopete::ContactPtrList members = manager->members();
		TQString channel = static_cast<IRCContact*>( members.first() )->nickName();
		if (KIRC::Entity::isChannel(channel))
			static_cast<IRCAccount*>(manager->account())->engine()->kick(nick, channel, reason);
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("You must be a channel operator to perform this operation."), IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotBanCommand( const TQString &args, Kopete::ChatSession *manager )
{
	if( manager->contactOnlineStatus( manager->myself() ) == m_UserStatusOp )
	{
		TQStringList argsList = Kopete::CommandHandler::parseArguments( args );
		Kopete::ContactPtrList members = manager->members();
		IRCChannelContact *chan = static_cast<IRCChannelContact*>( members.first() );
		if( chan && chan->locateUser( argsList.front() ) )
			chan->setMode( TQString::fromLatin1("+b %1").arg( argsList.front() ) );
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("You must be a channel operator to perform this operation."), IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotPartCommand( const TQString &args, Kopete::ChatSession *manager )
{
	TQStringList argsList = Kopete::CommandHandler::parseArguments(args);
	Kopete::ContactPtrList members = manager->members();
	IRCChannelContact *chan = static_cast<IRCChannelContact*>(members.first());

	if (chan)
	{
		if(!args.isEmpty())
			static_cast<IRCAccount*>(manager->account())->engine()->part(chan->nickName(), args);
		else
			chan->part();
		if( manager->view() )
			manager->view()->closeView(true);
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("You must be in a channel to use this command."), IRCAccount::ErrorReply );
	}
}

void IRCProtocol::slotOpCommand( const TQString &args, Kopete::ChatSession *manager )
{
	simpleModeChange( args, manager, TQString::fromLatin1("+o") );
}

void IRCProtocol::slotDeopCommand( const TQString &args, Kopete::ChatSession *manager )
{
	simpleModeChange( args, manager, TQString::fromLatin1("-o") );
}

void IRCProtocol::slotVoiceCommand( const TQString &args, Kopete::ChatSession *manager )
{
	simpleModeChange( args, manager, TQString::fromLatin1("+v") );
}

void IRCProtocol::slotDevoiceCommand( const TQString &args, Kopete::ChatSession *manager )
{
	simpleModeChange( args, manager, TQString::fromLatin1("-v") );
}

void IRCProtocol::simpleModeChange( const TQString &args, Kopete::ChatSession *manager, const TQString &mode )
{
	if( manager->contactOnlineStatus( manager->myself() ) == m_UserStatusOp )
	{
		TQStringList argsList = Kopete::CommandHandler::parseArguments( args );
		Kopete::ContactPtrList members = manager->members();
		IRCChannelContact *chan = static_cast<IRCChannelContact*>( members.first() );
		if( chan )
		{
			for( TQStringList::iterator it = argsList.begin(); it != argsList.end(); ++it )
			{
				if( chan->locateUser( *it ) )
					chan->setMode( TQString::fromLatin1("%1 %2").arg( mode ).arg( *it ) );
			}
		}
	}
	else
	{
		static_cast<IRCAccount*>( manager->account() )->appendMessage(
			i18n("You must be a channel operator to perform this operation."), IRCAccount::ErrorReply );
	}
}

void IRCProtocol::editNetworks( const TQString &networkName )
{
	if( !netConf )
	{
		netConf = new NetworkConfig( Kopete::UI::Global::mainWidget(), "network_config", true );
		netConf->host->setValidator( new TQRegExpValidator( TQString::fromLatin1("^[\\w-\\.]*$"), netConf ) );
		netConf->upButton->setIconSet( SmallIconSet( "go-up" )  );
		netConf->downButton->setIconSet( SmallIconSet( "go-down" ) );

		connect( netConf->networkList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkConfig() ) );
		connect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
		connect( netConf, TQ_SIGNAL( accepted() ), this, TQ_SLOT( slotSaveNetworkConfig() ) );
		connect( netConf, TQ_SIGNAL( rejected() ), this, TQ_SLOT( slotReadNetworks() ) );
		connect( netConf->upButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotMoveServerUp() ) );
		connect( netConf->downButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotMoveServerDown() ) );
		connect( netConf->removeNetwork, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotDeleteNetwork() ) );
		connect( netConf->removeHost, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotDeleteHost() ) );
		connect( netConf->newHost, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotNewHost() ) );
		connect( netConf->newNetwork, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotNewNetwork() ) );
		connect( netConf->renameNetwork, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotRenameNetwork() ) );
		connect( netConf->port, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( slotHostPortChanged( int ) ) );
		connect( netConf->networkList, TQ_SIGNAL( doubleClicked ( TQListBoxItem * )), TQ_SLOT(slotRenameNetwork()));
				
	}

	disconnect( netConf->networkList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkConfig() ) );
	disconnect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );

	netConf->networkList->clear();

	for( TQDictIterator<IRCNetwork> it( m_networks ); it.current(); ++it )
	{
		IRCNetwork *net = it.current();
		netConf->networkList->insertItem( net->name );
	}

	netConf->networkList->sort();

	connect( netConf->networkList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkConfig() ) );
	connect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );

	if( !networkName.isEmpty() )
		netConf->networkList->setSelected( netConf->networkList->findItem( networkName ), true );

	//slotUpdateNetworkConfig(); // unnecessary, setSelected emits selectionChanged

	netConf->show();
}

void IRCProtocol::slotUpdateNetworkConfig()
{
	// update the data structure of the previous selection from the UI
	storeCurrentNetwork();

	// update the UI from the data for the current selection
	IRCNetwork *net = m_networks[ netConf->networkList->currentText() ];
	if( net )
	{
		netConf->description->setText( net->description );
		netConf->hostList->clear();

		for( TQValueList<IRCHost*>::iterator it = net->hosts.begin(); it != net->hosts.end(); ++it )
			netConf->hostList->insertItem( (*it)->host + TQString::fromLatin1(":") + TQString::number((*it)->port) );

		// prevent nested event loop crash
		disconnect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
		netConf->hostList->setSelected( 0, true );
		slotUpdateNetworkHostConfig();
		connect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
	}

	// record the current selection
	m_uiCurrentNetworkSelection = netConf->networkList->currentText();
}

void IRCProtocol::storeCurrentNetwork()
{
	if ( !m_uiCurrentNetworkSelection.isEmpty() )
	{
		IRCNetwork *net = m_networks[ m_uiCurrentNetworkSelection ];
		if ( net )
		{
			net->description = netConf->description->text(); // crash on 2nd dialog show here!
		}
		else
			kdDebug( 14120 ) << m_uiCurrentNetworkSelection << " was already gone from the cache!" << endl;
	}
}

void IRCProtocol::storeCurrentHost()
{
	if ( !m_uiCurrentHostSelection.isEmpty()  )
	{
		IRCHost *host = m_hosts[ m_uiCurrentHostSelection ];
		if ( host )
		{
			host->host = netConf->host->text();
			host->password = netConf->password->text();
			host->port = netConf->port->text().toInt();
			host->ssl = netConf->useSSL->isChecked();
		}
	}
}

void IRCProtocol::slotHostPortChanged( int value )
{
	TQString entryText = m_uiCurrentHostSelection + TQString::fromLatin1(":") + TQString::number( value );
	// changeItem causes a take() and insert, and we don't want a selectionChanged() signal that sets all this off again.
	disconnect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
	netConf->hostList->changeItem( entryText, netConf->hostList->currentItem() );
	connect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
}

void IRCProtocol::slotUpdateNetworkHostConfig()
{
	storeCurrentHost();

	if ( netConf->hostList->selectedItem() )
	{
		m_uiCurrentHostSelection = netConf->hostList->currentText().section(':', 0, 0);
		IRCHost *host = m_hosts[ m_uiCurrentHostSelection ];

		if( host )
		{
			netConf->host->setText( host->host );
			netConf->password->setText( host->password );
			disconnect( netConf->port, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( slotHostPortChanged( int ) ) );
			netConf->port->setValue( host->port );
			connect( netConf->port, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( slotHostPortChanged( int ) ) );
			netConf->useSSL->setChecked( host->ssl );

			netConf->upButton->setEnabled( netConf->hostList->currentItem() > 0 );
			netConf->downButton->setEnabled( netConf->hostList->currentItem() < (int)( netConf->hostList->count() - 1 ) );
		}
	}
	else
	{
		m_uiCurrentHostSelection = TQString();
		disconnect( netConf->port, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( slotHostPortChanged( int ) ) );
		netConf->host->clear();
		netConf->password->clear();
		netConf->port->setValue( 6667 );
		netConf->useSSL->setChecked( false );
		connect( netConf->port, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( slotHostPortChanged( int ) ) );
	}
}

void IRCProtocol::slotDeleteNetwork()
{
	TQString network = netConf->networkList->currentText();
	if( KMessageBox::warningContinueCancel(
		Kopete::UI::Global::mainWidget(), i18n("<qt>Are you sure you want to delete the network <b>%1</b>?<br>"
		"Any accounts which use this network will have to be modified.</qt>")
		.arg(network), i18n("Deleting Network"),
		KGuiItem(i18n("&Delete Network"),"edit-delete"), TQString::fromLatin1("AskIRCDeleteNetwork") ) == KMessageBox::Continue )
	{
		disconnect( netConf->networkList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkConfig() ) );
		disconnect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
		IRCNetwork *net = m_networks[ network ];
		for( TQValueList<IRCHost*>::iterator it = net->hosts.begin(); it != net->hosts.end(); ++it )
		{
			m_hosts.remove( (*it)->host );
			delete (*it);
		}
		m_networks.remove( network );
		delete net;
		netConf->networkList->removeItem( netConf->networkList->currentItem() );
 		connect( netConf->networkList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkConfig() ) );
		connect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
		slotUpdateNetworkHostConfig();

	}
}

void IRCProtocol::slotDeleteHost()
{
	TQString hostName = netConf->host->text();
	if ( KMessageBox::warningContinueCancel(
		Kopete::UI::Global::mainWidget(), i18n("<qt>Are you sure you want to delete the host <b>%1</b>?</qt>")
		.arg(hostName), i18n("Deleting Host"),
		KGuiItem(i18n("&Delete Host"),"edit-delete"), TQString::fromLatin1("AskIRCDeleteHost")) == KMessageBox::Continue )
	{
		IRCHost *host = m_hosts[ hostName ];
		if ( host )
		{
			disconnect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );
			TQString entryText = host->host + TQString::fromLatin1(":") + TQString::number(host->port);
			TQListBoxItem * justAdded = netConf->hostList->findItem( entryText );
			netConf->hostList->removeItem( netConf->hostList->index( justAdded ) );
			connect( netConf->hostList, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( slotUpdateNetworkHostConfig() ) );

			// remove from network as well
			IRCNetwork *net = m_networks[ m_uiCurrentNetworkSelection ];
			net->hosts.remove( host );

			m_hosts.remove( host->host );
			delete host;
		}
	}
}

void IRCProtocol::slotNewNetwork()
{
	// create a new network struct
	IRCNetwork *net = new IRCNetwork;
	// give it the name of 'New Network' (incrementing number if needed)
	TQString netName = TQString::fromLatin1( "New Network" );
	if ( m_networks.find( netName ) )
	{
		int newIdx = 1;
		do {
			netName = TQString::fromLatin1( "New Network #%1" ).arg( newIdx++ );
		}
		while ( m_networks.find( netName ) && newIdx < 100 );
		if ( newIdx == 100 ) // pathological case
			return;
	}
	net->name = netName;
	// and add it to the networks dict and list
	m_networks.insert( net->name, net );
	netConf->networkList->insertItem( net->name );
	TQListBoxItem * justAdded = netConf->networkList->findItem( net->name );
	netConf->networkList->setSelected( justAdded, true );
	netConf->networkList->setBottomItem( netConf->networkList->index( justAdded ) );
}

void IRCProtocol::slotNewHost()
{
	// create a new host
	IRCHost *host = new IRCHost;
	// prompt for a name
	bool ok;
	TQString name = KInputDialog::getText(
			i18n("New Host"),
			i18n("Enter the hostname of the new server:"),
			TQString(), &ok, Kopete::UI::Global::mainWidget() );
	if ( ok )
	{
		// dupe check
		if ( m_hosts[ name ] )
		{
			KMessageBox::sorry(netConf, i18n( "A host already exists with that name" ) );
			return;
		}
		// set defaults on others
		host->host = name;
		host->port = 6667;
		host->ssl = false;
		// add it to the dict
		m_hosts.insert( host->host, host );
		// add it to the network!
		IRCNetwork *net = m_networks[ netConf->networkList->currentText() ];
		net->hosts.append( host );
		// add it to the gui
		TQString entryText = host->host + TQString::fromLatin1(":") + TQString::number(host->port);
		netConf->hostList->insertItem( entryText );
		// select it in the gui
		TQListBoxItem * justAdded = netConf->hostList->findItem( entryText );
		netConf->hostList->setSelected( justAdded, true );
		//netConf->hostList->setBottomItem( netConf->hostList->index( justAdded ) );
	}
}

void IRCProtocol::slotRenameNetwork()
{
	IRCNetwork *net = m_networks[ m_uiCurrentNetworkSelection ];
	if ( net )
	{
		bool ok;
		// popup up a dialog containing the current name
		TQString name = KInputDialog::getText(
				i18n("Rename Network"),
				i18n("Enter the new name for this network:"),
				m_uiCurrentNetworkSelection, &ok,
				Kopete::UI::Global::mainWidget() );
		if ( ok )
		{
			if ( m_uiCurrentNetworkSelection != name )
			{
				// dupe check
				if ( m_networks[ name ] )
				{
					KMessageBox::sorry(netConf, i18n( "A network already exists with that name" ) );
					return;
				}

				net->name = name;
				// dict
				m_networks.remove( m_uiCurrentNetworkSelection );
				m_networks.insert( net->name, net );
				// ui
				int idx = netConf->networkList->index( netConf->networkList->findItem( m_uiCurrentNetworkSelection ) );
				m_uiCurrentNetworkSelection = net->name;
				netConf->networkList->changeItem( net->name, idx ); // changes the selection!!!
				netConf->networkList->sort();
			}
		}
	}
}

void IRCProtocol::addNetwork( IRCNetwork *network )
{
	m_networks.insert( network->name, network );
	slotSaveNetworkConfig();
}

void IRCProtocol::slotSaveNetworkConfig()
{
	// store any changes in the UI
	storeCurrentNetwork();
	kdDebug( 14120 ) <<  k_funcinfo << m_uiCurrentHostSelection << endl;
	storeCurrentHost();

	TQDomDocument doc("irc-networks");
	TQDomNode root = doc.appendChild( doc.createElement("networks") );

	for( TQDictIterator<IRCNetwork> it( m_networks ); it.current(); ++it )
	{
		IRCNetwork *net = it.current();

		TQDomNode networkNode = root.appendChild( doc.createElement("network") );
		TQDomNode nameNode = networkNode.appendChild( doc.createElement("name") );
		nameNode.appendChild( doc.createTextNode( net->name ) );

		TQDomNode descNode = networkNode.appendChild( doc.createElement("description") );
		descNode.appendChild( doc.createTextNode( net->description ) );

		TQDomNode serversNode = networkNode.appendChild( doc.createElement("servers") );

		for( TQValueList<IRCHost*>::iterator it2 = net->hosts.begin(); it2 != net->hosts.end(); ++it2 )
		{
			TQDomNode serverNode = serversNode.appendChild( doc.createElement("server") );

			TQDomNode hostNode = serverNode.appendChild( doc.createElement("host") );
			hostNode.appendChild( doc.createTextNode( (*it2)->host ) );

			TQDomNode portNode = serverNode.appendChild( doc.createElement("port" ) );
			portNode.appendChild( doc.createTextNode( TQString::number( (*it2)->port ) ) );

			TQDomNode sslNode = serverNode.appendChild( doc.createElement("useSSL") );
			sslNode.appendChild( doc.createTextNode( (*it2)->ssl ? "true" : "false" ) );
		}
	}

//	kdDebug(14121) << k_funcinfo << doc.toString(4) << endl;
	TQFile xmlFile( locateLocal( "appdata", "ircnetworks.xml" ) );

	if (xmlFile.open(IO_WriteOnly))
	{
		TQTextStream stream(&xmlFile);
		stream << doc.toString(4);
		xmlFile.close();
	}
	else
		kdDebug(14121) << k_funcinfo << "Failed to save the Networks definition file" << endl;

	if (netConf)
		emit networkConfigUpdated( netConf->networkList->currentText() );
}

void IRCProtocol::slotReadNetworks()
{
	m_networks.clear();
	m_hosts.clear();

	TQFile xmlFile( locate( "appdata", "ircnetworks.xml" ) );
	xmlFile.open( IO_ReadOnly );

	TQDomDocument doc;
	doc.setContent( &xmlFile );
	TQDomElement networkNode = doc.documentElement().firstChild().toElement();
	while( !networkNode.isNull () )
	{
		IRCNetwork *net = new IRCNetwork;

		TQDomElement networkChild = networkNode.firstChild().toElement();
		while( !networkChild.isNull() )
		{
			if( networkChild.tagName() == "name" )
				net->name = networkChild.text();
			else if( networkChild.tagName() == "description" )
				net->description = networkChild.text();
			else if( networkChild.tagName() == "servers" )
			{
				TQDomElement server = networkChild.firstChild().toElement();
				while( !server.isNull() )
				{
					IRCHost *host = new IRCHost;

					TQDomElement serverChild = server.firstChild().toElement();
					while( !serverChild.isNull() )
					{
						if( serverChild.tagName() == "host" )
							host->host = serverChild.text();
						else if( serverChild.tagName() == "port" )
							host->port = serverChild.text().toInt();
						else if( serverChild.tagName() == "useSSL" )
							host->ssl = ( serverChild.text() == "true" );

						serverChild = serverChild.nextSibling().toElement();
					}

					net->hosts.append( host );
					m_hosts.insert( host->host, host );
					server = server.nextSibling().toElement();
				}
			}
			networkChild = networkChild.nextSibling().toElement();
		}

		m_networks.insert( net->name, net );
		networkNode = networkNode.nextSibling().toElement();
	}

	xmlFile.close();
}

void IRCProtocol::slotMoveServerUp()
{
	IRCHost *selectedHost = m_hosts[ netConf->hostList->currentText().section(':', 0, 0) ];
	IRCNetwork *selectedNetwork = m_networks[ netConf->networkList->currentText() ];

	if( !selectedNetwork || !selectedHost )
		return;

	TQValueList<IRCHost*>::iterator pos = selectedNetwork->hosts.find( selectedHost );
	if( pos != selectedNetwork->hosts.begin() )
	{
		TQValueList<IRCHost*>::iterator lastPos = pos;
		lastPos--;
		selectedNetwork->hosts.insert( lastPos, selectedHost );
		selectedNetwork->hosts.remove( pos );
	}

	unsigned int currentPos = netConf->hostList->currentItem();
	if( currentPos > 0 )
	{
		netConf->hostList->removeItem( currentPos );
		TQString entryText = selectedHost->host + TQString::fromLatin1(":") + TQString::number( selectedHost->port );
		netConf->hostList->insertItem( entryText, --currentPos );
		netConf->hostList->setSelected( currentPos, true );
	}
}

void IRCProtocol::slotMoveServerDown()
{
	IRCHost *selectedHost = m_hosts[ netConf->hostList->currentText().section(':', 0, 0) ];
	IRCNetwork *selectedNetwork = m_networks[ netConf->networkList->currentText() ];

	if( !selectedNetwork || !selectedHost )
		return;

	TQValueList<IRCHost*>::iterator pos = selectedNetwork->hosts.find( selectedHost );
	if( *pos != selectedNetwork->hosts.back() )
	{
		TQValueList<IRCHost*>::iterator nextPos = selectedNetwork->hosts.remove( pos );
		selectedNetwork->hosts.insert( ++nextPos, selectedHost );
	}

	unsigned int currentPos = netConf->hostList->currentItem();
	if( currentPos < ( netConf->hostList->count() - 1 ) )
	{
		netConf->hostList->removeItem( currentPos );
		TQString entryText = selectedHost->host + TQString::fromLatin1(":") + TQString::number( selectedHost->port );
		netConf->hostList->insertItem( entryText, ++currentPos );
		netConf->hostList->setSelected( currentPos, true );
	}
}



#include "ircprotocol.moc"