You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
284 lines
11 KiB
284 lines
11 KiB
/*
|
|
gwprotocol.cpp - Kopete GroupWise Protocol
|
|
|
|
Copyright (c) 2004 SUSE Linux AG http://www.suse.com
|
|
|
|
Based on Testbed
|
|
Copyright (c) 2003 by Will Stephenson <will@stevello.free-online.co.uk>
|
|
rtfizeTest from nm_rtfize_text, from Gaim src/protocols/novell/nmuser.c
|
|
Copyright (c) 2004 Novell, Inc. All Rights Reserved
|
|
|
|
Kopete (c) 2002-2003 by the Kopete developers <kopete-devel@kde.org>
|
|
|
|
*************************************************************************
|
|
* *
|
|
* This library 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 <tqregexp.h>
|
|
#include <tqstringlist.h>
|
|
|
|
#include <kgenericfactory.h>
|
|
#include <kdebug.h>
|
|
|
|
#include "kopeteaccountmanager.h"
|
|
#include "kopeteonlinestatusmanager.h"
|
|
#include "kopeteglobal.h"
|
|
|
|
#include "gwaccount.h"
|
|
#include "gwerror.h"
|
|
#include "gwcontact.h"
|
|
#include "gwprotocol.h"
|
|
#include "ui/gwaddcontactpage.h"
|
|
#include "ui/gweditaccountwidget.h"
|
|
|
|
typedef KGenericFactory<GroupWiseProtocol> GroupWiseProtocolFactory;
|
|
K_EXPORT_COMPONENT_FACTORY( kopete_groupwise, GroupWiseProtocolFactory( "kopete_groupwise" ) )
|
|
|
|
GroupWiseProtocol *GroupWiseProtocol::s_protocol = 0L;
|
|
|
|
GroupWiseProtocol::GroupWiseProtocol( TQObject* parent, const char *name, const TQStringList &/*args*/ )
|
|
: Kopete::Protocol( GroupWiseProtocolFactory::instance(), parent, name ),
|
|
/* initialise Kopete::OnlineStatus that should be user selectable in the user interface */
|
|
groupwiseOffline ( Kopete::OnlineStatus::Offline, 0, this, GroupWise::Offline, TQString(),
|
|
i18n( "Offline" ), i18n( "O&ffline" ), Kopete::OnlineStatusManager::Offline ),
|
|
groupwiseAvailable ( Kopete::OnlineStatus::Online, 25, this, GroupWise::Available, TQString(),
|
|
i18n( "Online" ), i18n( "&Online" ), Kopete::OnlineStatusManager::Online ),
|
|
groupwiseBusy ( Kopete::OnlineStatus::Away, 18, this, GroupWise::Busy, "contact_busy_overlay",
|
|
i18n( "Busy" ), i18n( "&Busy" ), Kopete::OnlineStatusManager::Busy, Kopete::OnlineStatusManager::HasAwayMessage ),
|
|
groupwiseAway ( Kopete::OnlineStatus::Away, 20, this, GroupWise::Away, "contact_away_overlay",
|
|
i18n( "Away" ), i18n( "&Away" ), Kopete::OnlineStatusManager::Away, Kopete::OnlineStatusManager::HasAwayMessage ),
|
|
groupwiseAwayIdle ( Kopete::OnlineStatus::Away, 15, this, GroupWise::AwayIdle, "contact_away_overlay",
|
|
i18n( "Idle" ), "FIXME: Make groupwiseAwayIdle unselectable", Kopete::OnlineStatusManager::Idle,
|
|
Kopete::OnlineStatusManager::HideFromMenu ),
|
|
groupwiseAppearOffline( Kopete::OnlineStatus::Invisible, 2, this, 98, "contact_invisible_overlay",
|
|
i18n( "Appear Offline" ), i18n( "A&ppear Offline" ), Kopete::OnlineStatusManager::Invisible ),
|
|
/* initialise Kopete::OnlineStatus used by the protocol, but that are not user selectable */
|
|
groupwiseUnknown ( Kopete::OnlineStatus::Unknown, 25, this, GroupWise::Unknown, "status_unknown",
|
|
i18n( "Unknown" ) ),
|
|
groupwiseInvalid ( Kopete::OnlineStatus::Unknown, 25, this, GroupWise::Invalid, "status_unknown",
|
|
i18n( "Invalid tqStatus" ) ),
|
|
groupwiseConnecting ( Kopete::OnlineStatus::Connecting, 25, this, 99, "groupwise_connecting",
|
|
i18n( "Connecting" ) ),
|
|
propGivenName( Kopete::Global::Properties::self()->firstName() ),
|
|
propLastName( Kopete::Global::Properties::self()->lastName() ),
|
|
propFullName( Kopete::Global::Properties::self()->fullName() ),
|
|
propAwayMessage( Kopete::Global::Properties::self()->awayMessage() ),
|
|
propAutoReply( "groupwiseAutoReply", i18n( "Auto Reply Message" ), TQString(), false, false ),
|
|
propCN( "groupwiseCommonName", i18n( "Common Name" ), TQString(), true, false ),
|
|
propPhoneWork( Kopete::Global::Properties::self()->workPhone() ),
|
|
propPhoneMobile( Kopete::Global::Properties::self()->privateMobilePhone() ),
|
|
propEmail( Kopete::Global::Properties::self()->emailAddress() )
|
|
{
|
|
// ^^ That is all member initialiser syntax, not broken indentation!
|
|
kdDebug( GROUPWISE_DEBUG_GLOBAL ) << k_funcinfo << endl;
|
|
|
|
s_protocol = this;
|
|
|
|
addAddressBookField( "messaging/groupwise", Kopete::Plugin::MakeIndexField );
|
|
}
|
|
|
|
GroupWiseProtocol::~GroupWiseProtocol()
|
|
{
|
|
}
|
|
|
|
Kopete::Contact *GroupWiseProtocol::deserializeContact(
|
|
Kopete::MetaContact *metaContact, const TQMap<TQString, TQString> &serializedData,
|
|
const TQMap<TQString, TQString> &/* addressBookData */)
|
|
{
|
|
TQString dn = serializedData[ "DN" ];
|
|
TQString accountId = serializedData[ "accountId" ];
|
|
TQString displayName = serializedData[ "displayName" ];
|
|
int objectId = serializedData[ "objectId" ].toInt();
|
|
int parentId = serializedData[ "parentId" ].toInt();
|
|
int sequence = serializedData[ "sequenceNumber" ].toInt();
|
|
|
|
TQDict<Kopete::Account> accounts = Kopete::AccountManager::self()->accounts( this );
|
|
|
|
Kopete::Account *account = accounts[ accountId ];
|
|
if ( !account )
|
|
{
|
|
kdDebug(GROUPWISE_DEBUG_GLOBAL) << "Account doesn't exist, skipping" << endl;
|
|
return 0;
|
|
}
|
|
|
|
// FIXME: creating a contact with a userId here
|
|
return new GroupWiseContact(account, dn, metaContact, objectId, parentId, sequence );
|
|
}
|
|
|
|
AddContactPage * GroupWiseProtocol::createAddContactWidget( TQWidget *parent, Kopete::Account * account )
|
|
{
|
|
kdDebug( GROUPWISE_DEBUG_GLOBAL ) << "Creating Add Contact Page" << endl;
|
|
return new GroupWiseAddContactPage( account, parent, "addcontactpage");
|
|
}
|
|
|
|
KopeteEditAccountWidget * GroupWiseProtocol::createEditAccountWidget( Kopete::Account *account, TQWidget *parent )
|
|
{
|
|
kdDebug(GROUPWISE_DEBUG_GLOBAL) << "Creating Edit Account Page" << endl;
|
|
return new GroupWiseEditAccountWidget( parent, account );
|
|
}
|
|
|
|
Kopete::Account *GroupWiseProtocol::createNewAccount( const TQString &accountId )
|
|
{
|
|
return new GroupWiseAccount( this, accountId );
|
|
}
|
|
|
|
GroupWiseProtocol *GroupWiseProtocol::protocol()
|
|
{
|
|
return s_protocol;
|
|
}
|
|
|
|
Kopete::OnlineStatus GroupWiseProtocol::gwStatusToKOS( const int gwInternal )
|
|
{
|
|
Kopete::OnlineStatus status;
|
|
switch ( gwInternal )
|
|
{
|
|
case GroupWise::Unknown:
|
|
status = groupwiseUnknown;
|
|
break;
|
|
case GroupWise::Offline:
|
|
status = groupwiseOffline;
|
|
break;
|
|
case GroupWise::Available:
|
|
status = groupwiseAvailable;
|
|
break;
|
|
case GroupWise::Busy:
|
|
status = groupwiseBusy;
|
|
break;
|
|
case GroupWise::Away:
|
|
status = groupwiseAway;
|
|
break;
|
|
case GroupWise::AwayIdle:
|
|
status = groupwiseAwayIdle;
|
|
break;
|
|
case GroupWise::Invalid:
|
|
status = groupwiseInvalid;
|
|
break;
|
|
default:
|
|
status = groupwiseInvalid;
|
|
kdWarning( GROUPWISE_DEBUG_GLOBAL ) << k_funcinfo << "Got unrecognised status value" << gwInternal << endl;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
TQString GroupWiseProtocol::rtfizeText( const TQString & plain )
|
|
{
|
|
// transcode a utf-8 encoded string into an rtf string
|
|
// iterate through the input string converting each char into the equivalent rtf
|
|
// of single-byte characters with first byte =< 0x7f (127), { } \ are escaped. \n are converted into \par , the rest are appended verbatim
|
|
// of multi-byte UTF-8 characters 2 to 6 bytes long (with first byte > 0x7f), these are recoded as 32 bit values, escaped as \u<val>? strings
|
|
|
|
// vanilla RTF "envelope" that doesn't say much but causes other clients to accept the message
|
|
TQString rtfTemplate = TQString::tqfromLatin1("{\\rtf1\\ansi\n"
|
|
"{\\fonttbl{\\f0\\fnil Unknown;}}\n"
|
|
"{\\colortbl ;\\red0\\green0\\blue0;}\n"
|
|
"\\uc1\\cf1\\f0\\fs18 %1\\par\n}");
|
|
TQString outputText; // output text
|
|
TQCString plainUtf8 = plain.utf8(); // encoded as UTF8, because that's what this encoding algorithm, taken from Gaim's Novell plugin
|
|
uint index = 0; // current char to transcode
|
|
while ( index < plainUtf8.length() )
|
|
{
|
|
TQ_UINT8 current = plainUtf8.data()[ index ];
|
|
if ( current <= 0x7F )
|
|
{
|
|
switch ( current )
|
|
{
|
|
case '{':
|
|
case '}':
|
|
case '\\':
|
|
outputText.append( TQString( "\\%1" ).tqarg( TQChar( current ) ) );
|
|
break;
|
|
case '\n':
|
|
outputText.append( "\\par " );
|
|
break;
|
|
default:
|
|
outputText.append( TQChar( current ) );
|
|
break;
|
|
}
|
|
++index;
|
|
}
|
|
else
|
|
{
|
|
TQ_UINT32 ucs4Char;
|
|
int bytesEncoded;
|
|
TQString escapedUnicodeChar;
|
|
if ( current <= 0xDF )
|
|
{
|
|
ucs4Char = (( plainUtf8.data()[ index ] & 0x001F) << 6) |
|
|
( plainUtf8.data()[ index+1 ] & 0x003F);
|
|
bytesEncoded = 2;
|
|
}
|
|
else if ( current <= 0xEF )
|
|
{
|
|
ucs4Char = (( plainUtf8.data()[ index ] & 0x000F) << 12) |
|
|
(( plainUtf8.data()[ index+1 ] & 0x003F) << 6) |
|
|
( plainUtf8.data()[ index+2 ] & 0x003F);
|
|
bytesEncoded = 3;
|
|
}
|
|
else if ( current <= 0xF7 )
|
|
{
|
|
ucs4Char = (( plainUtf8.data()[ index ] & 0x0007) << 18) |
|
|
(( plainUtf8.data()[ index+1 ] & 0x003F) << 12) |
|
|
(( plainUtf8.data()[ index+2 ] & 0x003F) << 6) |
|
|
( plainUtf8.data()[ index+3 ] & 0x003F);
|
|
bytesEncoded = 4;
|
|
}
|
|
else if ( current <= 0xFB )
|
|
{
|
|
ucs4Char = (( plainUtf8.data()[ index ] & 0x0003) << 24 ) |
|
|
(( plainUtf8.data()[ index+1 ] & 0x003F) << 18) |
|
|
(( plainUtf8.data()[ index+2 ] & 0x003F) << 12) |
|
|
(( plainUtf8.data()[ index+3 ] & 0x003F) << 6) |
|
|
( plainUtf8.data()[ index+4 ] & 0x003F);
|
|
bytesEncoded = 5;
|
|
}
|
|
else if ( current <= 0xFD )
|
|
{
|
|
ucs4Char = (( plainUtf8.data()[ index ] & 0x0001) << 30 ) |
|
|
(( plainUtf8.data()[ index+1 ] & 0x003F) << 24) |
|
|
(( plainUtf8.data()[ index+2 ] & 0x003F) << 18) |
|
|
(( plainUtf8.data()[ index+3 ] & 0x003F) << 12) |
|
|
(( plainUtf8.data()[ index+4 ] & 0x003F) << 6) |
|
|
( plainUtf8.data()[ index+5 ] & 0x003F);
|
|
bytesEncoded = 6;
|
|
}
|
|
else
|
|
{
|
|
kdDebug( GROUPWISE_DEBUG_GLOBAL ) << k_funcinfo << "bogus utf-8 lead byte: 0x" << TQTextStream::hex << current << endl;
|
|
ucs4Char = 0x003F;
|
|
bytesEncoded = 1;
|
|
}
|
|
index += bytesEncoded;
|
|
escapedUnicodeChar = TQString("\\u%1?").tqarg( ucs4Char );
|
|
kdDebug( GROUPWISE_DEBUG_GLOBAL ) << "tqunicode escaped char: " << escapedUnicodeChar << endl;
|
|
outputText.append( escapedUnicodeChar );
|
|
}
|
|
}
|
|
return rtfTemplate.tqarg( outputText );
|
|
}
|
|
|
|
TQString GroupWiseProtocol::dnToDotted( const TQString & dn )
|
|
{
|
|
TQRegExp rx("[a-zA-Z]*=(.*)$", false );
|
|
if( !dn.find( '=' ) ) // if it's not a DN, return it unprocessed
|
|
return dn;
|
|
|
|
// split the dn into elements
|
|
TQStringList elements = TQStringList::split( ',', dn );
|
|
// remove the key, keep the value
|
|
for ( TQStringList::Iterator it = elements.begin(); it != elements.end(); ++it )
|
|
{
|
|
if ( rx.search( *it ) != -1 )
|
|
*it = rx.cap( 1 );
|
|
}
|
|
TQString dotted = elements.join( "." );
|
|
// reassemble as dotted
|
|
|
|
return dotted;
|
|
}
|
|
#include "gwprotocol.moc"
|