mirror of
https://github.com/RetroShare/RetroShare.git
synced 2024-12-12 01:14:29 -05:00
642 lines
22 KiB
C++
642 lines
22 KiB
C++
/*******************************************************************************
|
|
* gui/NewsFeed.cpp *
|
|
* *
|
|
* Copyright (c) 2008 Robert Fernie <retroshare.project@gmail.com> *
|
|
* *
|
|
* This program is free software: you can redistribute it and/or modify *
|
|
* it under the terms of the GNU Affero 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 Affero General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU Affero General Public License *
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>. *
|
|
* *
|
|
*******************************************************************************/
|
|
|
|
#include <QDateTime>
|
|
|
|
#include "NewsFeed.h"
|
|
#include "ui_NewsFeed.h"
|
|
|
|
#include <retroshare/rsbanlist.h>
|
|
#include <retroshare/rsgxschannels.h>
|
|
#include <retroshare/rsgxsforums.h>
|
|
#include <retroshare/rsmsgs.h>
|
|
#include <retroshare/rsnotify.h>
|
|
#include <retroshare/rspeers.h>
|
|
#include <retroshare/rsplugin.h>
|
|
#include <retroshare/rsposted.h>
|
|
|
|
#include "util/qtthreadsutils.h"
|
|
#include "feeds/ChatMsgItem.h"
|
|
#include "feeds/GxsCircleItem.h"
|
|
#include "feeds/GxsChannelGroupItem.h"
|
|
#include "feeds/GxsChannelPostItem.h"
|
|
#include "feeds/GxsForumGroupItem.h"
|
|
#include "feeds/GxsForumMsgItem.h"
|
|
#include "feeds/MsgItem.h"
|
|
#include "feeds/NewsFeedUserNotify.h"
|
|
#include "feeds/PeerItem.h"
|
|
#include "feeds/PostedGroupItem.h"
|
|
#include "feeds/SecurityItem.h"
|
|
#include "feeds/SecurityIpItem.h"
|
|
|
|
#include "settings/rsettingswin.h"
|
|
#include "settings/rsharesettings.h"
|
|
|
|
#include "chat/ChatDialog.h"
|
|
#include "Posted/PostedItem.h"
|
|
#include "msgs/MessageComposer.h"
|
|
#include "msgs/MessageInterface.h"
|
|
|
|
#include "common/FeedNotify.h"
|
|
#include "notifyqt.h"
|
|
|
|
#define ROLE_RECEIVED FEED_TREEWIDGET_SORTROLE
|
|
|
|
#define TOKEN_TYPE_GROUP 1
|
|
#define TOKEN_TYPE_MESSAGE 2
|
|
#define TOKEN_TYPE_PUBLISHKEY 3
|
|
|
|
/*****
|
|
* #define NEWS_DEBUG 1
|
|
****/
|
|
|
|
static NewsFeed* instance = nullptr;
|
|
|
|
/** Constructor */
|
|
NewsFeed::NewsFeed(QWidget *parent) : MainPage(parent), ui(new Ui::NewsFeed),
|
|
mEventTypes({
|
|
RsEventType::AUTHSSL_CONNECTION_AUTENTICATION,
|
|
RsEventType::PEER_CONNECTION ,
|
|
RsEventType::GXS_CIRCLES ,
|
|
RsEventType::GXS_CHANNELS ,
|
|
RsEventType::GXS_FORUMS ,
|
|
RsEventType::GXS_POSTED ,
|
|
RsEventType::MAIL_STATUS
|
|
})
|
|
{
|
|
for(uint32_t i=0;i<mEventTypes.size();++i)
|
|
{
|
|
mEventHandlerIds.push_back(0); // needed to force intialization by registerEventsHandler()
|
|
rsEvents->registerEventsHandler(
|
|
[this](std::shared_ptr<const RsEvent> event) { handleEvent(event); },
|
|
mEventHandlerIds.back(), mEventTypes[i] );
|
|
}
|
|
|
|
/* Invoke the Qt Designer generated object setup routine */
|
|
ui->setupUi(this);
|
|
|
|
if (!instance) {
|
|
instance = this;
|
|
}
|
|
|
|
ui->feedWidget->enableRemove(true);
|
|
|
|
ui->sortComboBox->addItem(tr("Newest on top"), Qt::DescendingOrder);
|
|
ui->sortComboBox->addItem(tr("Oldest on top"), Qt::AscendingOrder);
|
|
|
|
connect(ui->sortComboBox, SIGNAL(currentIndexChanged(int)), this, SLOT(sortChanged(int)));
|
|
|
|
connect(ui->removeAllButton, SIGNAL(clicked()), ui->feedWidget, SLOT(clear()));
|
|
connect(ui->feedWidget, SIGNAL(feedCountChanged()), this, SLOT(sendNewsFeedChanged()),Qt::QueuedConnection);
|
|
|
|
connect(ui->feedOptionsButton, SIGNAL(clicked()), this, SLOT(feedoptions()));
|
|
ui->feedOptionsButton->hide(); // (csoler) Hidden until we repare the system to display a specific settings page.
|
|
|
|
QString hlp_str = tr(
|
|
" <h1><img width=\"32\" src=\":/icons/help_64.png\"> News Feed</h1> \
|
|
<p>The Log Feed displays the last events on your network, sorted by the time you received them. \
|
|
This gives you a summary of the activity of your friends. \
|
|
You can configure which events to show by pressing on <b>Options</b>. </p> \
|
|
<p>The various events shown are: \
|
|
<ul> \
|
|
<li>Connection attempts (useful to make friends with new people and control who's trying to reach you)</li> \
|
|
<li>Channel and Forum posts</li> \
|
|
<li>New Channels and Forums you can subscribe to</li> \
|
|
<li>Private messages from your friends</li> \
|
|
</ul> </p> \
|
|
") ;
|
|
|
|
registerHelpButton(ui->helpButton,hlp_str,"NewFeed") ;
|
|
|
|
// load settings
|
|
processSettings(true);
|
|
}
|
|
|
|
NewsFeed::~NewsFeed()
|
|
{
|
|
for(uint32_t i=0;i<mEventHandlerIds.size();++i)
|
|
rsEvents->unregisterEventsHandler(mEventHandlerIds[i]);
|
|
|
|
// save settings
|
|
processSettings(false);
|
|
|
|
if (instance == this) {
|
|
instance = NULL;
|
|
}
|
|
}
|
|
|
|
UserNotify *NewsFeed::createUserNotify(QObject *parent)
|
|
{
|
|
return new NewsFeedUserNotify(this, parent);
|
|
}
|
|
|
|
void NewsFeed::processSettings(bool load)
|
|
{
|
|
Settings->beginGroup("NewsFeed");
|
|
|
|
if (load) {
|
|
// load settings
|
|
|
|
// state of sort order
|
|
Qt::SortOrder sortOrder = (Qt::SortOrder) Settings->value("SortOrder", Qt::AscendingOrder).toInt();
|
|
ui->sortComboBox->setCurrentIndex(ui->sortComboBox->findData(sortOrder));
|
|
sortChanged(ui->sortComboBox->currentIndex());
|
|
} else {
|
|
// save settings
|
|
|
|
// state of sort order
|
|
Settings->setValue("SortOrder", ui->sortComboBox->itemData(ui->sortComboBox->currentIndex()).toInt());
|
|
}
|
|
|
|
Settings->endGroup();
|
|
}
|
|
|
|
void NewsFeed::sortChanged(int index)
|
|
{
|
|
Qt::SortOrder sortOrder = (Qt::SortOrder) ui->sortComboBox->itemData(index).toInt();
|
|
ui->feedWidget->setSortRole(ROLE_RECEIVED, sortOrder);
|
|
}
|
|
|
|
// handler for the new notification system in libretroshare.
|
|
|
|
void NewsFeed::handleEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
// /!\ Absolutely no access to Qt structures (such as Settings) should happen here!!!
|
|
|
|
RsQThreadUtils::postToObject( [=]() { handleEvent_main_thread(event); }, this );
|
|
}
|
|
|
|
void NewsFeed::handleEvent_main_thread(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
uint flags = Settings->getNewsFeedFlags();
|
|
|
|
if(event->mType == RsEventType::AUTHSSL_CONNECTION_AUTENTICATION && (flags & RS_FEED_TYPE_SECURITY))
|
|
handleSecurityEvent(event);
|
|
|
|
if(event->mType == RsEventType::PEER_CONNECTION && (flags & RS_FEED_TYPE_PEER))
|
|
handleConnectionEvent(event);
|
|
|
|
if(event->mType == RsEventType::GXS_CIRCLES && (flags & RS_FEED_TYPE_CIRCLE))
|
|
handleCircleEvent(event);
|
|
|
|
if(event->mType == RsEventType::GXS_CHANNELS && (flags & RS_FEED_TYPE_CHANNEL))
|
|
handleChannelEvent(event);
|
|
|
|
if(event->mType == RsEventType::GXS_FORUMS && (flags & RS_FEED_TYPE_FORUM))
|
|
handleForumEvent(event);
|
|
|
|
if(event->mType == RsEventType::GXS_POSTED && (flags & RS_FEED_TYPE_POSTED))
|
|
handlePostedEvent(event);
|
|
|
|
if(event->mType == RsEventType::MAIL_STATUS && (flags & RS_FEED_TYPE_MSG))
|
|
handleMailEvent(event);
|
|
}
|
|
|
|
void NewsFeed::handleMailEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
const RsMailStatusEvent *pe =
|
|
dynamic_cast<const RsMailStatusEvent*>(event.get());
|
|
if(!pe) return;
|
|
|
|
|
|
switch(pe->mMailStatusEventCode)
|
|
{
|
|
case RsMailStatusEventCode::NEW_MESSAGE:
|
|
for(auto msgid: pe->mChangedMsgIds)
|
|
addFeedItem( new MsgItem(this, NEWSFEED_MESSAGELIST, msgid, false));
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
void NewsFeed::handlePostedEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
const RsGxsPostedEvent *pe =
|
|
dynamic_cast<const RsGxsPostedEvent*>(event.get());
|
|
if(!pe) return;
|
|
|
|
switch(pe->mPostedEventCode)
|
|
{
|
|
case RsPostedEventCode::NEW_POSTED_GROUP:
|
|
addFeedItem( new PostedGroupItem(this, NEWSFEED_POSTEDNEWLIST, pe->mPostedGroupId, false, true));
|
|
break;
|
|
case RsPostedEventCode::NEW_MESSAGE:
|
|
addFeedItem( new PostedItem(this, NEWSFEED_POSTEDMSGLIST, pe->mPostedGroupId, pe->mPostedMsgId, false, true));
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
void NewsFeed::handleForumEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
const RsGxsForumEvent *pe = dynamic_cast<const RsGxsForumEvent*>(event.get());
|
|
if(!pe) return;
|
|
|
|
switch(pe->mForumEventCode)
|
|
{
|
|
case RsForumEventCode::MODERATOR_LIST_CHANGED:
|
|
addFeedItem(new GxsForumGroupItem(this, NEWSFEED_UPDATED_FORUM, pe->mForumGroupId,pe->mModeratorsAdded,pe->mModeratorsRemoved, false, true));
|
|
break;
|
|
|
|
case RsForumEventCode::UPDATED_FORUM:
|
|
case RsForumEventCode::NEW_FORUM:
|
|
addFeedItem(new GxsForumGroupItem(this, NEWSFEED_NEW_FORUM, pe->mForumGroupId, false, true));
|
|
break;
|
|
|
|
case RsForumEventCode::UPDATED_MESSAGE:
|
|
case RsForumEventCode::NEW_MESSAGE:
|
|
addFeedItem(new GxsForumMsgItem(this, NEWSFEED_NEW_FORUM, pe->mForumGroupId, pe->mForumMsgId, false, true));
|
|
break;
|
|
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
void NewsFeed::handleChannelEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
const RsGxsChannelEvent* pe =
|
|
dynamic_cast<const RsGxsChannelEvent*>(event.get());
|
|
if(!pe) return;
|
|
|
|
switch(pe->mChannelEventCode)
|
|
{
|
|
case RsChannelEventCode::UPDATED_CHANNEL: // [[fallthrough]];
|
|
case RsChannelEventCode::NEW_CHANNEL:
|
|
addFeedItem(new GxsChannelGroupItem(this, NEWSFEED_CHANNELNEWLIST, pe->mChannelGroupId, false, true));
|
|
break;
|
|
case RsChannelEventCode::UPDATED_MESSAGE: // [[fallthrough]];
|
|
case RsChannelEventCode::NEW_MESSAGE:
|
|
addFeedItem(new GxsChannelPostItem(this, NEWSFEED_CHANNELNEWLIST, pe->mChannelGroupId, pe->mChannelMsgId, false, true));
|
|
break;
|
|
case RsChannelEventCode::RECEIVED_PUBLISH_KEY:
|
|
addFeedItem(new GxsChannelGroupItem(this, NEWSFEED_CHANNELPUBKEYLIST, pe->mChannelGroupId, false, true));
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
void NewsFeed::handleCircleEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
// Gives the backend a few secs to load the cache data while not blocking the UI. This is not so nice, but there's no proper
|
|
// other way to do that.
|
|
|
|
RsThread::async( [event,this]()
|
|
{
|
|
const RsGxsCircleEvent *pe = dynamic_cast<const RsGxsCircleEvent*>(event.get());
|
|
if(!pe)
|
|
return;
|
|
|
|
if(pe->mCircleId.isNull()) // probably an item for cache update
|
|
return ;
|
|
|
|
RsGxsCircleDetails details;
|
|
bool loaded = false;
|
|
|
|
for(int i=0;i<5 && !loaded;++i)
|
|
if(rsGxsCircles->getCircleDetails(pe->mCircleId,details))
|
|
{
|
|
std::cerr << "Cache item loaded for circle " << pe->mCircleId << std::endl;
|
|
loaded = true;
|
|
}
|
|
else
|
|
{
|
|
std::cerr << "Cache item for circle " << pe->mCircleId << " not loaded. Waiting " << i << "s" << std::endl;
|
|
rstime::rs_usleep(1000*1000);
|
|
}
|
|
|
|
if(!loaded)
|
|
{
|
|
std::cerr << "(EE) Cannot get information about circle " << pe->mCircleId << ". Not in cache?" << std::endl;
|
|
return;
|
|
}
|
|
|
|
if(!details.isGxsIdBased()) // not handled yet.
|
|
return;
|
|
|
|
// Check if the circle is one of which we belong to or we are an admin of.
|
|
// If so, then notify in the GUI about other members leaving/subscribing, according
|
|
// to the following rules. The names correspond to the RS_FEED_CIRCLE_* variables:
|
|
//
|
|
// Message-based notifications:
|
|
//
|
|
// +---------------------------+----------------------------+
|
|
// | Membership request | Membership cancellation |
|
|
// +-------------+-------------+-------------+--------------+
|
|
// | Admin | Not admin | Admin | Not admin |
|
|
// +--------------------+-------------+-------------+----------------------------+
|
|
// | in invitee list | MEMB_JOIN | MEMB_JOIN | MEMB_LEAVE | MEMB_LEAVE |
|
|
// +--------------------+-------------+-------------+-------------+--------------+
|
|
// |not in invitee list | MEMB_REQ | X | X | X |
|
|
// +--------------------+-------------+-------------+-------------+--------------+
|
|
//
|
|
// Note: in this case, the GxsId never belongs to you, since you dont need to handle
|
|
// notifications for actions you took yourself (leave/join a circle)
|
|
//
|
|
// GroupData-based notifications, the GxsId belongs to you:
|
|
//
|
|
// +---------------------------+----------------------------+
|
|
// | GxsId joins invitee list | GxsId leaves invitee list |
|
|
// +-------------+-------------+-------------+--------------+
|
|
// | Id is yours| Id is not | Id is yours | Id is not |
|
|
// +--------------------+-------------+-------------+-------------+--------------+
|
|
// | Has Member request | MEMB_ACCEPT | (MEMB_JOIN) | MEMB_REVOKED| (MEMB_LEAVE) |
|
|
// +--------------------+-------------+-------------+-------------+--------------+
|
|
// | No Member request | INVITE_REC | X | INVITE_REM | X |
|
|
// +--------------------+-------------+-------------+-------------+--------------+
|
|
//
|
|
// Note: In this case you're never an admin of the circle, since these notification
|
|
// would be a direct consequence of your own actions.
|
|
|
|
RsQThreadUtils::postToObject( [event,details,this]()
|
|
{
|
|
const RsGxsCircleEvent *pe = static_cast<const RsGxsCircleEvent*>(event.get());
|
|
|
|
switch(pe->mCircleEventType)
|
|
{
|
|
case RsGxsCircleEventCode::CIRCLE_MEMBERSHIP_REQUEST:
|
|
// only show membership requests if we're an admin of that circle
|
|
if(details.isIdInInviteeList(pe->mGxsId))
|
|
addFeedItemIfUnique(new GxsCircleItem(this, NEWSFEED_CIRCLELIST, pe->mCircleId, pe->mGxsId, RS_FEED_ITEM_CIRCLE_MEMB_JOIN),true);
|
|
else if(details.mAmIAdmin)
|
|
addFeedItemIfUnique(new GxsCircleItem(this, NEWSFEED_CIRCLELIST, pe->mCircleId, pe->mGxsId, RS_FEED_ITEM_CIRCLE_MEMB_REQ),true);
|
|
|
|
break;
|
|
|
|
case RsGxsCircleEventCode::CIRCLE_MEMBERSHIP_LEAVE:
|
|
|
|
if(details.isIdInInviteeList(pe->mGxsId))
|
|
addFeedItemIfUnique(new GxsCircleItem(this, NEWSFEED_CIRCLELIST, pe->mCircleId, pe->mGxsId, RS_FEED_ITEM_CIRCLE_MEMB_LEAVE),true);
|
|
break;
|
|
|
|
case RsGxsCircleEventCode::CIRCLE_MEMBERSHIP_ID_ADDED_TO_INVITEE_LIST:
|
|
if(rsIdentity->isOwnId(pe->mGxsId))
|
|
{
|
|
if(details.isIdRequestingMembership(pe->mGxsId))
|
|
addFeedItemIfUnique(new GxsCircleItem(this, NEWSFEED_CIRCLELIST, pe->mCircleId, pe->mGxsId, RS_FEED_ITEM_CIRCLE_MEMB_ACCEPTED),true);
|
|
else
|
|
addFeedItemIfUnique(new GxsCircleItem(this, NEWSFEED_CIRCLELIST, pe->mCircleId, pe->mGxsId, RS_FEED_ITEM_CIRCLE_INVITE_REC),true);
|
|
}
|
|
break;
|
|
|
|
case RsGxsCircleEventCode::CIRCLE_MEMBERSHIP_ID_REMOVED_FROM_INVITEE_LIST:
|
|
if(rsIdentity->isOwnId(pe->mGxsId))
|
|
{
|
|
if(details.isIdRequestingMembership(pe->mGxsId))
|
|
addFeedItemIfUnique(new GxsCircleItem(this, NEWSFEED_CIRCLELIST, pe->mCircleId, pe->mGxsId, RS_FEED_ITEM_CIRCLE_MEMB_REVOKED),true);
|
|
else
|
|
addFeedItemIfUnique(new GxsCircleItem(this, NEWSFEED_CIRCLELIST, pe->mCircleId, pe->mGxsId, RS_FEED_ITEM_CIRCLE_INVITE_CANCELLED),true);
|
|
}
|
|
break;
|
|
|
|
default: break;
|
|
}
|
|
}, this ); }); // damn!
|
|
}
|
|
|
|
void NewsFeed::handleConnectionEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
const RsConnectionEvent *pe = dynamic_cast<const RsConnectionEvent*>(event.get());
|
|
if(!pe) return;
|
|
|
|
auto& e(*pe);
|
|
|
|
#ifdef NEWS_DEBUG
|
|
std::cerr << "NotifyQt: handling connection event from peer " << e.mSslId << std::endl;
|
|
#endif
|
|
|
|
switch(e.mConnectionInfoCode)
|
|
{
|
|
case RsConnectionEventCode::PEER_CONNECTED:
|
|
addFeedItemIfUnique(new PeerItem(this, NEWSFEED_PEERLIST, e.mSslId, PEER_TYPE_CONNECT, false), true);
|
|
break;
|
|
case RsConnectionEventCode::PEER_DISCONNECTED: // not handled yet
|
|
break;
|
|
case RsConnectionEventCode::PEER_TIME_SHIFT:
|
|
addFeedItemIfUnique(new PeerItem(this, NEWSFEED_PEERLIST, e.mSslId, PEER_TYPE_OFFSET, false),false);
|
|
break;
|
|
case RsConnectionEventCode::PEER_REPORTS_WRONG_IP:
|
|
addFeedItemIfUnique(new SecurityIpItem(
|
|
this, e.mSslId, e.mOwnLocator.toString(),
|
|
e.mReportedLocator.toString(),
|
|
RS_FEED_ITEM_SEC_IP_WRONG_EXTERNAL_IP_REPORTED,
|
|
false ), false);
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
void NewsFeed::handleSecurityEvent(std::shared_ptr<const RsEvent> event)
|
|
{
|
|
const RsAuthSslConnectionAutenticationEvent *pe = dynamic_cast<const RsAuthSslConnectionAutenticationEvent*>(event.get());
|
|
|
|
if(!pe)
|
|
return;
|
|
|
|
auto& e(*pe);
|
|
#ifdef NEWS_DEBUG
|
|
std::cerr << "NotifyQt: handling security event from (" << e.mSslId << "," << e.mPgpId << ") error code: " << (int)e.mErrorCode << std::endl;
|
|
#endif
|
|
uint flags = Settings->getNewsFeedFlags();
|
|
|
|
if(e.mErrorCode == RsAuthSslError::PEER_REFUSED_CONNECTION)
|
|
{
|
|
addFeedItemIfUnique(new PeerItem(this, NEWSFEED_PEERLIST, e.mSslId, PEER_TYPE_HELLO, false), true );
|
|
return;
|
|
}
|
|
|
|
uint32_t FeedItemType=0;
|
|
|
|
switch(e.mErrorCode)
|
|
{
|
|
case RsAuthSslError::NO_CERTIFICATE_SUPPLIED: // fallthrough
|
|
case RsAuthSslError::MISMATCHED_PGP_ID: // fallthrough
|
|
case RsAuthSslError::MISSING_AUTHENTICATION_INFO:
|
|
FeedItemType = RS_FEED_ITEM_SEC_BAD_CERTIFICATE; break;
|
|
case RsAuthSslError::PGP_SIGNATURE_VALIDATION_FAILED:
|
|
FeedItemType = RS_FEED_ITEM_SEC_WRONG_SIGNATURE; break;
|
|
case RsAuthSslError::NOT_A_FRIEND:
|
|
FeedItemType = RS_FEED_ITEM_SEC_CONNECT_ATTEMPT; break;
|
|
case RsAuthSslError::IP_IS_BLACKLISTED:
|
|
FeedItemType = RS_FEED_ITEM_SEC_IP_BLACKLISTED; break;
|
|
case RsAuthSslError::MISSING_CERTIFICATE:
|
|
FeedItemType = RS_FEED_ITEM_SEC_MISSING_CERTIFICATE; break;
|
|
default:
|
|
return; // display nothing
|
|
}
|
|
|
|
RsPeerDetails det;
|
|
rsPeers->getPeerDetails(e.mSslId,det) || rsPeers->getGPGDetails(e.mPgpId,det);
|
|
|
|
addFeedItemIfUnique(new SecurityItem(this, NEWSFEED_SECLIST, e.mPgpId, e.mSslId, det.location, e.mLocator.toString(), FeedItemType, false), true );
|
|
|
|
if (Settings->getMessageFlags() & RS_MESSAGE_CONNECT_ATTEMPT)
|
|
MessageComposer::addConnectAttemptMsg(e.mPgpId, e.mSslId, QString::fromStdString(det.name + "(" + det.location + ")"));
|
|
}
|
|
|
|
void NewsFeed::testFeeds(uint notifyFlags)
|
|
{
|
|
#ifdef TO_REMOVE
|
|
if (!instance) {
|
|
return;
|
|
}
|
|
|
|
instance->ui->feedWidget->enableCountChangedSignal(false);
|
|
|
|
uint pos = 0;
|
|
|
|
while (notifyFlags) {
|
|
uint type = notifyFlags & (1 << pos);
|
|
notifyFlags &= ~(1 << pos);
|
|
++pos;
|
|
|
|
RsFeedItem fi;
|
|
}
|
|
|
|
instance->ui->feedWidget->enableCountChangedSignal(true);
|
|
|
|
instance->sendNewsFeedChanged();
|
|
#endif
|
|
std::cerr << "(EE) testFeeds() is currently disabled" << std::endl;
|
|
}
|
|
|
|
void NewsFeed::testFeed(FeedNotify *feedNotify)
|
|
{
|
|
if (!instance) {
|
|
return;
|
|
}
|
|
|
|
if (!feedNotify) {
|
|
return;
|
|
}
|
|
|
|
FeedItem *feedItem = feedNotify->testFeedItem(instance);
|
|
if (!feedItem) {
|
|
return;
|
|
}
|
|
|
|
instance->addFeedItem(feedItem);
|
|
}
|
|
|
|
void NewsFeed::addFeedItem(FeedItem *item)
|
|
{
|
|
static const int MAX_FEEDITEM_COUNT = 500 ;
|
|
|
|
item->setAttribute(Qt::WA_DeleteOnClose, true);
|
|
|
|
// costly, but not really a problem here
|
|
int feedItemCount;
|
|
bool fromTop = (ui->sortComboBox->itemData(ui->sortComboBox->currentIndex()).toInt() == Qt::AscendingOrder);
|
|
|
|
while ((feedItemCount = ui->feedWidget->feedItemCount()) >= MAX_FEEDITEM_COUNT) {
|
|
FeedItem *feedItem = ui->feedWidget->feedItem(fromTop ? 0 : feedItemCount - 1);
|
|
if (!feedItem) {
|
|
break;
|
|
}
|
|
|
|
ui->feedWidget->removeFeedItem(feedItem);
|
|
}
|
|
|
|
ui->feedWidget->addFeedItem(item, ROLE_RECEIVED, QDateTime::currentDateTime());
|
|
sendNewsFeedChanged();
|
|
}
|
|
|
|
void NewsFeed::addFeedItemIfUnique(FeedItem *item, bool replace)
|
|
{
|
|
FeedItem *feedItem = ui->feedWidget->findFeedItem(item->uniqueIdentifier());
|
|
|
|
if (feedItem)
|
|
{
|
|
if (!replace)
|
|
{
|
|
delete item;
|
|
return;
|
|
}
|
|
|
|
ui->feedWidget->removeFeedItem(feedItem);
|
|
}
|
|
|
|
addFeedItem(item);
|
|
//sendNewsFeedChanged(); //Already done by addFeedItem()
|
|
}
|
|
|
|
void NewsFeed::remUniqueFeedItem(FeedItem *item)
|
|
{
|
|
//FeedItem *feedItem = ui->feedWidget->findFeedItem(item->uniqueIdentifier());
|
|
|
|
ui->feedWidget->removeFeedItem(item);
|
|
delete item;
|
|
}
|
|
|
|
/* FeedHolder Functions (for FeedItem functionality) */
|
|
QScrollArea *NewsFeed::getScrollArea()
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
void NewsFeed::deleteFeedItem(FeedItem *item, uint32_t /*type*/)
|
|
{
|
|
#ifdef NEWS_DEBUG
|
|
std::cerr << "NewsFeed::deleteFeedItem()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
if (item) {
|
|
ui->feedWidget->removeFeedItem(item);
|
|
item->close ();
|
|
}
|
|
}
|
|
|
|
void NewsFeed::openChat(const RsPeerId &peerId)
|
|
{
|
|
#ifdef NEWS_DEBUG
|
|
std::cerr << "NewsFeed::openChat()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
ChatDialog::chatFriend(ChatId(peerId));
|
|
}
|
|
|
|
void NewsFeed::openComments(uint32_t /*type*/, const RsGxsGroupId &/*groupId*/, const QVector<RsGxsMessageId> &/*versions*/,const RsGxsMessageId &/*msgId*/, const QString &/*title*/)
|
|
{
|
|
std::cerr << "NewsFeed::openComments() Not Handled Yet";
|
|
std::cerr << std::endl;
|
|
}
|
|
|
|
static void sendNewsFeedChangedCallback(FeedItem *feedItem, void *data)
|
|
{
|
|
if (dynamic_cast<PeerItem*>(feedItem) == NULL) {
|
|
/* don't count PeerItem's */
|
|
++(*((int*) data));
|
|
}
|
|
}
|
|
|
|
void NewsFeed::sendNewsFeedChanged()
|
|
{
|
|
int count = 0;
|
|
ui->feedWidget->withAll(sendNewsFeedChangedCallback, &count);
|
|
|
|
emit newsFeedChanged(count);
|
|
}
|
|
|
|
void NewsFeed::feedoptions()
|
|
{
|
|
SettingsPage::showYourself(this, SettingsPage::Notify);
|
|
}
|