2019-09-01 19:46:12 +00:00
|
|
|
/*
|
|
|
|
* Squawk messenger.
|
|
|
|
* Copyright (C) 2019 Yury Gubich <blue@macaw.me>
|
|
|
|
*
|
|
|
|
* 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-04-05 15:12:59 +00:00
|
|
|
#include "contact.h"
|
2020-04-17 23:17:47 +00:00
|
|
|
#include "account.h"
|
2019-08-28 11:40:55 +00:00
|
|
|
|
2019-04-07 14:02:41 +00:00
|
|
|
#include <QDebug>
|
2019-04-05 15:12:59 +00:00
|
|
|
|
2020-04-17 23:17:47 +00:00
|
|
|
Models::Contact::Contact(const Account* acc, const QString& p_jid ,const QMap<QString, QVariant> &data, Item *parentItem):
|
2019-04-05 15:12:59 +00:00
|
|
|
Item(Item::contact, data, parentItem),
|
2019-04-07 20:14:15 +00:00
|
|
|
jid(p_jid),
|
2020-04-03 22:28:15 +00:00
|
|
|
availability(Shared::Availability::offline),
|
|
|
|
state(Shared::SubscriptionState::none),
|
2019-10-16 19:38:35 +00:00
|
|
|
avatarState(Shared::Avatar::empty),
|
2019-04-09 22:01:25 +00:00
|
|
|
presences(),
|
|
|
|
messages(),
|
2019-10-16 19:38:35 +00:00
|
|
|
childMessages(0),
|
|
|
|
status(),
|
2020-04-17 23:17:47 +00:00
|
|
|
avatarPath(),
|
|
|
|
account(acc)
|
2019-04-05 15:12:59 +00:00
|
|
|
{
|
2019-04-07 20:14:15 +00:00
|
|
|
QMap<QString, QVariant>::const_iterator itr = data.find("state");
|
|
|
|
if (itr != data.end()) {
|
|
|
|
setState(itr.value().toUInt());
|
|
|
|
}
|
2019-10-16 19:38:35 +00:00
|
|
|
|
|
|
|
itr = data.find("avatarState");
|
|
|
|
if (itr != data.end()) {
|
|
|
|
setAvatarState(itr.value().toUInt());
|
|
|
|
}
|
|
|
|
itr = data.find("avatarPath");
|
|
|
|
if (itr != data.end()) {
|
|
|
|
setAvatarPath(itr.value().toString());
|
|
|
|
}
|
2019-04-05 15:12:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Models::Contact::~Contact()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Models::Contact::getJid() const
|
|
|
|
{
|
|
|
|
return jid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::setJid(const QString p_jid)
|
|
|
|
{
|
|
|
|
if (jid != p_jid) {
|
|
|
|
jid = p_jid;
|
2019-04-07 14:02:41 +00:00
|
|
|
changed(1);
|
2019-04-05 15:12:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-07 20:14:15 +00:00
|
|
|
void Models::Contact::setAvailability(unsigned int p_state)
|
2019-04-05 15:12:59 +00:00
|
|
|
{
|
2020-04-03 22:28:15 +00:00
|
|
|
setAvailability(Shared::Global::fromInt<Shared::Availability>(p_state));
|
2019-04-05 15:12:59 +00:00
|
|
|
}
|
|
|
|
|
2019-04-07 20:14:15 +00:00
|
|
|
void Models::Contact::setState(unsigned int p_state)
|
2019-04-05 15:12:59 +00:00
|
|
|
{
|
2020-04-03 22:28:15 +00:00
|
|
|
setState(Shared::Global::fromInt<Shared::SubscriptionState>(p_state));
|
2019-04-07 20:14:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Shared::Availability Models::Contact::getAvailability() const
|
|
|
|
{
|
|
|
|
return availability;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::setAvailability(Shared::Availability p_state)
|
|
|
|
{
|
|
|
|
if (availability != p_state) {
|
|
|
|
availability = p_state;
|
|
|
|
changed(3);
|
2019-04-05 15:12:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-21 19:33:38 +00:00
|
|
|
QString Models::Contact::getStatus() const
|
|
|
|
{
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::setStatus(const QString& p_state)
|
|
|
|
{
|
|
|
|
if (status != p_state) {
|
|
|
|
status = p_state;
|
|
|
|
changed(5);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-05 15:12:59 +00:00
|
|
|
int Models::Contact::columnCount() const
|
|
|
|
{
|
2019-10-16 19:38:35 +00:00
|
|
|
return 8;
|
2019-04-05 15:12:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant Models::Contact::data(int column) const
|
|
|
|
{
|
|
|
|
switch (column) {
|
|
|
|
case 0:
|
2019-06-18 15:08:03 +00:00
|
|
|
return getContactName();
|
2019-04-05 15:12:59 +00:00
|
|
|
case 1:
|
|
|
|
return jid;
|
|
|
|
case 2:
|
2020-04-03 22:28:15 +00:00
|
|
|
return QVariant::fromValue(state);
|
2019-04-09 15:04:08 +00:00
|
|
|
case 3:
|
2020-04-03 22:28:15 +00:00
|
|
|
return QVariant::fromValue(availability);
|
2019-04-09 22:01:25 +00:00
|
|
|
case 4:
|
|
|
|
return getMessagesCount();
|
2019-06-21 19:33:38 +00:00
|
|
|
case 5:
|
|
|
|
return getStatus();
|
2019-10-16 19:38:35 +00:00
|
|
|
case 6:
|
2020-04-03 22:28:15 +00:00
|
|
|
return QVariant::fromValue(getAvatarState());
|
2019-10-16 19:38:35 +00:00
|
|
|
case 7:
|
|
|
|
return getAvatarPath();
|
2019-04-05 15:12:59 +00:00
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-18 15:08:03 +00:00
|
|
|
QString Models::Contact::getContactName() const
|
|
|
|
{
|
|
|
|
if (name == "") {
|
|
|
|
return jid;
|
|
|
|
} else {
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-05 15:12:59 +00:00
|
|
|
void Models::Contact::update(const QString& field, const QVariant& value)
|
|
|
|
{
|
|
|
|
if (field == "name") {
|
|
|
|
setName(value.toString());
|
|
|
|
} else if (field == "jid") {
|
|
|
|
setJid(value.toString());
|
2019-04-07 20:14:15 +00:00
|
|
|
} else if (field == "availability") {
|
|
|
|
setAvailability(value.toUInt());
|
2019-04-05 15:12:59 +00:00
|
|
|
} else if (field == "state") {
|
2019-04-07 20:14:15 +00:00
|
|
|
setState(value.toUInt());
|
2019-10-16 19:38:35 +00:00
|
|
|
} else if (field == "avatarState") {
|
|
|
|
setAvatarState(value.toUInt());
|
|
|
|
} else if (field == "avatarPath") {
|
|
|
|
setAvatarPath(value.toString());
|
2019-04-05 15:12:59 +00:00
|
|
|
}
|
|
|
|
}
|
2019-04-07 14:02:41 +00:00
|
|
|
|
|
|
|
void Models::Contact::addPresence(const QString& p_name, const QMap<QString, QVariant>& data)
|
|
|
|
{
|
|
|
|
QMap<QString, Presence*>::iterator itr = presences.find(p_name);
|
|
|
|
|
|
|
|
if (itr == presences.end()) {
|
|
|
|
Presence* pr = new Presence(data);
|
|
|
|
pr->setName(p_name);
|
|
|
|
presences.insert(p_name, pr);
|
|
|
|
appendChild(pr);
|
|
|
|
} else {
|
|
|
|
Presence* pr = itr.value();
|
|
|
|
for (QMap<QString, QVariant>::const_iterator itr = data.begin(), end = data.end(); itr != end; ++itr) {
|
|
|
|
pr->update(itr.key(), itr.value());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::removePresence(const QString& name)
|
|
|
|
{
|
|
|
|
QMap<QString, Presence*>::iterator itr = presences.find(name);
|
|
|
|
|
|
|
|
if (itr == presences.end()) {
|
2019-04-09 15:04:08 +00:00
|
|
|
qDebug() << "an attempt to remove non existing presence " << name << " from the contact " << jid << " of account " << getAccountName() << ", skipping";
|
2019-04-07 14:02:41 +00:00
|
|
|
} else {
|
|
|
|
Presence* pr = itr.value();
|
2019-05-30 09:36:21 +00:00
|
|
|
presences.erase(itr);
|
2019-07-11 08:51:52 +00:00
|
|
|
removeChild(pr->row());
|
2019-05-30 09:36:21 +00:00
|
|
|
pr->deleteLater();
|
2019-04-07 14:02:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::refresh()
|
|
|
|
{
|
|
|
|
QDateTime lastActivity;
|
|
|
|
Presence* presence = 0;
|
2019-04-09 22:01:25 +00:00
|
|
|
unsigned int count = 0;
|
2019-04-07 14:02:41 +00:00
|
|
|
for (QMap<QString, Presence*>::iterator itr = presences.begin(), end = presences.end(); itr != end; ++itr) {
|
|
|
|
Presence* pr = itr.value();
|
|
|
|
QDateTime la = pr->getLastActivity();
|
2019-04-09 22:01:25 +00:00
|
|
|
count += pr->getMessagesCount();
|
2019-04-07 14:02:41 +00:00
|
|
|
|
|
|
|
if (la > lastActivity) {
|
|
|
|
lastActivity = la;
|
|
|
|
presence = pr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (presence != 0) {
|
2019-04-07 20:14:15 +00:00
|
|
|
setAvailability(presence->getAvailability());
|
2019-06-21 19:33:38 +00:00
|
|
|
setStatus(presence->getStatus());
|
2019-04-07 20:14:15 +00:00
|
|
|
} else {
|
2020-04-03 22:28:15 +00:00
|
|
|
setAvailability(Shared::Availability::offline);
|
2019-06-21 19:33:38 +00:00
|
|
|
setStatus("");
|
2019-04-07 14:02:41 +00:00
|
|
|
}
|
2019-04-09 22:01:25 +00:00
|
|
|
|
|
|
|
if (childMessages != count) {
|
|
|
|
childMessages = count;
|
|
|
|
changed(4);
|
|
|
|
}
|
2019-04-07 14:02:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::_removeChild(int index)
|
|
|
|
{
|
2019-04-09 15:04:08 +00:00
|
|
|
Item* child = childItems[index];
|
2019-11-03 18:46:40 +00:00
|
|
|
disconnect(child, &Item::childChanged, this, &Contact::refresh);
|
2019-04-07 14:02:41 +00:00
|
|
|
Item::_removeChild(index);
|
|
|
|
refresh();
|
|
|
|
}
|
|
|
|
|
2020-04-18 12:02:01 +00:00
|
|
|
void Models::Contact::_appendChild(Models::Item* child)
|
2019-04-07 14:02:41 +00:00
|
|
|
{
|
2020-04-18 12:02:01 +00:00
|
|
|
Item::_appendChild(child);
|
2019-11-03 18:46:40 +00:00
|
|
|
connect(child, &Item::childChanged, this, &Contact::refresh);
|
2019-04-07 14:02:41 +00:00
|
|
|
refresh();
|
|
|
|
}
|
2019-04-07 20:14:15 +00:00
|
|
|
|
|
|
|
Shared::SubscriptionState Models::Contact::getState() const
|
|
|
|
{
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::setState(Shared::SubscriptionState p_state)
|
|
|
|
{
|
|
|
|
if (state != p_state) {
|
|
|
|
state = p_state;
|
|
|
|
changed(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-23 21:09:39 +00:00
|
|
|
QIcon Models::Contact::getStatusIcon(bool big) const
|
2019-04-07 20:14:15 +00:00
|
|
|
{
|
2019-04-09 22:01:25 +00:00
|
|
|
if (getMessagesCount() > 0) {
|
2019-08-31 20:50:05 +00:00
|
|
|
return Shared::icon("mail-message", big);
|
2020-04-03 22:28:15 +00:00
|
|
|
} else if (state == Shared::SubscriptionState::both || state == Shared::SubscriptionState::to) {
|
2019-06-23 21:09:39 +00:00
|
|
|
return Shared::availabilityIcon(availability, big);;
|
2019-04-07 20:14:15 +00:00
|
|
|
} else {
|
2019-08-31 20:50:05 +00:00
|
|
|
return Shared::subscriptionStateIcon(state, big);
|
2019-04-07 20:14:15 +00:00
|
|
|
}
|
|
|
|
}
|
2019-04-09 15:04:08 +00:00
|
|
|
|
2019-04-11 14:58:59 +00:00
|
|
|
void Models::Contact::addMessage(const Shared::Message& data)
|
2019-04-09 22:01:25 +00:00
|
|
|
{
|
2019-04-11 14:58:59 +00:00
|
|
|
const QString& res = data.getPenPalResource();
|
2019-04-09 22:01:25 +00:00
|
|
|
if (res.size() > 0) {
|
|
|
|
QMap<QString, Presence*>::iterator itr = presences.find(res);
|
|
|
|
if (itr == presences.end()) {
|
2019-06-22 20:37:22 +00:00
|
|
|
// this is actually the place when I can spot someone's invisible presence, and there is nothing criminal in it, cuz the sender sent us a message
|
|
|
|
// therefore he have revealed himself
|
|
|
|
// the only issue is to find out when the sender is gone offline
|
|
|
|
Presence* pr = new Presence({});
|
|
|
|
pr->setName(res);
|
2020-04-03 22:28:15 +00:00
|
|
|
pr->setAvailability(Shared::Availability::invisible);
|
2020-03-30 21:17:10 +00:00
|
|
|
pr->setLastActivity(QDateTime::currentDateTimeUtc());
|
2019-06-22 20:37:22 +00:00
|
|
|
presences.insert(res, pr);
|
|
|
|
appendChild(pr);
|
|
|
|
pr->addMessage(data);
|
2019-04-09 22:01:25 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
itr.value()->addMessage(data);
|
|
|
|
} else {
|
|
|
|
messages.emplace_back(data);
|
|
|
|
changed(4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-08 11:44:15 +00:00
|
|
|
void Models::Contact::changeMessage(const QString& id, const QMap<QString, QVariant>& data)
|
|
|
|
{
|
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
for (Shared::Message& msg : messages) {
|
|
|
|
if (msg.getId() == id) {
|
|
|
|
msg.change(data);
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
for (Presence* pr : presences) {
|
|
|
|
found = pr->changeMessage(id, data);
|
|
|
|
if (found) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-09 22:01:25 +00:00
|
|
|
unsigned int Models::Contact::getMessagesCount() const
|
|
|
|
{
|
|
|
|
return messages.size() + childMessages;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::dropMessages()
|
|
|
|
{
|
|
|
|
if (messages.size() > 0) {
|
|
|
|
messages.clear();
|
|
|
|
changed(4);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (QMap<QString, Presence*>::iterator itr = presences.begin(), end = presences.end(); itr != end; ++itr) {
|
|
|
|
itr.value()->dropMessages();
|
|
|
|
}
|
|
|
|
}
|
2019-04-10 20:53:42 +00:00
|
|
|
|
|
|
|
void Models::Contact::getMessages(Models::Contact::Messages& container) const
|
|
|
|
{
|
|
|
|
for (Messages::const_iterator itr = messages.begin(), end = messages.end(); itr != end; ++itr) {
|
2019-04-11 14:58:59 +00:00
|
|
|
const Shared::Message& msg = *itr;
|
2019-04-10 20:53:42 +00:00
|
|
|
container.push_back(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (QMap<QString, Presence*>::const_iterator itr = presences.begin(), end = presences.end(); itr != end; ++itr) {
|
|
|
|
itr.value()->getMessages(container);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-23 11:31:03 +00:00
|
|
|
void Models::Contact::toOfflineState()
|
|
|
|
{
|
2020-04-18 12:02:01 +00:00
|
|
|
std::deque<Item*>::size_type size = childItems.size();
|
|
|
|
if (size > 0) {
|
|
|
|
emit childIsAboutToBeRemoved(this, 0, size - 1);
|
|
|
|
for (std::deque<Item*>::size_type i = 0; i < size; ++i) {
|
|
|
|
Item* item = childItems[0];
|
|
|
|
disconnect(item, &Item::childChanged, this, &Contact::refresh);
|
|
|
|
Item::_removeChild(0);
|
|
|
|
item->deleteLater();
|
|
|
|
}
|
|
|
|
childItems.clear();
|
|
|
|
presences.clear();
|
|
|
|
emit childRemoved();
|
|
|
|
refresh();
|
2019-06-23 11:31:03 +00:00
|
|
|
}
|
|
|
|
}
|
2019-09-24 09:21:29 +00:00
|
|
|
|
|
|
|
QString Models::Contact::getDisplayedName() const
|
|
|
|
{
|
|
|
|
return getContactName();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Models::Contact::columnInvolvedInDisplay(int col)
|
|
|
|
{
|
|
|
|
return Item::columnInvolvedInDisplay(col) && col == 1;
|
|
|
|
}
|
2019-09-28 14:30:16 +00:00
|
|
|
|
|
|
|
Models::Contact * Models::Contact::copy() const
|
|
|
|
{
|
|
|
|
Contact* cnt = new Contact(*this);
|
|
|
|
return cnt;
|
|
|
|
}
|
|
|
|
|
|
|
|
Models::Contact::Contact(const Models::Contact& other):
|
|
|
|
Item(other),
|
|
|
|
jid(other.jid),
|
|
|
|
availability(other.availability),
|
|
|
|
state(other.state),
|
|
|
|
presences(),
|
|
|
|
messages(other.messages),
|
2020-04-17 23:17:47 +00:00
|
|
|
childMessages(0),
|
|
|
|
account(other.account)
|
2019-09-28 14:30:16 +00:00
|
|
|
{
|
|
|
|
for (const Presence* pres : other.presences) {
|
|
|
|
Presence* pCopy = new Presence(*pres);
|
|
|
|
presences.insert(pCopy->getName(), pCopy);
|
|
|
|
Item::appendChild(pCopy);
|
2019-11-03 18:46:40 +00:00
|
|
|
connect(pCopy, &Item::childChanged, this, &Contact::refresh);
|
2019-09-28 14:30:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
refresh();
|
|
|
|
}
|
2019-10-16 19:38:35 +00:00
|
|
|
|
|
|
|
QString Models::Contact::getAvatarPath() const
|
|
|
|
{
|
|
|
|
return avatarPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
Shared::Avatar Models::Contact::getAvatarState() const
|
|
|
|
{
|
|
|
|
return avatarState;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::setAvatarPath(const QString& path)
|
|
|
|
{
|
|
|
|
if (path != avatarPath) {
|
|
|
|
avatarPath = path;
|
|
|
|
changed(7);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::setAvatarState(Shared::Avatar p_state)
|
|
|
|
{
|
|
|
|
if (avatarState != p_state) {
|
|
|
|
avatarState = p_state;
|
|
|
|
changed(6);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Models::Contact::setAvatarState(unsigned int p_state)
|
|
|
|
{
|
|
|
|
if (p_state <= static_cast<quint8>(Shared::Avatar::valid)) {
|
|
|
|
Shared::Avatar state = static_cast<Shared::Avatar>(p_state);
|
|
|
|
setAvatarState(state);
|
|
|
|
} else {
|
|
|
|
qDebug() << "An attempt to set invalid avatar state" << p_state << "to the contact" << jid << ", skipping";
|
|
|
|
}
|
|
|
|
}
|
2020-04-17 23:17:47 +00:00
|
|
|
|
|
|
|
const Models::Account * Models::Contact::getParentAccount() const
|
|
|
|
{
|
|
|
|
return account;
|
|
|
|
}
|
|
|
|
|