2019-08-14 14:54:46 +00:00
/*
* Squawk messenger .
2019-08-21 09:35:07 +00:00
* Copyright ( C ) 2019 Yury Gubich < blue @ macaw . me >
2019-08-14 14:54:46 +00:00
*
* 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 3 of the License , or
* ( at your option ) any later version .
*
* This program 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 this program . If not , see < http : //www.gnu.org/licenses/>.
*/
2019-03-29 14:54:34 +00:00
# include "account.h"
2019-06-28 15:15:30 +00:00
# include <QXmppMessage.h>
2019-04-07 14:02:41 +00:00
# include <QDateTime>
2019-05-15 17:36:37 +00:00
# include <QTimer>
2019-03-29 14:54:34 +00:00
using namespace Core ;
2019-03-30 20:13:13 +00:00
Account : : Account ( const QString & p_login , const QString & p_server , const QString & p_password , const QString & p_name , QObject * parent ) :
2019-03-29 14:54:34 +00:00
QObject ( parent ) ,
2019-03-30 20:13:13 +00:00
name ( p_name ) ,
2019-04-13 20:38:20 +00:00
achiveQueries ( ) ,
2019-03-31 21:05:09 +00:00
client ( ) ,
2019-04-07 20:14:15 +00:00
config ( ) ,
presence ( ) ,
2019-04-03 21:23:51 +00:00
state ( Shared : : disconnected ) ,
2019-04-12 15:22:10 +00:00
groups ( ) ,
2019-04-13 20:38:20 +00:00
cm ( new QXmppCarbonManager ( ) ) ,
2019-04-17 20:08:56 +00:00
am ( new QXmppMamManager ( ) ) ,
2019-07-11 08:51:52 +00:00
mm ( new QXmppMucManager ( ) ) ,
bm ( new QXmppBookmarkManager ( ) ) ,
2019-11-02 20:50:25 +00:00
rm ( client . findExtension < QXmppRosterManager > ( ) ) ,
vm ( client . findExtension < QXmppVCardManager > ( ) ) ,
2019-05-24 14:46:34 +00:00
contacts ( ) ,
2019-08-21 09:35:07 +00:00
conferences ( ) ,
2019-05-24 14:46:34 +00:00
maxReconnectTimes ( 0 ) ,
2019-06-18 15:08:03 +00:00
reconnectTimes ( 0 ) ,
queuedContacts ( ) ,
2019-10-16 19:38:35 +00:00
outOfRosterContacts ( ) ,
avatarHash ( ) ,
avatarType ( ) ,
ownVCardRequestInProgress ( false )
2019-03-29 14:54:34 +00:00
{
2019-04-07 20:14:15 +00:00
config . setUser ( p_login ) ;
config . setDomain ( p_server ) ;
config . setPassword ( p_password ) ;
2019-06-12 17:18:18 +00:00
config . setAutoAcceptSubscriptions ( true ) ;
2019-04-07 20:14:15 +00:00
2019-10-15 19:25:40 +00:00
QObject : : connect ( & client , & QXmppClient : : connected , this , & Account : : onClientConnected ) ;
QObject : : connect ( & client , & QXmppClient : : disconnected , this , & Account : : onClientDisconnected ) ;
QObject : : connect ( & client , & QXmppClient : : presenceReceived , this , & Account : : onPresenceReceived ) ;
QObject : : connect ( & client , & QXmppClient : : messageReceived , this , & Account : : onMessageReceived ) ;
QObject : : connect ( & client , & QXmppClient : : error , this , & Account : : onClientError ) ;
2019-04-03 21:23:51 +00:00
2019-11-02 20:50:25 +00:00
QObject : : connect ( rm , & QXmppRosterManager : : rosterReceived , this , & Account : : onRosterReceived ) ;
QObject : : connect ( rm , & QXmppRosterManager : : itemAdded , this , & Account : : onRosterItemAdded ) ;
QObject : : connect ( rm , & QXmppRosterManager : : itemRemoved , this , & Account : : onRosterItemRemoved ) ;
QObject : : connect ( rm , & QXmppRosterManager : : itemChanged , this , & Account : : onRosterItemChanged ) ;
2019-10-15 19:25:40 +00:00
//QObject::connect(&rm, &QXmppRosterManager::presenceChanged, this, &Account::onRosterPresenceChanged);
2019-04-12 15:22:10 +00:00
client . addExtension ( cm ) ;
2019-10-15 19:25:40 +00:00
QObject : : connect ( cm , & QXmppCarbonManager : : messageReceived , this , & Account : : onCarbonMessageReceived ) ;
QObject : : connect ( cm , & QXmppCarbonManager : : messageSent , this , & Account : : onCarbonMessageSent ) ;
2019-04-13 20:38:20 +00:00
client . addExtension ( am ) ;
2019-10-15 19:25:40 +00:00
QObject : : connect ( am , & QXmppMamManager : : logMessage , this , & Account : : onMamLog ) ;
QObject : : connect ( am , & QXmppMamManager : : archivedMessageReceived , this , & Account : : onMamMessageReceived ) ;
QObject : : connect ( am , & QXmppMamManager : : resultsRecieved , this , & Account : : onMamResultsReceived ) ;
2019-07-11 08:51:52 +00:00
client . addExtension ( mm ) ;
2019-10-15 19:25:40 +00:00
QObject : : connect ( mm , & QXmppMucManager : : roomAdded , this , & Account : : onMucRoomAdded ) ;
2019-07-11 08:51:52 +00:00
client . addExtension ( bm ) ;
2019-10-15 19:25:40 +00:00
QObject : : connect ( bm , & QXmppBookmarkManager : : bookmarksReceived , this , & Account : : bookmarksReceived ) ;
2019-11-02 20:50:25 +00:00
QObject : : connect ( vm , & QXmppVCardManager : : vCardReceived , this , & Account : : onVCardReceived ) ;
2019-10-16 19:38:35 +00:00
//QObject::connect(&vm, &QXmppVCardManager::clientVCardReceived, this, &Account::onOwnVCardReceived); //for some reason it doesn't work, launching from common handler
QString path ( QStandardPaths : : writableLocation ( QStandardPaths : : CacheLocation ) ) ;
path + = " / " + name ;
QDir dir ( path ) ;
if ( ! dir . exists ( ) ) {
bool res = dir . mkpath ( path ) ;
if ( ! res ) {
qDebug ( ) < < " Couldn't create a cache directory for account " < < name ;
throw 22 ;
}
}
QFile * avatar = new QFile ( path + " /avatar.png " ) ;
QString type = " png " ;
if ( ! avatar - > exists ( ) ) {
delete avatar ;
avatar = new QFile ( path + " /avatar.jpg " ) ;
2019-10-24 09:42:38 +00:00
type = " jpg " ;
2019-10-16 19:38:35 +00:00
if ( ! avatar - > exists ( ) ) {
delete avatar ;
avatar = new QFile ( path + " /avatar.jpeg " ) ;
2019-10-24 09:42:38 +00:00
type = " jpeg " ;
2019-10-16 19:38:35 +00:00
if ( ! avatar - > exists ( ) ) {
delete avatar ;
avatar = new QFile ( path + " /avatar.gif " ) ;
2019-10-24 09:42:38 +00:00
type = " gif " ;
2019-10-16 19:38:35 +00:00
}
}
}
if ( avatar - > exists ( ) ) {
if ( avatar - > open ( QFile : : ReadOnly ) ) {
QCryptographicHash sha1 ( QCryptographicHash : : Sha1 ) ;
sha1 . addData ( avatar ) ;
avatarHash = sha1 . result ( ) ;
avatarType = type ;
}
}
if ( avatarType . size ( ) ! = 0 ) {
presence . setVCardUpdateType ( QXmppPresence : : VCardUpdateValidPhoto ) ;
presence . setPhotoHash ( avatarHash . toUtf8 ( ) ) ;
} else {
presence . setVCardUpdateType ( QXmppPresence : : VCardUpdateNotReady ) ;
}
2019-03-29 14:54:34 +00:00
}
Account : : ~ Account ( )
{
2019-04-19 09:12:12 +00:00
for ( std : : map < QString , Contact * > : : const_iterator itr = contacts . begin ( ) , end = contacts . end ( ) ; itr ! = end ; + + itr ) {
delete itr - > second ;
}
2019-07-11 08:51:52 +00:00
2019-08-21 09:35:07 +00:00
for ( std : : map < QString , Conference * > : : const_iterator itr = conferences . begin ( ) , end = conferences . end ( ) ; itr ! = end ; + + itr ) {
delete itr - > second ;
}
2019-07-11 08:51:52 +00:00
delete bm ;
delete mm ;
delete am ;
delete cm ;
2019-03-31 21:05:09 +00:00
}
2019-03-29 14:54:34 +00:00
2019-03-31 21:05:09 +00:00
Shared : : ConnectionState Core : : Account : : getState ( ) const
{
return state ;
2019-03-29 14:54:34 +00:00
}
2019-03-31 21:05:09 +00:00
void Core : : Account : : connect ( )
{
if ( state = = Shared : : disconnected ) {
2019-05-24 14:46:34 +00:00
reconnectTimes = maxReconnectTimes ;
2019-03-31 21:05:09 +00:00
state = Shared : : connecting ;
2019-05-24 14:46:34 +00:00
client . connectToServer ( config , presence ) ;
2019-03-31 21:05:09 +00:00
emit connectionStateChanged ( state ) ;
} else {
qDebug ( " An attempt to connect an account which is already connected, skipping " ) ;
}
}
void Core : : Account : : disconnect ( )
{
2019-05-24 14:46:34 +00:00
reconnectTimes = 0 ;
2019-03-31 21:05:09 +00:00
if ( state ! = Shared : : disconnected ) {
2019-09-03 20:28:58 +00:00
clearConferences ( ) ;
2019-04-02 21:58:43 +00:00
client . disconnectFromServer ( ) ;
2019-03-31 21:05:09 +00:00
state = Shared : : disconnected ;
emit connectionStateChanged ( state ) ;
}
}
void Core : : Account : : onClientConnected ( )
{
if ( state = = Shared : : connecting ) {
2019-05-24 14:46:34 +00:00
reconnectTimes = maxReconnectTimes ;
2019-03-31 21:05:09 +00:00
state = Shared : : connected ;
2019-04-12 15:22:10 +00:00
cm - > setCarbonsEnabled ( true ) ;
2019-03-31 21:05:09 +00:00
emit connectionStateChanged ( state ) ;
} else {
2019-05-24 14:46:34 +00:00
qDebug ( ) < < " Something weird had happened - xmpp client reported about successful connection but account wasn't in " < < state < < " state " ;
2019-03-31 21:05:09 +00:00
}
}
2019-04-02 15:46:18 +00:00
void Core : : Account : : onClientDisconnected ( )
2019-03-31 21:05:09 +00:00
{
2019-09-03 20:28:58 +00:00
clearConferences ( ) ;
2019-03-31 21:05:09 +00:00
if ( state ! = Shared : : disconnected ) {
2019-05-24 14:46:34 +00:00
if ( reconnectTimes > 0 ) {
2019-09-03 20:28:58 +00:00
qDebug ( ) < < " Account " < < name < < " is reconnecting for " < < reconnectTimes < < " more times " ;
2019-05-24 14:46:34 +00:00
- - reconnectTimes ;
state = Shared : : connecting ;
client . connectToServer ( config , presence ) ;
emit connectionStateChanged ( state ) ;
} else {
2019-09-03 20:28:58 +00:00
qDebug ( ) < < " Account " < < name < < " has been disconnected " ;
2019-05-24 14:46:34 +00:00
state = Shared : : disconnected ;
emit connectionStateChanged ( state ) ;
}
2019-03-31 21:05:09 +00:00
} else {
2019-04-03 18:15:36 +00:00
//qDebug("Something weird had happened - xmpp client reported about being disconnection but account was already in disconnected state");
2019-03-31 21:05:09 +00:00
}
}
2019-05-24 14:46:34 +00:00
void Core : : Account : : reconnect ( )
{
if ( state = = Shared : : connected ) {
+ + reconnectTimes ;
client . disconnectFromServer ( ) ;
} else {
qDebug ( ) < < " An attempt to reconnect account " < < getName ( ) < < " which was not connected " ;
}
}
2019-03-31 21:05:09 +00:00
QString Core : : Account : : getName ( ) const
{
return name ;
}
2019-04-02 21:58:43 +00:00
QString Core : : Account : : getLogin ( ) const
{
2019-04-07 20:14:15 +00:00
return config . user ( ) ;
2019-04-02 21:58:43 +00:00
}
2019-03-31 21:05:09 +00:00
2019-04-02 21:58:43 +00:00
QString Core : : Account : : getPassword ( ) const
{
2019-04-07 20:14:15 +00:00
return config . password ( ) ;
2019-04-02 21:58:43 +00:00
}
QString Core : : Account : : getServer ( ) const
{
2019-04-07 20:14:15 +00:00
return config . domain ( ) ;
2019-04-02 21:58:43 +00:00
}
2019-04-03 21:23:51 +00:00
void Core : : Account : : onRosterReceived ( )
{
2019-11-02 20:50:25 +00:00
vm - > requestClientVCard ( ) ; //TODO need to make sure server actually supports vCards
2019-10-16 19:38:35 +00:00
ownVCardRequestInProgress = true ;
2019-11-02 20:50:25 +00:00
QStringList bj = rm - > getRosterBareJids ( ) ;
2019-04-03 21:23:51 +00:00
for ( int i = 0 ; i < bj . size ( ) ; + + i ) {
const QString & jid = bj [ i ] ;
2019-04-06 10:14:32 +00:00
addedAccount ( jid ) ;
}
}
2019-05-24 14:46:34 +00:00
void Core : : Account : : setReconnectTimes ( unsigned int times )
{
maxReconnectTimes = times ;
if ( state = = Shared : : connected ) {
reconnectTimes = times ;
}
}
2019-04-06 10:14:32 +00:00
void Core : : Account : : onRosterItemAdded ( const QString & bareJid )
{
addedAccount ( bareJid ) ;
2019-06-18 15:08:03 +00:00
std : : map < QString , QString > : : const_iterator itr = queuedContacts . find ( bareJid ) ;
if ( itr ! = queuedContacts . end ( ) ) {
2019-11-02 20:50:25 +00:00
rm - > subscribe ( bareJid , itr - > second ) ;
2019-06-18 15:08:03 +00:00
queuedContacts . erase ( itr ) ;
}
2019-04-06 10:14:32 +00:00
}
void Core : : Account : : onRosterItemChanged ( const QString & bareJid )
{
2019-04-19 09:12:12 +00:00
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( bareJid ) ;
if ( itr = = contacts . end ( ) ) {
qDebug ( ) < < " An attempt to change non existing contact " < < bareJid < < " from account " < < name < < " , skipping " ;
return ;
}
Contact * contact = itr - > second ;
2019-11-02 20:50:25 +00:00
QXmppRosterIq : : Item re = rm - > getRosterEntry ( bareJid ) ;
2019-04-07 14:02:41 +00:00
2019-04-19 09:12:12 +00:00
Shared : : SubscriptionState state = castSubscriptionState ( re . subscriptionType ( ) ) ;
contact - > setGroups ( re . groups ( ) ) ;
contact - > setSubscriptionState ( state ) ;
2019-06-12 17:18:18 +00:00
contact - > setName ( re . name ( ) ) ;
2019-04-06 10:14:32 +00:00
}
void Core : : Account : : onRosterItemRemoved ( const QString & bareJid )
{
2019-04-19 09:12:12 +00:00
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( bareJid ) ;
if ( itr = = contacts . end ( ) ) {
qDebug ( ) < < " An attempt to remove non existing contact " < < bareJid < < " from account " < < name < < " , skipping " ;
return ;
2019-04-06 10:14:32 +00:00
}
2019-04-19 09:12:12 +00:00
Contact * contact = itr - > second ;
2019-06-15 15:29:15 +00:00
contacts . erase ( itr ) ;
2019-04-19 09:12:12 +00:00
QSet < QString > cGroups = contact - > getGroups ( ) ;
for ( QSet < QString > : : const_iterator itr = cGroups . begin ( ) , end = cGroups . end ( ) ; itr ! = end ; + + itr ) {
removeFromGroup ( bareJid , * itr ) ;
2019-04-06 10:14:32 +00:00
}
2019-04-19 09:12:12 +00:00
emit removeContact ( bareJid ) ;
2019-06-15 15:29:15 +00:00
contact - > deleteLater ( ) ;
2019-04-06 10:14:32 +00:00
}
void Core : : Account : : addedAccount ( const QString & jid )
{
2019-04-17 20:08:56 +00:00
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( jid ) ;
2019-11-02 20:50:25 +00:00
QXmppRosterIq : : Item re = rm - > getRosterEntry ( jid ) ;
2019-04-19 09:12:12 +00:00
Contact * contact ;
bool newContact = false ;
2019-04-17 20:08:56 +00:00
if ( itr = = contacts . end ( ) ) {
2019-04-19 09:12:12 +00:00
newContact = true ;
contact = new Contact ( jid , name ) ;
contacts . insert ( std : : make_pair ( jid , contact ) ) ;
2019-04-17 20:08:56 +00:00
2019-04-19 09:12:12 +00:00
} else {
contact = itr - > second ;
2019-04-17 20:08:56 +00:00
}
2019-04-06 10:14:32 +00:00
QSet < QString > gr = re . groups ( ) ;
2019-04-19 09:12:12 +00:00
Shared : : SubscriptionState state = castSubscriptionState ( re . subscriptionType ( ) ) ;
contact - > setGroups ( gr ) ;
contact - > setSubscriptionState ( state ) ;
contact - > setName ( re . name ( ) ) ;
2019-04-06 10:14:32 +00:00
2019-04-19 09:12:12 +00:00
if ( newContact ) {
QMap < QString , QVariant > cData ( {
{ " name " , re . name ( ) } ,
{ " state " , state }
} ) ;
2019-10-15 19:25:40 +00:00
if ( contact - > hasAvatar ( ) ) {
2019-10-16 19:38:35 +00:00
if ( ! contact - > isAvatarAutoGenerated ( ) ) {
cData . insert ( " avatarState " , static_cast < uint > ( Shared : : Avatar : : valid ) ) ;
2019-10-15 19:25:40 +00:00
} else {
2019-10-16 19:38:35 +00:00
cData . insert ( " avatarState " , static_cast < uint > ( Shared : : Avatar : : autocreated ) ) ;
2019-10-15 19:25:40 +00:00
}
cData . insert ( " avatarPath " , contact - > avatarPath ( ) ) ;
} else {
2019-10-16 19:38:35 +00:00
cData . insert ( " avatarState " , static_cast < uint > ( Shared : : Avatar : : empty ) ) ;
cData . insert ( " avatarPath " , " " ) ;
2019-10-22 15:13:56 +00:00
requestVCard ( jid ) ;
2019-10-15 19:25:40 +00:00
}
2019-04-19 09:12:12 +00:00
int grCount = 0 ;
for ( QSet < QString > : : const_iterator itr = gr . begin ( ) , end = gr . end ( ) ; itr ! = end ; + + itr ) {
const QString & groupName = * itr ;
addToGroup ( jid , groupName ) ;
emit addContact ( jid , groupName , cData ) ;
grCount + + ;
}
if ( grCount = = 0 ) {
emit addContact ( jid , " " , cData ) ;
}
2019-06-15 15:29:15 +00:00
handleNewContact ( contact ) ;
2019-04-03 21:23:51 +00:00
}
}
2019-08-21 09:35:07 +00:00
void Core : : Account : : handleNewRosterItem ( Core : : RosterItem * contact )
{
2019-10-15 19:25:40 +00:00
QObject : : connect ( contact , & RosterItem : : needHistory , this , & Account : : onContactNeedHistory ) ;
QObject : : connect ( contact , & RosterItem : : historyResponse , this , & Account : : onContactHistoryResponse ) ;
QObject : : connect ( contact , & RosterItem : : nameChanged , this , & Account : : onContactNameChanged ) ;
QObject : : connect ( contact , & RosterItem : : avatarChanged , this , & Account : : onContactAvatarChanged ) ;
2019-08-21 09:35:07 +00:00
}
2019-06-15 15:29:15 +00:00
void Core : : Account : : handleNewContact ( Core : : Contact * contact )
{
2019-08-21 09:35:07 +00:00
handleNewRosterItem ( contact ) ;
2019-10-15 19:25:40 +00:00
QObject : : connect ( contact , & Contact : : groupAdded , this , & Account : : onContactGroupAdded ) ;
QObject : : connect ( contact , & Contact : : groupRemoved , this , & Account : : onContactGroupRemoved ) ;
QObject : : connect ( contact , & Contact : : subscriptionStateChanged , this , & Account : : onContactSubscriptionStateChanged ) ;
2019-08-21 09:35:07 +00:00
}
void Core : : Account : : handleNewConference ( Core : : Conference * contact )
{
handleNewRosterItem ( contact ) ;
2019-10-15 19:25:40 +00:00
QObject : : connect ( contact , & Conference : : nickChanged , this , & Account : : onMucNickNameChanged ) ;
QObject : : connect ( contact , & Conference : : subjectChanged , this , & Account : : onMucSubjectChanged ) ;
QObject : : connect ( contact , & Conference : : joinedChanged , this , & Account : : onMucJoinedChanged ) ;
QObject : : connect ( contact , & Conference : : autoJoinChanged , this , & Account : : onMucAutoJoinChanged ) ;
QObject : : connect ( contact , & Conference : : addParticipant , this , & Account : : onMucAddParticipant ) ;
QObject : : connect ( contact , & Conference : : changeParticipant , this , & Account : : onMucChangeParticipant ) ;
QObject : : connect ( contact , & Conference : : removeParticipant , this , & Account : : onMucRemoveParticipant ) ;
2019-06-15 15:29:15 +00:00
}
2019-10-16 19:38:35 +00:00
void Core : : Account : : onPresenceReceived ( const QXmppPresence & p_presence )
2019-04-07 14:02:41 +00:00
{
2019-10-16 19:38:35 +00:00
QString id = p_presence . from ( ) ;
2019-04-07 14:02:41 +00:00
QStringList comps = id . split ( " / " ) ;
QString jid = comps . front ( ) ;
QString resource = comps . back ( ) ;
2019-04-07 20:14:15 +00:00
QString myJid = getLogin ( ) + " @ " + getServer ( ) ;
if ( jid = = myJid ) {
if ( resource = = getResource ( ) ) {
2019-10-16 19:38:35 +00:00
emit availabilityChanged ( p_presence . availableStatusType ( ) ) ;
2019-04-07 20:14:15 +00:00
} else {
2019-10-16 19:38:35 +00:00
if ( ! ownVCardRequestInProgress ) {
switch ( p_presence . vCardUpdateType ( ) ) {
case QXmppPresence : : VCardUpdateNone : //this presence has nothing to do with photo
break ;
case QXmppPresence : : VCardUpdateNotReady : //let's say the photo didn't change here
break ;
case QXmppPresence : : VCardUpdateNoPhoto : //there is no photo, need to drop if any
if ( avatarType . size ( ) > 0 ) {
2019-11-02 20:50:25 +00:00
vm - > requestClientVCard ( ) ;
2019-10-16 19:38:35 +00:00
ownVCardRequestInProgress = true ;
}
break ;
case QXmppPresence : : VCardUpdateValidPhoto : //there is a photo, need to load
if ( avatarHash ! = p_presence . photoHash ( ) ) {
2019-11-02 20:50:25 +00:00
vm - > requestClientVCard ( ) ;
2019-10-16 19:38:35 +00:00
ownVCardRequestInProgress = true ;
}
break ;
}
}
2019-04-07 20:14:15 +00:00
}
2019-10-15 19:25:40 +00:00
} else {
if ( pendingVCardRequests . find ( jid ) = = pendingVCardRequests . end ( ) ) {
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( jid ) ;
if ( itr ! = contacts . end ( ) ) {
Contact * cnt = itr - > second ;
2019-10-16 19:38:35 +00:00
switch ( p_presence . vCardUpdateType ( ) ) {
2019-10-15 19:25:40 +00:00
case QXmppPresence : : VCardUpdateNone : //this presence has nothing to do with photo
break ;
case QXmppPresence : : VCardUpdateNotReady : //let's say the photo didn't change here
break ;
case QXmppPresence : : VCardUpdateNoPhoto : //there is no photo, need to drop if any
if ( ! cnt - > hasAvatar ( ) | | ( cnt - > hasAvatar ( ) & & ! cnt - > isAvatarAutoGenerated ( ) ) ) {
cnt - > setAutoGeneratedAvatar ( ) ;
}
break ;
case QXmppPresence : : VCardUpdateValidPhoto : //there is a photo, need to load
if ( cnt - > hasAvatar ( ) ) {
if ( cnt - > isAvatarAutoGenerated ( ) ) {
2019-10-22 15:13:56 +00:00
requestVCard ( jid ) ;
2019-10-15 19:25:40 +00:00
} else {
2019-10-16 19:38:35 +00:00
if ( cnt - > avatarHash ( ) ! = p_presence . photoHash ( ) ) {
2019-10-22 15:13:56 +00:00
requestVCard ( jid ) ;
2019-10-15 19:25:40 +00:00
}
}
} else {
2019-10-22 15:13:56 +00:00
requestVCard ( jid ) ;
2019-10-15 19:25:40 +00:00
}
break ;
}
}
}
2019-04-07 20:14:15 +00:00
}
2019-10-16 19:38:35 +00:00
switch ( p_presence . type ( ) ) {
2019-04-07 14:02:41 +00:00
case QXmppPresence : : Error :
2019-10-16 19:38:35 +00:00
qDebug ( ) < < " An error reported by presence from " < < id < < p_presence . error ( ) . text ( ) ;
2019-04-07 14:02:41 +00:00
break ;
case QXmppPresence : : Available : {
2019-10-16 19:38:35 +00:00
QDateTime lastInteraction = p_presence . lastUserInteraction ( ) ;
2019-04-07 14:02:41 +00:00
if ( ! lastInteraction . isValid ( ) ) {
lastInteraction = QDateTime : : currentDateTime ( ) ;
}
emit addPresence ( jid , resource , {
{ " lastActivity " , lastInteraction } ,
2019-10-16 19:38:35 +00:00
{ " availability " , p_presence . availableStatusType ( ) } , //TODO check and handle invisible
{ " status " , p_presence . statusText ( ) }
2019-04-07 14:02:41 +00:00
} ) ;
}
break ;
case QXmppPresence : : Unavailable :
emit removePresence ( jid , resource ) ;
break ;
case QXmppPresence : : Subscribe :
qDebug ( " xmpp presence \" subscribe \" received, do not yet know what to do, skipping " ) ;
case QXmppPresence : : Subscribed :
qDebug ( " xmpp presence \" subscribed \" received, do not yet know what to do, skipping " ) ;
case QXmppPresence : : Unsubscribe :
qDebug ( " xmpp presence \" unsubscribe \" received, do not yet know what to do, skipping " ) ;
case QXmppPresence : : Unsubscribed :
qDebug ( " xmpp presence \" unsubscribed \" received, do not yet know what to do, skipping " ) ;
case QXmppPresence : : Probe :
qDebug ( " xmpp presence \" probe \" received, do not yet know what to do, skipping " ) ;
break ;
}
}
void Core : : Account : : onRosterPresenceChanged ( const QString & bareJid , const QString & resource )
{
//not used for now;
qDebug ( ) < < " presence changed for " < < bareJid < < " resource " < < resource ;
2019-11-02 20:50:25 +00:00
const QXmppPresence & presence = rm - > getPresence ( bareJid , resource ) ;
2019-04-07 14:02:41 +00:00
}
2019-04-07 20:14:15 +00:00
void Core : : Account : : setLogin ( const QString & p_login )
{
config . setUser ( p_login ) ;
}
void Core : : Account : : setName ( const QString & p_name )
{
name = p_name ;
}
void Core : : Account : : setPassword ( const QString & p_password )
{
config . setPassword ( p_password ) ;
}
void Core : : Account : : setServer ( const QString & p_server )
{
config . setDomain ( p_server ) ;
}
Shared : : Availability Core : : Account : : getAvailability ( ) const
{
2019-06-22 20:37:22 +00:00
if ( state = = Shared : : connected ) {
QXmppPresence : : AvailableStatusType pres = presence . availableStatusType ( ) ;
return static_cast < Shared : : Availability > ( pres ) ; //they are compatible;
} else {
return Shared : : offline ;
}
2019-04-07 20:14:15 +00:00
}
void Core : : Account : : setAvailability ( Shared : : Availability avail )
{
QXmppPresence : : AvailableStatusType pres = static_cast < QXmppPresence : : AvailableStatusType > ( avail ) ;
presence . setAvailableStatusType ( pres ) ;
if ( state ! = Shared : : disconnected ) { //TODO not sure how to do here - changing state may cause connection or disconnection
client . setClientPresence ( presence ) ;
}
}
QString Core : : Account : : getResource ( ) const
{
return config . resource ( ) ;
}
void Core : : Account : : setResource ( const QString & p_resource )
{
config . setResource ( p_resource ) ;
}
2019-04-09 15:04:08 +00:00
2019-04-09 22:01:25 +00:00
void Core : : Account : : onMessageReceived ( const QXmppMessage & msg )
2019-04-09 15:04:08 +00:00
{
2019-04-09 22:01:25 +00:00
bool handled = false ;
switch ( msg . type ( ) ) {
case QXmppMessage : : Normal :
qDebug ( ) < < " received a message with type \" Normal \" , not sure what to do with it now, skipping " ;
break ;
2019-04-12 15:22:10 +00:00
case QXmppMessage : : Chat :
handled = handleChatMessage ( msg ) ;
2019-04-09 22:01:25 +00:00
break ;
case QXmppMessage : : GroupChat :
2019-08-28 11:40:55 +00:00
handled = handleGroupMessage ( msg ) ;
2019-04-09 22:01:25 +00:00
break ;
case QXmppMessage : : Error :
qDebug ( ) < < " received a message with type \" Error \" , not sure what to do with it now, skipping " ;
break ;
case QXmppMessage : : Headline :
qDebug ( ) < < " received a message with type \" Headline \" , not sure what to do with it now, skipping " ;
break ;
}
if ( ! handled ) {
2019-06-28 15:15:30 +00:00
logMessage ( msg ) ;
}
}
void Core : : Account : : logMessage ( const QXmppMessage & msg , const QString & reason )
{
qDebug ( ) < < reason ;
2019-04-12 15:22:10 +00:00
qDebug ( ) < < " - from: " < < msg . from ( ) ;
qDebug ( ) < < " - to: " < < msg . to ( ) ;
2019-04-09 22:01:25 +00:00
qDebug ( ) < < " - body: " < < msg . body ( ) ;
qDebug ( ) < < " - type: " < < msg . type ( ) ;
qDebug ( ) < < " - state: " < < msg . state ( ) ;
qDebug ( ) < < " - stamp: " < < msg . stamp ( ) ;
qDebug ( ) < < " - id: " < < msg . id ( ) ;
2019-09-10 14:33:39 +00:00
qDebug ( ) < < " - outOfBandUrl: " < < msg . outOfBandUrl ( ) ;
2019-04-09 22:01:25 +00:00
qDebug ( ) < < " - isAttentionRequested: " < < msg . isAttentionRequested ( ) ;
qDebug ( ) < < " - isReceiptRequested: " < < msg . isReceiptRequested ( ) ;
qDebug ( ) < < " - receiptId: " < < msg . receiptId ( ) ;
qDebug ( ) < < " - subject: " < < msg . subject ( ) ;
qDebug ( ) < < " - thread: " < < msg . thread ( ) ;
qDebug ( ) < < " - isMarkable: " < < msg . isMarkable ( ) ;
qDebug ( ) < < " ============================== " ;
2019-04-09 15:04:08 +00:00
}
2019-04-09 22:01:25 +00:00
2019-06-28 15:15:30 +00:00
2019-04-09 22:01:25 +00:00
QString Core : : Account : : getFullJid ( ) const
{
return getLogin ( ) + " @ " + getServer ( ) + " / " + getResource ( ) ;
}
2019-04-11 14:58:59 +00:00
void Core : : Account : : sendMessage ( const Shared : : Message & data )
2019-04-10 20:53:42 +00:00
{
if ( state = = Shared : : connected ) {
2019-04-12 15:22:10 +00:00
QXmppMessage msg ( data . getFrom ( ) , data . getTo ( ) , data . getBody ( ) , data . getThread ( ) ) ;
msg . setId ( data . getId ( ) ) ;
msg . setType ( static_cast < QXmppMessage : : Type > ( data . getType ( ) ) ) ; //it is safe here, my type is compatible
2019-05-15 17:36:37 +00:00
2019-08-28 11:40:55 +00:00
RosterItem * ri = 0 ;
2019-05-15 17:36:37 +00:00
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( data . getPenPalJid ( ) ) ;
2019-08-28 11:40:55 +00:00
if ( itr ! = contacts . end ( ) ) {
ri = itr - > second ;
} else {
std : : map < QString , Conference * > : : const_iterator ritr = conferences . find ( data . getPenPalJid ( ) ) ;
if ( ritr ! = conferences . end ( ) ) {
ri = ritr - > second ;
}
}
2019-07-11 08:51:52 +00:00
2019-08-28 11:40:55 +00:00
if ( ri ! = 0 ) {
if ( ! ri - > isMuc ( ) ) {
ri - > appendMessageToArchive ( data ) ;
}
}
2019-05-15 17:36:37 +00:00
2019-04-12 15:22:10 +00:00
client . sendPacket ( msg ) ;
2019-08-28 11:40:55 +00:00
2019-04-10 20:53:42 +00:00
} else {
qDebug ( ) < < " An attempt to send message with not connected account " < < name < < " , skipping " ;
}
}
2019-04-12 15:22:10 +00:00
void Core : : Account : : onCarbonMessageReceived ( const QXmppMessage & msg )
{
2019-04-13 20:38:20 +00:00
handleChatMessage ( msg , false , true ) ;
2019-04-12 15:22:10 +00:00
}
void Core : : Account : : onCarbonMessageSent ( const QXmppMessage & msg )
{
2019-04-13 20:38:20 +00:00
handleChatMessage ( msg , true , true ) ;
2019-04-12 15:22:10 +00:00
}
2019-04-13 20:38:20 +00:00
bool Core : : Account : : handleChatMessage ( const QXmppMessage & msg , bool outgoing , bool forwarded , bool guessing )
2019-04-12 15:22:10 +00:00
{
2019-05-15 17:36:37 +00:00
const QString & body ( msg . body ( ) ) ;
2019-04-12 15:22:10 +00:00
if ( body . size ( ) ! = 0 ) {
2019-05-15 17:36:37 +00:00
const QString & id ( msg . id ( ) ) ;
2019-04-12 15:22:10 +00:00
Shared : : Message sMsg ( Shared : : Message : : chat ) ;
2019-05-15 17:36:37 +00:00
initializeMessage ( sMsg , msg , outgoing , forwarded , guessing ) ;
2019-06-15 15:29:15 +00:00
QString jid = sMsg . getPenPalJid ( ) ;
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( jid ) ;
Contact * cnt ;
if ( itr ! = contacts . end ( ) ) {
cnt = itr - > second ;
} else {
cnt = new Contact ( jid , name ) ;
contacts . insert ( std : : make_pair ( jid , cnt ) ) ;
2019-06-18 15:08:03 +00:00
outOfRosterContacts . insert ( jid ) ;
2019-06-15 15:29:15 +00:00
cnt - > setSubscriptionState ( Shared : : unknown ) ;
emit addContact ( jid , " " , QMap < QString , QVariant > ( {
{ " state " , Shared : : unknown }
} ) ) ;
handleNewContact ( cnt ) ;
}
cnt - > appendMessageToArchive ( sMsg ) ;
2019-05-15 17:36:37 +00:00
2019-04-12 15:22:10 +00:00
emit message ( sMsg ) ;
if ( ! forwarded & & ! outgoing ) {
if ( msg . isReceiptRequested ( ) & & id . size ( ) > 0 ) {
QXmppMessage receipt ( getFullJid ( ) , msg . from ( ) , " " ) ;
receipt . setReceiptId ( id ) ;
client . sendPacket ( receipt ) ;
}
}
return true ;
}
return false ;
}
2019-04-13 20:38:20 +00:00
2019-08-28 11:40:55 +00:00
bool Core : : Account : : handleGroupMessage ( const QXmppMessage & msg , bool outgoing , bool forwarded , bool guessing )
{
const QString & body ( msg . body ( ) ) ;
if ( body . size ( ) ! = 0 ) {
const QString & id ( msg . id ( ) ) ;
Shared : : Message sMsg ( Shared : : Message : : groupChat ) ;
initializeMessage ( sMsg , msg , outgoing , forwarded , guessing ) ;
QString jid = sMsg . getPenPalJid ( ) ;
std : : map < QString , Conference * > : : const_iterator itr = conferences . find ( jid ) ;
Conference * cnt ;
if ( itr ! = conferences . end ( ) ) {
cnt = itr - > second ;
} else {
return false ;
}
cnt - > appendMessageToArchive ( sMsg ) ;
2019-08-31 20:50:05 +00:00
QDateTime fiveMinsAgo = QDateTime : : currentDateTime ( ) . addSecs ( - 300 ) ;
if ( sMsg . getTime ( ) > fiveMinsAgo ) { //otherwise it's considered a delayed delivery, most probably MUC history receipt
emit message ( sMsg ) ;
}
2019-08-28 11:40:55 +00:00
if ( ! forwarded & & ! outgoing ) {
if ( msg . isReceiptRequested ( ) & & id . size ( ) > 0 ) {
QXmppMessage receipt ( getFullJid ( ) , msg . from ( ) , " " ) ;
receipt . setReceiptId ( id ) ;
client . sendPacket ( receipt ) ;
}
}
return true ;
}
return false ;
}
2019-05-15 17:36:37 +00:00
void Core : : Account : : initializeMessage ( Shared : : Message & target , const QXmppMessage & source , bool outgoing , bool forwarded , bool guessing ) const
{
const QDateTime & time ( source . stamp ( ) ) ;
target . setId ( source . id ( ) ) ;
target . setFrom ( source . from ( ) ) ;
target . setTo ( source . to ( ) ) ;
target . setBody ( source . body ( ) ) ;
target . setForwarded ( forwarded ) ;
2019-09-10 14:33:39 +00:00
target . setOutOfBandUrl ( source . outOfBandUrl ( ) ) ;
2019-05-15 17:36:37 +00:00
if ( guessing ) {
if ( target . getFromJid ( ) = = getLogin ( ) + " @ " + getServer ( ) ) {
outgoing = true ;
} else {
outgoing = false ;
}
}
target . setOutgoing ( outgoing ) ;
if ( time . isValid ( ) ) {
target . setTime ( time ) ;
} else {
target . setCurrentTime ( ) ;
}
}
void Core : : Account : : onMamMessageReceived ( const QString & queryId , const QXmppMessage & msg )
2019-04-13 20:38:20 +00:00
{
2019-09-10 14:33:39 +00:00
if ( msg . id ( ) . size ( ) > 0 & & ( msg . body ( ) . size ( ) > 0 | | msg . outOfBandUrl ( ) . size ( ) > 0 ) ) {
2019-08-21 09:35:07 +00:00
std : : map < QString , QString > : : const_iterator itr = achiveQueries . find ( queryId ) ;
QString jid = itr - > second ;
RosterItem * item = 0 ;
std : : map < QString , Contact * > : : const_iterator citr = contacts . find ( jid ) ;
if ( citr ! = contacts . end ( ) ) {
item = citr - > second ;
} else {
std : : map < QString , Conference * > : : const_iterator coitr = conferences . find ( jid ) ;
if ( coitr ! = conferences . end ( ) ) {
item = coitr - > second ;
}
2019-05-15 17:36:37 +00:00
}
2019-08-21 09:35:07 +00:00
Shared : : Message sMsg ( Shared : : Message : : chat ) ;
initializeMessage ( sMsg , msg , false , true , true ) ;
item - > addMessageToArchive ( sMsg ) ;
}
2019-05-15 17:36:37 +00:00
//handleChatMessage(msg, false, true, true);
2019-04-13 20:38:20 +00:00
}
2019-04-21 19:17:04 +00:00
void Core : : Account : : requestArchive ( const QString & jid , int count , const QString & before )
2019-04-13 20:38:20 +00:00
{
2019-05-15 17:36:37 +00:00
qDebug ( ) < < " An archive request for " < < jid < < " , before " < < before ;
2019-08-21 09:35:07 +00:00
RosterItem * contact = 0 ;
2019-04-21 19:17:04 +00:00
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( jid ) ;
2019-08-28 11:40:55 +00:00
bool gr = false ;
2019-08-21 09:35:07 +00:00
if ( itr ! = contacts . end ( ) ) {
contact = itr - > second ;
} else {
std : : map < QString , Conference * > : : const_iterator citr = conferences . find ( jid ) ;
if ( citr ! = conferences . end ( ) ) {
contact = citr - > second ;
2019-08-28 11:40:55 +00:00
gr = true ;
2019-08-21 09:35:07 +00:00
}
}
if ( contact = = 0 ) {
2019-04-21 19:17:04 +00:00
qDebug ( ) < < " An attempt to request archive for " < < jid < < " in account " < < name < < " , but the contact with such id wasn't found, skipping " ;
2019-08-28 11:40:55 +00:00
emit responseArchive ( contact - > jid , std : : list < Shared : : Message > ( ) ) ;
2019-04-21 19:17:04 +00:00
return ;
}
2019-08-14 14:54:46 +00:00
if ( contact - > getArchiveState ( ) = = RosterItem : : empty & & before . size ( ) = = 0 ) {
2019-05-15 17:36:37 +00:00
QXmppMessage msg ( getFullJid ( ) , jid , " " , " " ) ;
QString last = Shared : : generateUUID ( ) ;
msg . setId ( last ) ;
2019-08-28 11:40:55 +00:00
if ( gr ) {
msg . setType ( QXmppMessage : : GroupChat ) ;
} else {
msg . setType ( QXmppMessage : : Chat ) ;
}
2019-05-15 17:36:37 +00:00
msg . setState ( QXmppMessage : : Active ) ;
client . sendPacket ( msg ) ;
QTimer * timer = new QTimer ;
QObject : : connect ( timer , & QTimer : : timeout , [ timer , contact , count , last ] ( ) {
contact - > requestFromEmpty ( count , last ) ;
timer - > deleteLater ( ) ;
} ) ;
timer - > setSingleShot ( true ) ;
timer - > start ( 1000 ) ;
} else {
contact - > requestHistory ( count , before ) ;
2019-04-21 19:17:04 +00:00
}
2019-05-15 17:36:37 +00:00
}
2019-07-01 06:31:38 +00:00
void Core : : Account : : onContactNeedHistory ( const QString & before , const QString & after , const QDateTime & at )
2019-05-15 17:36:37 +00:00
{
2019-08-14 14:54:46 +00:00
RosterItem * contact = static_cast < RosterItem * > ( sender ( ) ) ;
2019-08-28 11:40:55 +00:00
QString to = " " ;
2019-04-13 20:38:20 +00:00
QXmppResultSetQuery query ;
query . setMax ( 100 ) ;
2019-05-15 17:36:37 +00:00
if ( before . size ( ) > 0 ) {
query . setBefore ( before ) ;
}
2019-07-01 06:31:38 +00:00
QDateTime start ;
if ( after . size ( ) > 0 ) { //there is some strange behavior of ejabberd server returning empty result set
if ( at . isValid ( ) ) { //there can be some useful information about it here https://github.com/processone/ejabberd/issues/2924
start = at ;
} else {
query . setAfter ( after ) ;
}
2019-05-15 17:36:37 +00:00
}
2019-06-28 15:15:30 +00:00
qDebug ( ) < < " Remote query from " < < after < < " , to " < < before ;
2019-05-15 17:36:37 +00:00
2019-08-28 11:40:55 +00:00
QString q = am - > retrieveArchivedMessages ( to , " " , contact - > jid , start , QDateTime ( ) , query ) ;
2019-05-15 17:36:37 +00:00
achiveQueries . insert ( std : : make_pair ( q , contact - > jid ) ) ;
2019-04-13 20:38:20 +00:00
}
2019-05-15 17:36:37 +00:00
2019-04-13 20:38:20 +00:00
void Core : : Account : : onMamResultsReceived ( const QString & queryId , const QXmppResultSetReply & resultSetReply , bool complete )
{
std : : map < QString , QString > : : const_iterator itr = achiveQueries . find ( queryId ) ;
QString jid = itr - > second ;
2019-08-21 09:35:07 +00:00
RosterItem * ri = 0 ;
2019-06-28 15:15:30 +00:00
2019-04-13 20:38:20 +00:00
achiveQueries . erase ( itr ) ;
2019-05-15 17:36:37 +00:00
std : : map < QString , Contact * > : : const_iterator citr = contacts . find ( jid ) ;
if ( citr ! = contacts . end ( ) ) {
2019-08-21 09:35:07 +00:00
ri = citr - > second ;
} else {
std : : map < QString , Conference * > : : const_iterator coitr = conferences . find ( jid ) ;
if ( coitr ! = conferences . end ( ) ) {
ri = coitr - > second ;
}
}
if ( ri ! = 0 ) {
2019-05-15 17:36:37 +00:00
qDebug ( ) < < " Flushing messages for " < < jid ;
2019-08-21 09:35:07 +00:00
ri - > flushMessagesToArchive ( complete , resultSetReply . first ( ) , resultSetReply . last ( ) ) ;
2019-04-13 20:38:20 +00:00
}
}
2019-04-19 09:12:12 +00:00
2019-06-28 15:15:30 +00:00
void Core : : Account : : onMamLog ( QXmppLogger : : MessageType type , const QString & msg )
{
qDebug ( ) < < " MAM MESSAGE LOG:: " ;
qDebug ( ) < < msg ;
}
2019-04-19 09:12:12 +00:00
void Core : : Account : : onContactGroupAdded ( const QString & group )
{
Contact * contact = static_cast < Contact * > ( sender ( ) ) ;
if ( contact - > groupsCount ( ) = = 1 ) {
// not sure i need to handle it here, the situation with grouped and ungrouped contacts handled on the client anyway
}
QMap < QString , QVariant > cData ( {
{ " name " , contact - > getName ( ) } ,
{ " state " , contact - > getSubscriptionState ( ) }
} ) ;
addToGroup ( contact - > jid , group ) ;
emit addContact ( contact - > jid , group , cData ) ;
}
void Core : : Account : : onContactGroupRemoved ( const QString & group )
{
Contact * contact = static_cast < Contact * > ( sender ( ) ) ;
if ( contact - > groupsCount ( ) = = 0 ) {
// not sure i need to handle it here, the situation with grouped and ungrouped contacts handled on the client anyway
}
emit removeContact ( contact - > jid , group ) ;
removeFromGroup ( contact - > jid , group ) ;
}
void Core : : Account : : onContactNameChanged ( const QString & cname )
{
Contact * contact = static_cast < Contact * > ( sender ( ) ) ;
QMap < QString , QVariant > cData ( {
{ " name " , cname } ,
} ) ;
emit changeContact ( contact - > jid , cData ) ;
}
void Core : : Account : : onContactSubscriptionStateChanged ( Shared : : SubscriptionState cstate )
{
Contact * contact = static_cast < Contact * > ( sender ( ) ) ;
QMap < QString , QVariant > cData ( {
{ " state " , cstate } ,
} ) ;
emit changeContact ( contact - > jid , cData ) ;
}
void Core : : Account : : addToGroup ( const QString & jid , const QString & group )
{
std : : map < QString , std : : set < QString > > : : iterator gItr = groups . find ( group ) ;
if ( gItr = = groups . end ( ) ) {
gItr = groups . insert ( std : : make_pair ( group , std : : set < QString > ( ) ) ) . first ;
emit addGroup ( group ) ;
}
gItr - > second . insert ( jid ) ;
}
void Core : : Account : : removeFromGroup ( const QString & jid , const QString & group )
{
QSet < QString > toRemove ;
std : : map < QString , std : : set < QString > > : : iterator itr = groups . find ( group ) ;
if ( itr = = groups . end ( ) ) {
qDebug ( ) < < " An attempt to remove contact " < < jid < < " of account " < < name < < " from non existing group " < < group < < " , skipping " ;
return ;
}
std : : set < QString > contacts = itr - > second ;
std : : set < QString > : : const_iterator cItr = contacts . find ( jid ) ;
if ( cItr ! = contacts . end ( ) ) {
contacts . erase ( cItr ) ;
if ( contacts . size ( ) = = 0 ) {
emit removeGroup ( group ) ;
groups . erase ( group ) ;
}
}
}
Shared : : SubscriptionState Core : : Account : : castSubscriptionState ( QXmppRosterIq : : Item : : SubscriptionType qs ) const
{
Shared : : SubscriptionState state ;
if ( qs = = QXmppRosterIq : : Item : : NotSet ) {
state = Shared : : unknown ;
} else {
state = static_cast < Shared : : SubscriptionState > ( qs ) ;
}
return state ;
}
2019-05-15 17:36:37 +00:00
void Core : : Account : : onContactHistoryResponse ( const std : : list < Shared : : Message > & list )
{
2019-08-14 14:54:46 +00:00
RosterItem * contact = static_cast < RosterItem * > ( sender ( ) ) ;
2019-05-15 17:36:37 +00:00
qDebug ( ) < < " Collected history for contact " < < contact - > jid < < list . size ( ) < < " elements " ;
emit responseArchive ( contact - > jid , list ) ;
}
void Core : : Account : : onClientError ( QXmppClient : : Error err )
{
2019-05-24 14:46:34 +00:00
QString errorText ;
QString errorType ;
2019-05-15 17:36:37 +00:00
switch ( err ) {
case QXmppClient : : SocketError :
2019-05-24 14:46:34 +00:00
errorText = client . socketErrorString ( ) ;
errorType = " Client socket error " ;
2019-05-15 17:36:37 +00:00
break ;
2019-05-24 14:46:34 +00:00
case QXmppClient : : XmppStreamError : {
QXmppStanza : : Error : : Condition cnd = client . xmppStreamError ( ) ;
switch ( cnd ) {
case QXmppStanza : : Error : : BadRequest :
errorText = " Bad request " ;
break ;
case QXmppStanza : : Error : : Conflict :
errorText = " Conflict " ;
break ;
case QXmppStanza : : Error : : FeatureNotImplemented :
errorText = " Feature is not implemented " ;
break ;
case QXmppStanza : : Error : : Forbidden :
errorText = " Forbidden " ;
break ;
case QXmppStanza : : Error : : Gone :
errorText = " Gone " ;
break ;
case QXmppStanza : : Error : : InternalServerError :
errorText = " Internal server error " ;
break ;
case QXmppStanza : : Error : : ItemNotFound :
errorText = " Item was not found " ;
break ;
case QXmppStanza : : Error : : JidMalformed :
errorText = " Malformed JID " ;
break ;
case QXmppStanza : : Error : : NotAcceptable :
errorText = " Not acceptable " ;
break ;
case QXmppStanza : : Error : : NotAllowed :
errorText = " Not allowed " ;
break ;
case QXmppStanza : : Error : : NotAuthorized :
errorText = " Authentication error " ;
break ;
case QXmppStanza : : Error : : PaymentRequired :
errorText = " Payment is required " ;
break ;
case QXmppStanza : : Error : : RecipientUnavailable :
errorText = " Recipient is unavailable " ;
break ;
case QXmppStanza : : Error : : Redirect :
errorText = " Redirected " ;
break ;
case QXmppStanza : : Error : : RegistrationRequired :
errorText = " Registration is required " ;
break ;
case QXmppStanza : : Error : : RemoteServerNotFound :
errorText = " Remote server was not found " ;
break ;
case QXmppStanza : : Error : : RemoteServerTimeout :
errorText = " Remote server timeout " ;
break ;
case QXmppStanza : : Error : : ResourceConstraint :
errorText = " Resource constraint " ;
break ;
case QXmppStanza : : Error : : ServiceUnavailable :
errorText = " Redirected " ;
break ;
case QXmppStanza : : Error : : SubscriptionRequired :
errorText = " Subscription is required " ;
break ;
case QXmppStanza : : Error : : UndefinedCondition :
errorText = " Undefined condition " ;
break ;
case QXmppStanza : : Error : : UnexpectedRequest :
errorText = " Unexpected request " ;
break ;
}
errorType = " Client stream error " ;
}
2019-05-15 17:36:37 +00:00
break ;
case QXmppClient : : KeepAliveError :
2019-05-24 14:46:34 +00:00
errorText = " Client keep alive error " ;
2019-05-15 17:36:37 +00:00
break ;
}
2019-05-24 14:46:34 +00:00
qDebug ( ) < < errorType < < errorText ;
emit error ( errorText ) ;
2019-05-15 17:36:37 +00:00
}
2019-06-12 17:18:18 +00:00
void Core : : Account : : subscribeToContact ( const QString & jid , const QString & reason )
{
if ( state = = Shared : : connected ) {
2019-11-02 20:50:25 +00:00
rm - > subscribe ( jid , reason ) ;
2019-06-12 17:18:18 +00:00
} else {
2019-06-14 16:36:04 +00:00
qDebug ( ) < < " An attempt to subscribe account " < < name < < " to contact " < < jid < < " but the account is not in the connected state, skipping " ;
2019-06-12 17:18:18 +00:00
}
}
void Core : : Account : : unsubscribeFromContact ( const QString & jid , const QString & reason )
{
if ( state = = Shared : : connected ) {
2019-11-02 20:50:25 +00:00
rm - > unsubscribe ( jid , reason ) ;
2019-06-12 17:18:18 +00:00
} else {
2019-06-14 16:36:04 +00:00
qDebug ( ) < < " An attempt to unsubscribe account " < < name < < " from contact " < < jid < < " but the account is not in the connected state, skipping " ;
2019-06-12 17:18:18 +00:00
}
}
2019-06-14 16:36:04 +00:00
void Core : : Account : : removeContactRequest ( const QString & jid )
{
if ( state = = Shared : : connected ) {
2019-06-18 15:08:03 +00:00
std : : set < QString > : : const_iterator itr = outOfRosterContacts . find ( jid ) ;
if ( itr ! = outOfRosterContacts . end ( ) ) {
outOfRosterContacts . erase ( itr ) ;
onRosterItemRemoved ( jid ) ;
} else {
2019-11-02 20:50:25 +00:00
rm - > removeItem ( jid ) ;
2019-06-18 15:08:03 +00:00
}
2019-06-14 16:36:04 +00:00
} else {
qDebug ( ) < < " An attempt to remove contact " < < jid < < " from account " < < name < < " but the account is not in the connected state, skipping " ;
}
}
void Core : : Account : : addContactRequest ( const QString & jid , const QString & name , const QSet < QString > & groups )
{
if ( state = = Shared : : connected ) {
2019-06-18 15:08:03 +00:00
std : : map < QString , QString > : : const_iterator itr = queuedContacts . find ( jid ) ;
if ( itr ! = queuedContacts . end ( ) ) {
qDebug ( ) < < " An attempt to add contact " < < jid < < " to account " < < name < < " but the account is already queued for adding, skipping " ;
} else {
queuedContacts . insert ( std : : make_pair ( jid , " " ) ) ; //TODO need to add reason here;
2019-11-02 20:50:25 +00:00
rm - > addItem ( jid , name , groups ) ;
2019-06-18 15:08:03 +00:00
}
2019-06-14 16:36:04 +00:00
} else {
qDebug ( ) < < " An attempt to add contact " < < jid < < " to account " < < name < < " but the account is not in the connected state, skipping " ;
}
}
2019-07-11 08:51:52 +00:00
void Core : : Account : : onMucRoomAdded ( QXmppMucRoom * room )
{
qDebug ( ) < < " room " < < room - > jid ( ) < < " added with name " < < room - > name ( ) < < " , account " < < getName ( ) < < " joined: " < < room - > isJoined ( ) ;
}
void Core : : Account : : bookmarksReceived ( const QXmppBookmarkSet & bookmarks )
{
2019-08-21 09:35:07 +00:00
QList < QXmppBookmarkConference > confs = bookmarks . conferences ( ) ;
for ( QList < QXmppBookmarkConference > : : const_iterator itr = confs . begin ( ) , end = confs . end ( ) ; itr ! = end ; + + itr ) {
2019-07-11 08:51:52 +00:00
const QXmppBookmarkConference & c = * itr ;
QString jid = c . jid ( ) ;
2019-08-21 09:35:07 +00:00
std : : map < QString , Conference * > : : const_iterator cItr = conferences . find ( jid ) ;
if ( cItr = = conferences . end ( ) ) {
2019-09-04 16:38:52 +00:00
addNewRoom ( jid , c . nickName ( ) , c . name ( ) , c . autoJoin ( ) ) ;
2019-07-11 08:51:52 +00:00
} else {
qDebug ( ) < < " Received a bookmark to a MUC " < < jid < < " which is already booked by another bookmark, skipping " ;
}
}
}
2019-08-21 09:35:07 +00:00
void Core : : Account : : onMucJoinedChanged ( bool joined )
2019-07-11 08:51:52 +00:00
{
2019-08-21 09:35:07 +00:00
Conference * room = static_cast < Conference * > ( sender ( ) ) ;
emit changeRoom ( room - > jid , {
{ " joined " , joined }
2019-07-11 08:51:52 +00:00
} ) ;
}
2019-08-21 09:35:07 +00:00
void Core : : Account : : onMucAutoJoinChanged ( bool autoJoin )
2019-07-11 08:51:52 +00:00
{
2019-09-03 20:28:58 +00:00
storeConferences ( ) ;
2019-08-21 09:35:07 +00:00
Conference * room = static_cast < Conference * > ( sender ( ) ) ;
emit changeRoom ( room - > jid , {
{ " autoJoin " , autoJoin }
2019-07-11 08:51:52 +00:00
} ) ;
}
void Core : : Account : : onMucNickNameChanged ( const QString & nickName )
{
2019-09-03 20:28:58 +00:00
storeConferences ( ) ;
2019-08-21 09:35:07 +00:00
Conference * room = static_cast < Conference * > ( sender ( ) ) ;
emit changeRoom ( room - > jid , {
2019-07-11 08:51:52 +00:00
{ " nick " , nickName }
} ) ;
}
2019-08-29 14:19:35 +00:00
void Core : : Account : : setRoomAutoJoin ( const QString & jid , bool joined )
{
std : : map < QString , Conference * > : : const_iterator cItr = conferences . find ( jid ) ;
if ( cItr = = conferences . end ( ) ) {
qDebug ( ) < < " An attempt to set auto join to the non existing room " < < jid < < " of the account " < < getName ( ) < < " , skipping " ;
return ;
}
cItr - > second - > setAutoJoin ( joined ) ;
}
void Core : : Account : : setRoomJoined ( const QString & jid , bool joined )
{
std : : map < QString , Conference * > : : const_iterator cItr = conferences . find ( jid ) ;
if ( cItr = = conferences . end ( ) ) {
qDebug ( ) < < " An attempt to set joined to the non existing room " < < jid < < " of the account " < < getName ( ) < < " , skipping " ;
return ;
}
cItr - > second - > setJoined ( joined ) ;
}
2019-09-01 19:46:12 +00:00
void Core : : Account : : onMucAddParticipant ( const QString & nickName , const QMap < QString , QVariant > & data )
{
Conference * room = static_cast < Conference * > ( sender ( ) ) ;
emit addRoomParticipant ( room - > jid , nickName , data ) ;
}
void Core : : Account : : onMucChangeParticipant ( const QString & nickName , const QMap < QString , QVariant > & data )
{
Conference * room = static_cast < Conference * > ( sender ( ) ) ;
emit changeRoomParticipant ( room - > jid , nickName , data ) ;
}
void Core : : Account : : onMucRemoveParticipant ( const QString & nickName )
{
Conference * room = static_cast < Conference * > ( sender ( ) ) ;
emit removeRoomParticipant ( room - > jid , nickName ) ;
}
2019-09-03 20:28:58 +00:00
void Core : : Account : : onMucSubjectChanged ( const QString & subject )
{
Conference * room = static_cast < Conference * > ( sender ( ) ) ;
emit changeRoom ( room - > jid , {
{ " subject " , subject }
} ) ;
}
void Core : : Account : : storeConferences ( )
{
QXmppBookmarkSet bms = bm - > bookmarks ( ) ;
QList < QXmppBookmarkConference > confs ;
for ( std : : map < QString , Conference * > : : const_iterator itr = conferences . begin ( ) , end = conferences . end ( ) ; itr ! = end ; + + itr ) {
Conference * conference = itr - > second ;
QXmppBookmarkConference conf ;
conf . setJid ( conference - > jid ) ;
conf . setName ( conference - > getName ( ) ) ;
conf . setNickName ( conference - > getNick ( ) ) ;
conf . setAutoJoin ( conference - > getAutoJoin ( ) ) ;
confs . push_back ( conf ) ;
}
bms . setConferences ( confs ) ;
bm - > setBookmarks ( bms ) ;
}
void Core : : Account : : clearConferences ( )
{
for ( std : : map < QString , Conference * > : : const_iterator itr = conferences . begin ( ) , end = conferences . end ( ) ; itr ! = end ; itr + + ) {
itr - > second - > deleteLater ( ) ;
emit removeRoom ( itr - > first ) ;
}
conferences . clear ( ) ;
}
void Core : : Account : : removeRoomRequest ( const QString & jid )
{
std : : map < QString , Conference * > : : const_iterator itr = conferences . find ( jid ) ;
if ( itr = = conferences . end ( ) ) {
qDebug ( ) < < " An attempt to remove non existing room " < < jid < < " from account " < < name < < " , skipping " ;
}
itr - > second - > deleteLater ( ) ;
conferences . erase ( itr ) ;
emit removeRoom ( jid ) ;
storeConferences ( ) ;
}
2019-09-04 16:38:52 +00:00
void Core : : Account : : addRoomRequest ( const QString & jid , const QString & nick , const QString & password , bool autoJoin )
{
std : : map < QString , Conference * > : : const_iterator cItr = conferences . find ( jid ) ;
if ( cItr = = conferences . end ( ) ) {
addNewRoom ( jid , nick , " " , autoJoin ) ;
storeConferences ( ) ;
} else {
qDebug ( ) < < " An attempt to add a MUC " < < jid < < " which is already present in the rester, skipping " ;
}
}
void Core : : Account : : addNewRoom ( const QString & jid , const QString & nick , const QString & roomName , bool autoJoin )
2019-09-03 20:28:58 +00:00
{
2019-09-04 16:38:52 +00:00
QXmppMucRoom * room = mm - > addRoom ( jid ) ;
QString lNick = nick ;
if ( lNick . size ( ) = = 0 ) {
lNick = getName ( ) ;
}
Conference * conf = new Conference ( jid , getName ( ) , autoJoin , roomName , lNick , room ) ;
conferences . insert ( std : : make_pair ( jid , conf ) ) ;
handleNewConference ( conf ) ;
emit addRoom ( jid , {
{ " autoJoin " , conf - > getAutoJoin ( ) } ,
{ " joined " , conf - > getJoined ( ) } ,
{ " nick " , conf - > getNick ( ) } ,
{ " name " , conf - > getName ( ) }
} ) ;
2019-09-03 20:28:58 +00:00
}
2019-09-28 14:30:16 +00:00
void Core : : Account : : addContactToGroupRequest ( const QString & jid , const QString & groupName )
{
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( jid ) ;
if ( itr = = contacts . end ( ) ) {
qDebug ( ) < < " An attempt to add non existing contact " < < jid < < " of account " < < name < < " to the group " < < groupName < < " , skipping " ;
} else {
2019-11-02 20:50:25 +00:00
QXmppRosterIq : : Item item = rm - > getRosterEntry ( jid ) ;
2019-09-28 14:30:16 +00:00
QSet < QString > groups = item . groups ( ) ;
if ( groups . find ( groupName ) = = groups . end ( ) ) { //TODO need to change it, I guess that sort of code is better in qxmpp lib
groups . insert ( groupName ) ;
item . setGroups ( groups ) ;
QXmppRosterIq iq ;
iq . setType ( QXmppIq : : Set ) ;
iq . addItem ( item ) ;
client . sendPacket ( iq ) ;
} else {
qDebug ( ) < < " An attempt to add contact " < < jid < < " of account " < < name < < " to the group " < < groupName < < " but it's already in that group, skipping " ;
}
}
}
void Core : : Account : : removeContactFromGroupRequest ( const QString & jid , const QString & groupName )
{
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( jid ) ;
if ( itr = = contacts . end ( ) ) {
qDebug ( ) < < " An attempt to remove non existing contact " < < jid < < " of account " < < name < < " from the group " < < groupName < < " , skipping " ;
} else {
2019-11-02 20:50:25 +00:00
QXmppRosterIq : : Item item = rm - > getRosterEntry ( jid ) ;
2019-09-28 14:30:16 +00:00
QSet < QString > groups = item . groups ( ) ;
QSet < QString > : : const_iterator gItr = groups . find ( groupName ) ;
if ( gItr ! = groups . end ( ) ) {
groups . erase ( gItr ) ;
item . setGroups ( groups ) ;
QXmppRosterIq iq ;
iq . setType ( QXmppIq : : Set ) ;
iq . addItem ( item ) ;
client . sendPacket ( iq ) ;
} else {
qDebug ( ) < < " An attempt to remove contact " < < jid < < " of account " < < name < < " from the group " < < groupName < < " but it's not in that group, skipping " ;
}
}
}
2019-10-01 08:47:40 +00:00
void Core : : Account : : renameContactRequest ( const QString & jid , const QString & newName )
{
std : : map < QString , Contact * > : : const_iterator itr = contacts . find ( jid ) ;
if ( itr = = contacts . end ( ) ) {
qDebug ( ) < < " An attempt to rename non existing contact " < < jid < < " of account " < < name < < " , skipping " ;
} else {
2019-11-02 20:50:25 +00:00
rm - > renameItem ( jid , newName ) ;
2019-10-01 08:47:40 +00:00
}
}
2019-10-15 19:25:40 +00:00
void Core : : Account : : onVCardReceived ( const QXmppVCardIq & card )
{
QString jid = card . from ( ) ;
pendingVCardRequests . erase ( jid ) ;
RosterItem * item = 0 ;
std : : map < QString , Contact * > : : const_iterator contItr = contacts . find ( jid ) ;
if ( contItr = = contacts . end ( ) ) {
std : : map < QString , Conference * > : : const_iterator confItr = conferences . find ( jid ) ;
if ( confItr = = conferences . end ( ) ) {
2019-10-16 19:38:35 +00:00
if ( jid = = getLogin ( ) + " @ " + getServer ( ) ) {
onOwnVCardReceived ( card ) ;
} else {
qDebug ( ) < < " received vCard " < < jid < < " doesn't belong to any of known contacts or conferences, skipping " ;
}
2019-10-15 19:25:40 +00:00
return ;
} else {
item = confItr - > second ;
}
} else {
item = contItr - > second ;
}
QByteArray ava = card . photo ( ) ;
2019-10-19 19:34:25 +00:00
2019-10-15 19:25:40 +00:00
if ( ava . size ( ) > 0 ) {
item - > setAvatar ( ava ) ;
} else {
2019-10-25 13:38:48 +00:00
if ( ! item - > hasAvatar ( ) | | ! item - > isAvatarAutoGenerated ( ) ) {
item - > setAutoGeneratedAvatar ( ) ;
}
2019-10-15 19:25:40 +00:00
}
2019-10-19 19:34:25 +00:00
Shared : : VCard vCard ;
2019-10-25 13:38:48 +00:00
vCard . setFullName ( card . fullName ( ) ) ;
2019-10-19 19:34:25 +00:00
vCard . setFirstName ( card . firstName ( ) ) ;
vCard . setMiddleName ( card . middleName ( ) ) ;
vCard . setLastName ( card . lastName ( ) ) ;
vCard . setBirthday ( card . birthday ( ) ) ;
vCard . setNickName ( card . nickName ( ) ) ;
vCard . setDescription ( card . description ( ) ) ;
2019-10-25 13:38:48 +00:00
vCard . setUrl ( card . url ( ) ) ;
QXmppVCardOrganization org = card . organization ( ) ;
vCard . setOrgName ( org . organization ( ) ) ;
vCard . setOrgRole ( org . role ( ) ) ;
vCard . setOrgUnit ( org . unit ( ) ) ;
vCard . setOrgTitle ( org . title ( ) ) ;
2019-11-04 15:22:39 +00:00
QList < QXmppVCardEmail > emails = card . emails ( ) ;
std : : deque < Shared : : VCard : : Email > & myEmails = vCard . getEmails ( ) ;
for ( const QXmppVCardEmail & em : emails ) {
Shared : : VCard : : Email mEm ( em . address ( ) ) ;
QXmppVCardEmail : : Type et = em . type ( ) ;
if ( et & QXmppVCardEmail : : Preferred ) {
mEm . prefered = true ;
}
if ( et & QXmppVCardEmail : : Home & & ! ( et & QXmppVCardEmail : : Work ) ) {
mEm . role = Shared : : VCard : : Email : : home ;
} else if ( ! ( et & QXmppVCardEmail : : Home ) & & et & QXmppVCardEmail : : Work ) {
mEm . role = Shared : : VCard : : Email : : work ;
}
myEmails . emplace_back ( mEm ) ;
}
2019-10-19 19:34:25 +00:00
if ( item - > hasAvatar ( ) ) {
2019-10-25 13:38:48 +00:00
if ( ! item - > isAvatarAutoGenerated ( ) ) {
2019-10-19 19:34:25 +00:00
vCard . setAvatarType ( Shared : : Avatar : : valid ) ;
} else {
vCard . setAvatarType ( Shared : : Avatar : : autocreated ) ;
}
vCard . setAvatarPath ( item - > avatarPath ( ) ) ;
} else {
vCard . setAvatarType ( Shared : : Avatar : : empty ) ;
}
2019-10-24 09:42:38 +00:00
QMap < QString , QVariant > cd = {
{ " avatarState " , static_cast < quint8 > ( vCard . getAvatarType ( ) ) } ,
{ " avatarPath " , vCard . getAvatarPath ( ) }
} ;
emit changeContact ( jid , cd ) ;
2019-10-19 19:34:25 +00:00
emit receivedVCard ( jid , vCard ) ;
2019-10-15 19:25:40 +00:00
}
2019-10-16 19:38:35 +00:00
void Core : : Account : : onOwnVCardReceived ( const QXmppVCardIq & card )
{
QByteArray ava = card . photo ( ) ;
2019-10-24 09:42:38 +00:00
bool avaChanged = false ;
2019-10-16 19:38:35 +00:00
QString path = QStandardPaths : : writableLocation ( QStandardPaths : : CacheLocation ) + " / " + name + " / " ;
if ( ava . size ( ) > 0 ) {
QCryptographicHash sha1 ( QCryptographicHash : : Sha1 ) ;
sha1 . addData ( ava ) ;
QString newHash ( sha1 . result ( ) ) ;
QMimeDatabase db ;
QMimeType newType = db . mimeTypeForData ( ava ) ;
if ( avatarType . size ( ) > 0 ) {
if ( avatarHash ! = newHash ) {
QString oldPath = path + " avatar. " + avatarType ;
QFile oldAvatar ( oldPath ) ;
bool oldToRemove = false ;
if ( oldAvatar . exists ( ) ) {
if ( oldAvatar . rename ( oldPath + " .bak " ) ) {
oldToRemove = true ;
} else {
qDebug ( ) < < " Received new avatar for account " < < name < < " but can't get rid of the old one, doing nothing " ;
}
}
QFile newAvatar ( path + " avatar. " + newType . preferredSuffix ( ) ) ;
if ( newAvatar . open ( QFile : : WriteOnly ) ) {
newAvatar . write ( ava ) ;
newAvatar . close ( ) ;
avatarHash = newHash ;
avatarType = newType . preferredSuffix ( ) ;
2019-10-24 09:42:38 +00:00
avaChanged = true ;
2019-10-16 19:38:35 +00:00
} else {
qDebug ( ) < < " Received new avatar for account " < < name < < " but can't save it " ;
if ( oldToRemove ) {
qDebug ( ) < < " rolling back to the old avatar " ;
if ( ! oldAvatar . rename ( oldPath ) ) {
qDebug ( ) < < " Couldn't roll back to the old avatar in account " < < name ;
}
}
}
}
} else {
QFile newAvatar ( path + " avatar. " + newType . preferredSuffix ( ) ) ;
if ( newAvatar . open ( QFile : : WriteOnly ) ) {
newAvatar . write ( ava ) ;
newAvatar . close ( ) ;
avatarHash = newHash ;
avatarType = newType . preferredSuffix ( ) ;
2019-10-24 09:42:38 +00:00
avaChanged = true ;
2019-10-16 19:38:35 +00:00
} else {
qDebug ( ) < < " Received new avatar for account " < < name < < " but can't save it " ;
}
}
} else {
if ( avatarType . size ( ) > 0 ) {
QFile oldAvatar ( path + " avatar. " + avatarType ) ;
if ( ! oldAvatar . remove ( ) ) {
qDebug ( ) < < " Received vCard for account " < < name < < " without avatar, but can't get rid of the file, doing nothing " ;
} else {
2019-10-24 09:42:38 +00:00
avatarType = " " ;
avatarHash = " " ;
avaChanged = true ;
2019-10-16 19:38:35 +00:00
}
}
}
2019-10-24 09:42:38 +00:00
if ( avaChanged ) {
2019-10-16 19:38:35 +00:00
QMap < QString , QVariant > change ;
if ( avatarType . size ( ) > 0 ) {
presence . setPhotoHash ( avatarHash . toUtf8 ( ) ) ;
presence . setVCardUpdateType ( QXmppPresence : : VCardUpdateValidPhoto ) ;
change . insert ( " avatarPath " , path + " avatar. " + avatarType ) ;
} else {
presence . setPhotoHash ( " " ) ;
presence . setVCardUpdateType ( QXmppPresence : : VCardUpdateNoPhoto ) ;
change . insert ( " avatarPath " , " " ) ;
}
client . setClientPresence ( presence ) ;
2019-10-24 09:42:38 +00:00
emit changed ( change ) ;
2019-10-16 19:38:35 +00:00
}
ownVCardRequestInProgress = false ;
2019-10-19 19:34:25 +00:00
Shared : : VCard vCard ;
2019-10-25 13:38:48 +00:00
vCard . setFullName ( card . fullName ( ) ) ;
2019-10-19 19:34:25 +00:00
vCard . setFirstName ( card . firstName ( ) ) ;
vCard . setMiddleName ( card . middleName ( ) ) ;
vCard . setLastName ( card . lastName ( ) ) ;
vCard . setBirthday ( card . birthday ( ) ) ;
vCard . setNickName ( card . nickName ( ) ) ;
vCard . setDescription ( card . description ( ) ) ;
2019-10-25 13:38:48 +00:00
vCard . setUrl ( card . url ( ) ) ;
QXmppVCardOrganization org = card . organization ( ) ;
vCard . setOrgName ( org . organization ( ) ) ;
vCard . setOrgRole ( org . role ( ) ) ;
vCard . setOrgUnit ( org . unit ( ) ) ;
vCard . setOrgTitle ( org . title ( ) ) ;
2019-10-19 19:34:25 +00:00
if ( avatarType . size ( ) > 0 ) {
vCard . setAvatarType ( Shared : : Avatar : : valid ) ;
vCard . setAvatarPath ( path + " avatar. " + avatarType ) ;
} else {
vCard . setAvatarType ( Shared : : Avatar : : empty ) ;
}
2019-11-04 15:22:39 +00:00
QList < QXmppVCardEmail > emails = card . emails ( ) ;
std : : deque < Shared : : VCard : : Email > & myEmails = vCard . getEmails ( ) ;
for ( const QXmppVCardEmail & em : emails ) {
Shared : : VCard : : Email mEm ( em . address ( ) ) ;
QXmppVCardEmail : : Type et = em . type ( ) ;
if ( et & QXmppVCardEmail : : Preferred ) {
mEm . prefered = true ;
}
if ( et & QXmppVCardEmail : : Home & & ! ( et & QXmppVCardEmail : : Work ) ) {
mEm . role = Shared : : VCard : : Email : : home ;
} else if ( ! ( et & QXmppVCardEmail : : Home ) & & et & QXmppVCardEmail : : Work ) {
mEm . role = Shared : : VCard : : Email : : work ;
}
myEmails . emplace_back ( mEm ) ;
}
2019-10-19 19:34:25 +00:00
emit receivedVCard ( getLogin ( ) + " @ " + getServer ( ) , vCard ) ;
2019-10-16 19:38:35 +00:00
}
QString Core : : Account : : getAvatarPath ( ) const
{
return QStandardPaths : : writableLocation ( QStandardPaths : : CacheLocation ) + " / " + name + " / " + " avatar. " + avatarType ;
}
2019-10-15 19:25:40 +00:00
void Core : : Account : : onContactAvatarChanged ( Shared : : Avatar type , const QString & path )
{
RosterItem * item = static_cast < RosterItem * > ( sender ( ) ) ;
QMap < QString , QVariant > cData ( {
2019-10-16 19:38:35 +00:00
{ " avatarState " , static_cast < uint > ( type ) } ,
{ " avatarPath " , path }
2019-10-15 19:25:40 +00:00
} ) ;
emit changeContact ( item - > jid , cData ) ;
}
2019-10-16 19:38:35 +00:00
2019-10-22 15:13:56 +00:00
void Core : : Account : : requestVCard ( const QString & jid )
{
if ( pendingVCardRequests . find ( jid ) = = pendingVCardRequests . end ( ) ) {
if ( jid = = getLogin ( ) + " @ " + getServer ( ) ) {
if ( ! ownVCardRequestInProgress ) {
2019-11-02 20:50:25 +00:00
vm - > requestClientVCard ( ) ;
2019-10-22 15:13:56 +00:00
ownVCardRequestInProgress = true ;
}
} else {
2019-11-02 20:50:25 +00:00
vm - > requestVCard ( jid ) ;
2019-10-22 15:13:56 +00:00
pendingVCardRequests . insert ( jid ) ;
}
}
}
2019-10-24 09:42:38 +00:00
void Core : : Account : : uploadVCard ( const Shared : : VCard & card )
{
QXmppVCardIq iq ;
2019-10-25 13:38:48 +00:00
iq . setFullName ( card . getFullName ( ) ) ;
2019-10-24 09:42:38 +00:00
iq . setFirstName ( card . getFirstName ( ) ) ;
iq . setMiddleName ( card . getMiddleName ( ) ) ;
iq . setLastName ( card . getLastName ( ) ) ;
iq . setNickName ( card . getNickName ( ) ) ;
iq . setBirthday ( card . getBirthday ( ) ) ;
iq . setDescription ( card . getDescription ( ) ) ;
2019-10-25 13:38:48 +00:00
iq . setUrl ( card . getUrl ( ) ) ;
QXmppVCardOrganization org ;
org . setOrganization ( card . getOrgName ( ) ) ;
org . setUnit ( card . getOrgUnit ( ) ) ;
org . setRole ( card . getOrgRole ( ) ) ;
org . setTitle ( card . getOrgTitle ( ) ) ;
iq . setOrganization ( org ) ;
2019-10-24 09:42:38 +00:00
2019-11-04 15:22:39 +00:00
const std : : deque < Shared : : VCard : : Email > & myEmails = card . getEmails ( ) ;
QList < QXmppVCardEmail > emails ;
for ( const Shared : : VCard : : Email & mEm : myEmails ) {
QXmppVCardEmail em ;
QXmppVCardEmail : : Type t = QXmppVCardEmail : : Internet ;
if ( mEm . prefered ) {
t = t | QXmppVCardEmail : : Preferred ;
}
if ( mEm . role = = Shared : : VCard : : Email : : home ) {
t = t | QXmppVCardEmail : : Home ;
} else if ( mEm . role = = Shared : : VCard : : Email : : work ) {
t = t | QXmppVCardEmail : : Work ;
}
em . setType ( t ) ;
em . setAddress ( mEm . address ) ;
emails . push_back ( em ) ;
}
iq . setEmails ( emails ) ;
2019-10-24 09:42:38 +00:00
bool avatarChanged = false ;
if ( card . getAvatarType ( ) = = Shared : : Avatar : : empty ) {
if ( avatarType . size ( ) > 0 ) {
avatarChanged = true ;
}
} else {
QString newPath = card . getAvatarPath ( ) ;
QString oldPath = getAvatarPath ( ) ;
QByteArray data ;
QString type ;
if ( newPath ! = oldPath ) {
QFile avatar ( newPath ) ;
if ( ! avatar . open ( QFile : : ReadOnly ) ) {
qDebug ( ) < < " An attempt to upload new vCard to account " < < name
< < " but it wasn't possible to read file " < < newPath
< < " which was supposed to be new avatar, uploading old avatar " ;
if ( avatarType . size ( ) > 0 ) {
QFile oA ( oldPath ) ;
if ( ! oA . open ( QFile : : ReadOnly ) ) {
qDebug ( ) < < " Couldn't read old avatar of account " < < name < < " , uploading empty avatar " ;
avatarChanged = true ;
} else {
data = oA . readAll ( ) ;
}
}
} else {
data = avatar . readAll ( ) ;
avatarChanged = true ;
}
} else {
if ( avatarType . size ( ) > 0 ) {
QFile oA ( oldPath ) ;
if ( ! oA . open ( QFile : : ReadOnly ) ) {
qDebug ( ) < < " Couldn't read old avatar of account " < < name < < " , uploading empty avatar " ;
avatarChanged = true ;
} else {
data = oA . readAll ( ) ;
}
}
}
if ( data . size ( ) > 0 ) {
QMimeDatabase db ;
type = db . mimeTypeForData ( data ) . name ( ) ;
iq . setPhoto ( data ) ;
iq . setPhotoType ( type ) ;
}
}
2019-11-02 20:50:25 +00:00
vm - > setClientVCard ( iq ) ;
2019-10-24 09:42:38 +00:00
onOwnVCardReceived ( iq ) ;
}