mirror of
https://github.com/RetroShare/RetroShare.git
synced 2024-12-16 11:24:24 -05:00
1233 lines
39 KiB
C++
1233 lines
39 KiB
C++
/****************************************************************
|
|
* RetroShare is distributed under the following license:
|
|
*
|
|
* Copyright (C) 2008 Robert Fernie
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
****************************************************************/
|
|
|
|
#include <QMenu>
|
|
#include <QMessageBox>
|
|
#include <QToolButton>
|
|
|
|
#include "GxsGroupFrameDialog.h"
|
|
#include "ui_GxsGroupFrameDialog.h"
|
|
#include "GxsMessageFrameWidget.h"
|
|
|
|
#include "gui/settings/rsharesettings.h"
|
|
#include "gui/RetroShareLink.h"
|
|
#include "gui/gxs/GxsGroupShareKey.h"
|
|
#include "gui/common/RSTreeWidget.h"
|
|
#include "gui/notifyqt.h"
|
|
#include "gui/common/UIStateHelper.h"
|
|
#include "GxsCommentDialog.h"
|
|
|
|
//#define DEBUG_GROUPFRAMEDIALOG
|
|
|
|
/* Images for TreeWidget */
|
|
#define IMAGE_SUBSCRIBE ":/images/edit_add24.png"
|
|
#define IMAGE_UNSUBSCRIBE ":/images/cancel.png"
|
|
#define IMAGE_INFO ":/images/info16.png"
|
|
//#define IMAGE_GROUPAUTHD ":/images/konv_message2.png"
|
|
#define IMAGE_COPYLINK ":/images/copyrslink.png"
|
|
#define IMAGE_EDIT ":/images/edit_16.png"
|
|
#define IMAGE_SHARE ":/images/share-icon-16.png"
|
|
#define IMAGE_TABNEW ":/images/tab-new.png"
|
|
#define IMAGE_DELETE ":/images/delete.png"
|
|
#define IMAGE_RETRIEVE ":/images/edit_add24.png"
|
|
#define IMAGE_COMMENT ""
|
|
|
|
#define TOKEN_TYPE_GROUP_SUMMARY 1
|
|
//#define TOKEN_TYPE_SUBSCRIBE_CHANGE 2
|
|
//#define TOKEN_TYPE_CURRENTGROUP 3
|
|
#define TOKEN_TYPE_STATISTICS 4
|
|
|
|
#define MAX_COMMENT_TITLE 32
|
|
|
|
/*
|
|
* Transformation Notes:
|
|
* there are still a couple of things that the new groups differ from Old version.
|
|
* these will need to be addressed in the future.
|
|
* -> Child TS (for sorting) is not handled by GXS, this will probably have to be done in the GUI.
|
|
* -> Need to handle IDs properly.
|
|
* -> Much more to do.
|
|
*/
|
|
|
|
/** Constructor */
|
|
GxsGroupFrameDialog::GxsGroupFrameDialog(RsGxsIfaceHelper *ifaceImpl, QWidget *parent,bool allow_dist_sync)
|
|
: RsGxsUpdateBroadcastPage(ifaceImpl, parent)
|
|
{
|
|
/* Invoke the Qt Designer generated object setup routine */
|
|
ui = new Ui::GxsGroupFrameDialog();
|
|
ui->setupUi(this);
|
|
|
|
mInitialized = false;
|
|
mDistSyncAllowed = allow_dist_sync;
|
|
mInFill = false;
|
|
mCountChildMsgs = false;
|
|
mYourGroups = NULL;
|
|
mSubscribedGroups = NULL;
|
|
mPopularGroups = NULL;
|
|
mOtherGroups = NULL;
|
|
mMessageWidget = NULL;
|
|
|
|
/* Setup Queue */
|
|
mInterface = ifaceImpl;
|
|
mTokenService = mInterface->getTokenService();
|
|
mTokenQueue = new TokenQueue(mInterface->getTokenService(), this);
|
|
|
|
/* Setup UI helper */
|
|
mStateHelper = new UIStateHelper(this);
|
|
|
|
mStateHelper->addWidget(TOKEN_TYPE_GROUP_SUMMARY, ui->loadingLabel, UISTATE_LOADING_VISIBLE);
|
|
|
|
connect(ui->groupTreeWidget, SIGNAL(treeCustomContextMenuRequested(QPoint)), this, SLOT(groupTreeCustomPopupMenu(QPoint)));
|
|
connect(ui->groupTreeWidget, SIGNAL(treeCurrentItemChanged(QString)), this, SLOT(changedCurrentGroup(QString)));
|
|
connect(ui->groupTreeWidget->treeWidget(), SIGNAL(signalMouseMiddleButtonClicked(QTreeWidgetItem*)), this, SLOT(groupTreeMiddleButtonClicked(QTreeWidgetItem*)));
|
|
connect(ui->messageTabWidget, SIGNAL(tabCloseRequested(int)), this, SLOT(messageTabCloseRequested(int)));
|
|
connect(ui->messageTabWidget, SIGNAL(currentChanged(int)), this, SLOT(messageTabChanged(int)));
|
|
|
|
connect(ui->todoPushButton, SIGNAL(clicked()), this, SLOT(todo()));
|
|
|
|
ui->groupTreeWidget->setDistSearchVisible(allow_dist_sync) ;
|
|
|
|
if(allow_dist_sync)
|
|
connect(ui->groupTreeWidget, SIGNAL(distantSearchRequested(const QString&)), this, SLOT(searchNetwork(const QString&)));
|
|
|
|
/* Set initial size the splitter */
|
|
ui->splitter->setStretchFactor(0, 0);
|
|
ui->splitter->setStretchFactor(1, 1);
|
|
|
|
QList<int> sizes;
|
|
sizes << 300 << width(); // Qt calculates the right sizes
|
|
ui->splitter->setSizes(sizes);
|
|
|
|
#ifndef UNFINISHED
|
|
ui->todoPushButton->hide();
|
|
#endif
|
|
}
|
|
|
|
GxsGroupFrameDialog::~GxsGroupFrameDialog()
|
|
{
|
|
// save settings
|
|
processSettings(false);
|
|
|
|
delete(mTokenQueue);
|
|
delete(ui);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::getGroupList(std::map<RsGxsGroupId, RsGroupMetaData> &group_list)
|
|
{
|
|
group_list = mCachedGroupMetas ;
|
|
|
|
if(group_list.empty())
|
|
requestGroupSummary();
|
|
}
|
|
void GxsGroupFrameDialog::initUi()
|
|
{
|
|
registerHelpButton(ui->helpButton, getHelpString(),pageName()) ;
|
|
|
|
ui->titleBarPixmap->setPixmap(QPixmap(icon(ICON_NAME)));
|
|
ui->titleBarLabel->setText(text(TEXT_NAME));
|
|
|
|
/* Initialize group tree */
|
|
QToolButton *newGroupButton = new QToolButton(this);
|
|
newGroupButton->setIcon(QIcon(icon(ICON_NEW)));
|
|
newGroupButton->setToolTip(text(TEXT_NEW));
|
|
connect(newGroupButton, SIGNAL(clicked()), this, SLOT(newGroup()));
|
|
ui->groupTreeWidget->addToolButton(newGroupButton);
|
|
|
|
/* Create group tree */
|
|
mYourGroups = ui->groupTreeWidget->addCategoryItem(text(TEXT_YOUR_GROUP), QIcon(icon(ICON_YOUR_GROUP)), true);
|
|
mSubscribedGroups = ui->groupTreeWidget->addCategoryItem(text(TEXT_SUBSCRIBED_GROUP), QIcon(icon(ICON_SUBSCRIBED_GROUP)), true);
|
|
mPopularGroups = ui->groupTreeWidget->addCategoryItem(text(TEXT_POPULAR_GROUP), QIcon(icon(ICON_POPULAR_GROUP)), false);
|
|
mOtherGroups = ui->groupTreeWidget->addCategoryItem(text(TEXT_OTHER_GROUP), QIcon(icon(ICON_OTHER_GROUP)), false);
|
|
|
|
if (text(TEXT_TODO).isEmpty()) {
|
|
ui->todoPushButton->hide();
|
|
}
|
|
|
|
// load settings
|
|
mSettingsName = settingsGroupName();
|
|
processSettings(true);
|
|
|
|
if (groupFrameSettingsType() != GroupFrameSettings::Nothing) {
|
|
connect(NotifyQt::getInstance(), SIGNAL(settingsChanged()), this, SLOT(settingsChanged()));
|
|
settingsChanged();
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::showEvent(QShowEvent *event)
|
|
{
|
|
if (!mInitialized) {
|
|
/* Problem: virtual methods cannot be used in constructor */
|
|
mInitialized = true;
|
|
|
|
initUi();
|
|
}
|
|
|
|
RsGxsUpdateBroadcastPage::showEvent(event);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::processSettings(bool load)
|
|
{
|
|
if (mSettingsName.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
Settings->beginGroup(mSettingsName);
|
|
|
|
if (load) {
|
|
// load settings
|
|
|
|
// state of splitter
|
|
ui->splitter->restoreState(Settings->value("Splitter").toByteArray());
|
|
} else {
|
|
// save settings
|
|
|
|
// state of splitter
|
|
Settings->setValue("Splitter", ui->splitter->saveState());
|
|
}
|
|
|
|
ui->groupTreeWidget->processSettings(load);
|
|
|
|
Settings->endGroup();
|
|
}
|
|
|
|
void GxsGroupFrameDialog::settingsChanged()
|
|
{
|
|
GroupFrameSettings groupFrameSettings;
|
|
if (Settings->getGroupFrameSettings(groupFrameSettingsType(), groupFrameSettings)) {
|
|
setSingleTab(!groupFrameSettings.mOpenAllInNewTab);
|
|
setHideTabBarWithOneTab(groupFrameSettings.mHideTabBarWithOneTab);
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::setSingleTab(bool singleTab)
|
|
{
|
|
if (singleTab) {
|
|
if (!mMessageWidget) {
|
|
mMessageWidget = createMessageWidget(RsGxsGroupId());
|
|
// remove close button of the the first tab
|
|
ui->messageTabWidget->hideCloseButton(ui->messageTabWidget->indexOf(mMessageWidget));
|
|
}
|
|
} else {
|
|
if (mMessageWidget) {
|
|
delete(mMessageWidget);
|
|
mMessageWidget = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::setHideTabBarWithOneTab(bool hideTabBarWithOneTab)
|
|
{
|
|
ui->messageTabWidget->setHideTabBarWithOneTab(hideTabBarWithOneTab);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::updateDisplay(bool complete)
|
|
{
|
|
if (complete || !getGrpIds().empty() || !getGrpIdsMeta().empty()) {
|
|
/* Update group list */
|
|
requestGroupSummary();
|
|
} else {
|
|
/* Update all groups of changed messages */
|
|
std::map<RsGxsGroupId, std::set<RsGxsMessageId> > msgIds;
|
|
getAllMsgIds(msgIds);
|
|
|
|
for (auto msgIt = msgIds.begin(); msgIt != msgIds.end(); ++msgIt) {
|
|
updateMessageSummaryList(msgIt->first);
|
|
}
|
|
}
|
|
|
|
updateSearchResults() ;
|
|
}
|
|
|
|
void GxsGroupFrameDialog::updateSearchResults()
|
|
{
|
|
const std::set<TurtleRequestId>& reqs = getSearchResults();
|
|
|
|
for(auto it(reqs.begin());it!=reqs.end();++it)
|
|
{
|
|
std::cerr << "updating search ID " << std::hex << *it << std::dec << std::endl;
|
|
|
|
std::map<RsGxsGroupId,RsGxsGroupSummary> group_infos;
|
|
|
|
getDistantSearchResults(*it,group_infos) ;
|
|
|
|
std::cerr << "retrieved " << std::endl;
|
|
|
|
auto it2 = mSearchGroupsItems.find(*it);
|
|
|
|
if(mSearchGroupsItems.end() == it2)
|
|
{
|
|
std::cerr << "GxsGroupFrameDialog::updateSearchResults(): received result notification for req " << std::hex << *it << std::dec << " but no item present!" << std::endl;
|
|
continue ; // we could create the item just as well but since this situation is not supposed to happen, I prefer to make this a failure case.
|
|
}
|
|
|
|
QList<GroupItemInfo> group_items ;
|
|
|
|
for(auto it3(group_infos.begin());it3!=group_infos.end();++it3)
|
|
if(mCachedGroupMetas.find(it3->first) == mCachedGroupMetas.end())
|
|
{
|
|
std::cerr << " adding new group " << it3->first << " " << it3->second.group_id << " \"" << it3->second.group_name << "\"" << std::endl;
|
|
|
|
GroupItemInfo i ;
|
|
i.id = QString(it3->second.group_id.toStdString().c_str()) ;
|
|
i.name = QString::fromUtf8(it3->second.group_name.c_str()) ;
|
|
i.description = QString::fromUtf8(it3->second.group_description.c_str()) ;
|
|
i.popularity = 0; // could be set to the number of hits
|
|
i.lastpost = QDateTime::fromTime_t(it3->second.last_message_ts);
|
|
i.subscribeFlags = 0; // irrelevant here
|
|
i.publishKey = false ; // IS_GROUP_PUBLISHER(groupInfo.mSubscribeFlags) ;
|
|
i.adminKey = false ; // IS_GROUP_ADMIN(groupInfo.mSubscribeFlags) ;
|
|
i.max_visible_posts = it3->second.number_of_messages ;
|
|
|
|
group_items.push_back(i);
|
|
}
|
|
|
|
ui->groupTreeWidget->fillGroupItems(it2->second, group_items);
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::todo()
|
|
{
|
|
QMessageBox::information(this, "Todo", text(TEXT_TODO));
|
|
}
|
|
|
|
void GxsGroupFrameDialog::removeCurrentSearch()
|
|
{
|
|
QAction *action = dynamic_cast<QAction*>(sender()) ;
|
|
|
|
if(!action)
|
|
return ;
|
|
|
|
TurtleRequestId search_request_id = action->data().toUInt();
|
|
|
|
auto it = mSearchGroupsItems.find(search_request_id) ;
|
|
|
|
if(it == mSearchGroupsItems.end())
|
|
return ;
|
|
|
|
ui->groupTreeWidget->removeSearchItem(it->second) ;
|
|
mSearchGroupsItems.erase(it);
|
|
|
|
mKnownGroups.erase(search_request_id);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::removeAllSearches()
|
|
{
|
|
for(auto it(mSearchGroupsItems.begin());it!=mSearchGroupsItems.end();++it)
|
|
ui->groupTreeWidget->removeSearchItem(it->second) ;
|
|
|
|
mSearchGroupsItems.clear();
|
|
mKnownGroups.clear();
|
|
}
|
|
void GxsGroupFrameDialog::groupTreeCustomPopupMenu(QPoint point)
|
|
{
|
|
// First separately handle the case of search top level items
|
|
|
|
TurtleRequestId search_request_id = 0 ;
|
|
|
|
if(ui->groupTreeWidget->isSearchRequestItem(point,search_request_id))
|
|
{
|
|
QMenu contextMnu(this);
|
|
|
|
contextMnu.addAction(QIcon(IMAGE_DELETE), tr("Remove this search"), this, SLOT(removeCurrentSearch()))->setData(search_request_id);
|
|
contextMnu.addAction(QIcon(IMAGE_DELETE), tr("Remove all searches"), this, SLOT(removeAllSearches()));
|
|
contextMnu.exec(QCursor::pos());
|
|
return ;
|
|
}
|
|
|
|
// Then check whether we have a searched item, or a normal group
|
|
|
|
QString group_id_s ;
|
|
|
|
if(ui->groupTreeWidget->isSearchRequestResult(point,group_id_s,search_request_id))
|
|
{
|
|
QMenu contextMnu(this);
|
|
|
|
contextMnu.addAction(QIcon(IMAGE_RETRIEVE), tr("Request data"), this, SLOT(distantRequestGroupData()))->setData(group_id_s);
|
|
contextMnu.exec(QCursor::pos());
|
|
return ;
|
|
}
|
|
|
|
QString id = ui->groupTreeWidget->itemIdAt(point);
|
|
if (id.isEmpty()) return;
|
|
|
|
mGroupId = RsGxsGroupId(id.toStdString());
|
|
int subscribeFlags = ui->groupTreeWidget->subscribeFlags(QString::fromStdString(mGroupId.toStdString()));
|
|
|
|
bool isAdmin = IS_GROUP_ADMIN(subscribeFlags);
|
|
bool isPublisher = IS_GROUP_PUBLISHER(subscribeFlags);
|
|
bool isSubscribed = IS_GROUP_SUBSCRIBED(subscribeFlags);
|
|
|
|
QMenu contextMnu(this);
|
|
QAction *action;
|
|
|
|
if (mMessageWidget) {
|
|
action = contextMnu.addAction(QIcon(IMAGE_TABNEW), tr("Open in new tab"), this, SLOT(openInNewTab()));
|
|
if (mGroupId.isNull() || messageWidget(mGroupId, true)) {
|
|
action->setEnabled(false);
|
|
}
|
|
}
|
|
|
|
if (isSubscribed) {
|
|
action = contextMnu.addAction(QIcon(IMAGE_UNSUBSCRIBE), tr("Unsubscribe"), this, SLOT(unsubscribeGroup()));
|
|
action->setEnabled (!mGroupId.isNull() && IS_GROUP_SUBSCRIBED(subscribeFlags));
|
|
} else {
|
|
action = contextMnu.addAction(QIcon(IMAGE_SUBSCRIBE), tr("Subscribe"), this, SLOT(subscribeGroup()));
|
|
action->setDisabled (mGroupId.isNull() || IS_GROUP_SUBSCRIBED(subscribeFlags));
|
|
}
|
|
|
|
contextMnu.addSeparator();
|
|
|
|
contextMnu.addAction(QIcon(icon(ICON_NEW)), text(TEXT_NEW), this, SLOT(newGroup()));
|
|
|
|
action = contextMnu.addAction(QIcon(IMAGE_INFO), tr("Show Details"), this, SLOT(showGroupDetails()));
|
|
action->setEnabled (!mGroupId.isNull());
|
|
|
|
action = contextMnu.addAction(QIcon(IMAGE_EDIT), tr("Edit Details"), this, SLOT(editGroupDetails()));
|
|
action->setEnabled (!mGroupId.isNull() && isAdmin);
|
|
|
|
uint32_t current_store_time = mInterface->getStoragePeriod(mGroupId)/86400 ;
|
|
uint32_t current_sync_time = mInterface->getSyncPeriod(mGroupId)/86400 ;
|
|
|
|
std::cerr << "Got sync=" << current_sync_time << ". store=" << current_store_time << std::endl;
|
|
QAction *actnn = NULL;
|
|
|
|
QMenu *ctxMenu2 = contextMnu.addMenu(tr("Synchronise posts of last...")) ;
|
|
actnn = ctxMenu2->addAction(tr(" 5 days" ),this,SLOT(setSyncPostsDelay())) ; actnn->setData(QVariant( 5)) ; if(current_sync_time == 5) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 2 weeks" ),this,SLOT(setSyncPostsDelay())) ; actnn->setData(QVariant( 15)) ; if(current_sync_time == 15) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 1 month" ),this,SLOT(setSyncPostsDelay())) ; actnn->setData(QVariant( 30)) ; if(current_sync_time == 30) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 3 months" ),this,SLOT(setSyncPostsDelay())) ; actnn->setData(QVariant( 90)) ; if(current_sync_time == 90) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 6 months" ),this,SLOT(setSyncPostsDelay())) ; actnn->setData(QVariant(180)) ; if(current_sync_time ==180) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 1 year " ),this,SLOT(setSyncPostsDelay())) ; actnn->setData(QVariant(372)) ; if(current_sync_time ==372) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" Indefinitly"),this,SLOT(setSyncPostsDelay())) ; actnn->setData(QVariant( 0)) ; if(current_sync_time == 0) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
|
|
ctxMenu2 = contextMnu.addMenu(tr("Store posts for at most...")) ;
|
|
actnn = ctxMenu2->addAction(tr(" 5 days" ),this,SLOT(setStorePostsDelay())) ; actnn->setData(QVariant( 5)) ; if(current_store_time == 5) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 2 weeks" ),this,SLOT(setStorePostsDelay())) ; actnn->setData(QVariant( 15)) ; if(current_store_time == 15) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 1 month" ),this,SLOT(setStorePostsDelay())) ; actnn->setData(QVariant( 30)) ; if(current_store_time == 30) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 3 months" ),this,SLOT(setStorePostsDelay())) ; actnn->setData(QVariant( 90)) ; if(current_store_time == 90) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 6 months" ),this,SLOT(setStorePostsDelay())) ; actnn->setData(QVariant(180)) ; if(current_store_time ==180) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" 1 year " ),this,SLOT(setStorePostsDelay())) ; actnn->setData(QVariant(372)) ; if(current_store_time ==372) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
actnn = ctxMenu2->addAction(tr(" Indefinitly"),this,SLOT(setStorePostsDelay())) ; actnn->setData(QVariant( 0)) ; if(current_store_time == 0) { actnn->setEnabled(false);actnn->setIcon(QIcon(":/images/start.png"));}
|
|
|
|
if (shareKeyType()) {
|
|
action = contextMnu.addAction(QIcon(IMAGE_SHARE), tr("Share publish permissions"), this, SLOT(sharePublishKey()));
|
|
action->setEnabled(!mGroupId.isNull() && isPublisher);
|
|
}
|
|
|
|
if (getLinkType() != RetroShareLink::TYPE_UNKNOWN) {
|
|
action = contextMnu.addAction(QIcon(IMAGE_COPYLINK), tr("Copy RetroShare Link"), this, SLOT(copyGroupLink()));
|
|
action->setEnabled(!mGroupId.isNull());
|
|
}
|
|
|
|
contextMnu.addSeparator();
|
|
|
|
action = contextMnu.addAction(QIcon(":/images/message-mail-read.png"), tr("Mark all as read"), this, SLOT(markMsgAsRead()));
|
|
action->setEnabled (!mGroupId.isNull() && isSubscribed);
|
|
|
|
action = contextMnu.addAction(QIcon(":/images/message-mail.png"), tr("Mark all as unread"), this, SLOT(markMsgAsUnread()));
|
|
action->setEnabled (!mGroupId.isNull() && isSubscribed);
|
|
|
|
/* Add special actions */
|
|
QList<QAction*> actions;
|
|
groupTreeCustomActions(mGroupId, subscribeFlags, actions);
|
|
if (!actions.isEmpty()) {
|
|
contextMnu.addSeparator();
|
|
contextMnu.addActions(actions);
|
|
}
|
|
|
|
//Add Standard Menu
|
|
ui->groupTreeWidget->treeWidget()->createStandardContextMenu(&contextMnu);
|
|
|
|
contextMnu.exec(QCursor::pos());
|
|
}
|
|
|
|
void GxsGroupFrameDialog::setStorePostsDelay()
|
|
{
|
|
QAction *action = dynamic_cast<QAction*>(sender()) ;
|
|
|
|
if(!action || mGroupId.isNull())
|
|
{
|
|
std::cerr << "(EE) Cannot find action/group that called me! Group is " << mGroupId << ", action is " << (void*)action << " " << __PRETTY_FUNCTION__ << std::endl;
|
|
return;
|
|
}
|
|
|
|
uint32_t duration = action->data().toUInt() ;
|
|
|
|
std::cerr << "Data is " << duration << std::endl;
|
|
|
|
mInterface->setStoragePeriod(mGroupId,duration * 86400) ;
|
|
|
|
// If the sync is larger, we reduce it. No need to sync more than we store. The machinery below also takes care of this.
|
|
//
|
|
uint32_t sync_period = mInterface->getSyncPeriod(mGroupId);
|
|
|
|
if(duration > 0) // the >0 test is to discard the indefinitly test. Basically, if we store for less than indefinitly, the sync is reduced accordingly.
|
|
{
|
|
if(sync_period == 0 || sync_period > duration*86400)
|
|
{
|
|
mInterface->setSyncPeriod(mGroupId,duration * 86400) ;
|
|
|
|
std::cerr << "(II) auto adjusting sync period to " << duration<< " days as well." << std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void GxsGroupFrameDialog::setSyncPostsDelay()
|
|
{
|
|
QAction *action = dynamic_cast<QAction*>(sender()) ;
|
|
|
|
if(!action || mGroupId.isNull())
|
|
{
|
|
std::cerr << "(EE) Cannot find action/group that called me! Group is " << mGroupId << ", action is " << (void*)action << " " << __PRETTY_FUNCTION__ << std::endl;
|
|
return;
|
|
}
|
|
|
|
uint32_t duration = action->data().toUInt() ;
|
|
|
|
std::cerr << "Data is " << duration << std::endl;
|
|
|
|
mInterface->setSyncPeriod(mGroupId,duration * 86400) ;
|
|
|
|
// If the store is smaller, we increase it accordingly. No need to sync more than we store. The machinery below also takes care of this.
|
|
//
|
|
uint32_t store_period = mInterface->getStoragePeriod(mGroupId);
|
|
|
|
if(duration == 0)
|
|
mInterface->setStoragePeriod(mGroupId,duration * 86400) ; // indefinite sync => indefinite storage
|
|
else
|
|
{
|
|
if(store_period != 0 && store_period < duration*86400)
|
|
{
|
|
mInterface->setStoragePeriod(mGroupId,duration * 86400) ; // indefinite sync => indefinite storage
|
|
std::cerr << "(II) auto adjusting storage period to " << duration<< " days as well." << std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::restoreGroupKeys(void)
|
|
{
|
|
QMessageBox::warning(this, "RetroShare", "ToDo");
|
|
|
|
#ifdef TOGXS
|
|
mInterface->groupRestoreKeys(mGroupId);
|
|
#endif
|
|
}
|
|
|
|
void GxsGroupFrameDialog::newGroup()
|
|
{
|
|
GxsGroupDialog *dialog = createNewGroupDialog(mTokenQueue);
|
|
if (!dialog) {
|
|
return;
|
|
}
|
|
|
|
dialog->exec();
|
|
delete(dialog);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::subscribeGroup()
|
|
{
|
|
groupSubscribe(true);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::unsubscribeGroup()
|
|
{
|
|
groupSubscribe(false);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::groupSubscribe(bool subscribe)
|
|
{
|
|
if (mGroupId.isNull()) {
|
|
return;
|
|
}
|
|
|
|
uint32_t token;
|
|
mInterface->subscribeToGroup(token, mGroupId, subscribe);
|
|
// Replaced by meta data changed
|
|
// mTokenQueue->queueRequest(token, 0, RS_TOKREQ_ANSTYPE_ACK, TOKEN_TYPE_SUBSCRIBE_CHANGE);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::showGroupDetails()
|
|
{
|
|
if (mGroupId.isNull()) {
|
|
return;
|
|
}
|
|
|
|
GxsGroupDialog *dialog = createGroupDialog(mTokenQueue, mInterface->getTokenService(), GxsGroupDialog::MODE_SHOW, mGroupId);
|
|
if (!dialog) {
|
|
return;
|
|
}
|
|
|
|
dialog->exec();
|
|
delete(dialog);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::editGroupDetails()
|
|
{
|
|
if (mGroupId.isNull()) {
|
|
return;
|
|
}
|
|
|
|
GxsGroupDialog *dialog = createGroupDialog(mTokenQueue, mInterface->getTokenService(), GxsGroupDialog::MODE_EDIT, mGroupId);
|
|
if (!dialog) {
|
|
return;
|
|
}
|
|
|
|
dialog->exec();
|
|
delete(dialog);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::copyGroupLink()
|
|
{
|
|
if (mGroupId.isNull()) {
|
|
return;
|
|
}
|
|
|
|
QString name;
|
|
if(!getCurrentGroupName(name)) return;
|
|
|
|
RetroShareLink link = RetroShareLink::createGxsGroupLink(getLinkType(), mGroupId, name);
|
|
|
|
if (link.valid()) {
|
|
QList<RetroShareLink> urls;
|
|
urls.push_back(link);
|
|
RSLinkClipboard::copyLinks(urls);
|
|
}
|
|
}
|
|
|
|
bool GxsGroupFrameDialog::getCurrentGroupName(QString& name)
|
|
{
|
|
return ui->groupTreeWidget->getGroupName(QString::fromStdString(mGroupId.toStdString()), name);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::markMsgAsRead()
|
|
{
|
|
GxsMessageFrameWidget *msgWidget = messageWidget(mGroupId, false);
|
|
if (msgWidget) {
|
|
msgWidget->setAllMessagesRead(true);
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::markMsgAsUnread()
|
|
{
|
|
GxsMessageFrameWidget *msgWidget = messageWidget(mGroupId, false);
|
|
if (msgWidget) {
|
|
msgWidget->setAllMessagesRead(false);
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::sharePublishKey()
|
|
{
|
|
if (mGroupId.isNull()) {
|
|
return;
|
|
}
|
|
|
|
// QMessageBox::warning(this, "", "ToDo");
|
|
|
|
GroupShareKey shareUi(this, mGroupId, shareKeyType());
|
|
shareUi.exec();
|
|
}
|
|
|
|
void GxsGroupFrameDialog::loadComment(const RsGxsGroupId &grpId, const QVector<RsGxsMessageId>& msg_versions, const RsGxsMessageId &most_recent_msgId, const QString &title)
|
|
{
|
|
RsGxsCommentService *commentService = getCommentService();
|
|
if (!commentService) {
|
|
/* No comment service available */
|
|
return;
|
|
}
|
|
|
|
GxsCommentDialog *commentDialog = commentWidget(most_recent_msgId);
|
|
|
|
if (!commentDialog) {
|
|
QString comments = title;
|
|
if (title.length() > MAX_COMMENT_TITLE)
|
|
{
|
|
comments.truncate(MAX_COMMENT_TITLE - 3);
|
|
comments += "...";
|
|
}
|
|
|
|
commentDialog = new GxsCommentDialog(this, mInterface->getTokenService(), commentService);
|
|
|
|
QWidget *commentHeader = createCommentHeaderWidget(grpId, most_recent_msgId);
|
|
if (commentHeader) {
|
|
commentDialog->setCommentHeader(commentHeader);
|
|
}
|
|
|
|
std::set<RsGxsMessageId> msgv;
|
|
for(int i=0;i<msg_versions.size();++i)
|
|
msgv.insert(msg_versions[i]);
|
|
|
|
commentDialog->commentLoad(grpId, msgv,most_recent_msgId);
|
|
|
|
int index = ui->messageTabWidget->addTab(commentDialog, comments);
|
|
ui->messageTabWidget->setTabIcon(index, QIcon(IMAGE_COMMENT));
|
|
}
|
|
|
|
ui->messageTabWidget->setCurrentWidget(commentDialog);
|
|
}
|
|
|
|
bool GxsGroupFrameDialog::navigate(const RsGxsGroupId &groupId, const RsGxsMessageId& msgId)
|
|
{
|
|
if (groupId.isNull()) {
|
|
return false;
|
|
}
|
|
|
|
if (mStateHelper->isLoading(TOKEN_TYPE_GROUP_SUMMARY)) {
|
|
mNavigatePendingGroupId = groupId;
|
|
mNavigatePendingMsgId = msgId;
|
|
|
|
/* No information if group is available */
|
|
return true;
|
|
}
|
|
|
|
QString groupIdString = QString::fromStdString(groupId.toStdString());
|
|
if (ui->groupTreeWidget->activateId(groupIdString, msgId.isNull()) == NULL) {
|
|
return false;
|
|
}
|
|
|
|
changedCurrentGroup(groupIdString);
|
|
|
|
/* search exisiting tab */
|
|
GxsMessageFrameWidget *msgWidget = messageWidget(mGroupId, false);
|
|
if (!msgWidget) {
|
|
return false;
|
|
}
|
|
|
|
if (msgId.isNull()) {
|
|
return true;
|
|
}
|
|
|
|
return msgWidget->navigate(msgId);
|
|
}
|
|
|
|
GxsMessageFrameWidget *GxsGroupFrameDialog::messageWidget(const RsGxsGroupId &groupId, bool ownTab)
|
|
{
|
|
int tabCount = ui->messageTabWidget->count();
|
|
for (int index = 0; index < tabCount; ++index) {
|
|
GxsMessageFrameWidget *childWidget = dynamic_cast<GxsMessageFrameWidget*>(ui->messageTabWidget->widget(index));
|
|
if (ownTab && mMessageWidget && childWidget == mMessageWidget) {
|
|
continue;
|
|
}
|
|
if (childWidget && childWidget->groupId() == groupId) {
|
|
return childWidget;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
GxsMessageFrameWidget *GxsGroupFrameDialog::createMessageWidget(const RsGxsGroupId &groupId)
|
|
{
|
|
GxsMessageFrameWidget *msgWidget = createMessageFrameWidget(groupId);
|
|
if (!msgWidget) {
|
|
return NULL;
|
|
}
|
|
|
|
int index = ui->messageTabWidget->addTab(msgWidget, msgWidget->groupName(true));
|
|
ui->messageTabWidget->setTabIcon(index, msgWidget->groupIcon());
|
|
|
|
connect(msgWidget, SIGNAL(groupChanged(QWidget*)), this, SLOT(messageTabInfoChanged(QWidget*)));
|
|
connect(msgWidget, SIGNAL(waitingChanged(QWidget*)), this, SLOT(messageTabWaitingChanged(QWidget*)));
|
|
connect(msgWidget, SIGNAL(loadComment(RsGxsGroupId,QVector<RsGxsMessageId>,RsGxsMessageId,QString)), this, SLOT(loadComment(RsGxsGroupId,QVector<RsGxsMessageId>,RsGxsMessageId,QString)));
|
|
|
|
return msgWidget;
|
|
}
|
|
|
|
GxsCommentDialog *GxsGroupFrameDialog::commentWidget(const RsGxsMessageId& msgId)
|
|
{
|
|
int tabCount = ui->messageTabWidget->count();
|
|
for (int index = 0; index < tabCount; ++index) {
|
|
GxsCommentDialog *childWidget = dynamic_cast<GxsCommentDialog*>(ui->messageTabWidget->widget(index));
|
|
if (childWidget && childWidget->messageId() == msgId) {
|
|
return childWidget;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void GxsGroupFrameDialog::changedCurrentGroup(const QString &groupId)
|
|
{
|
|
if (mInFill) {
|
|
return;
|
|
}
|
|
|
|
if (groupId.isEmpty()) {
|
|
if (mMessageWidget) {
|
|
mMessageWidget->setGroupId(RsGxsGroupId());
|
|
ui->messageTabWidget->setCurrentWidget(mMessageWidget);
|
|
}
|
|
return;
|
|
}
|
|
|
|
mGroupId = RsGxsGroupId(groupId.toStdString());
|
|
if (mGroupId.isNull()) {
|
|
return;
|
|
}
|
|
|
|
/* search exisiting tab */
|
|
GxsMessageFrameWidget *msgWidget = messageWidget(mGroupId, true);
|
|
|
|
if (!msgWidget) {
|
|
if (mMessageWidget) {
|
|
/* not found, use standard tab */
|
|
msgWidget = mMessageWidget;
|
|
msgWidget->setGroupId(mGroupId);
|
|
} else {
|
|
/* create new tab */
|
|
msgWidget = createMessageWidget(mGroupId);
|
|
}
|
|
}
|
|
|
|
ui->messageTabWidget->setCurrentWidget(msgWidget);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::groupTreeMiddleButtonClicked(QTreeWidgetItem *item)
|
|
{
|
|
openGroupInNewTab(RsGxsGroupId(ui->groupTreeWidget->itemId(item).toStdString()));
|
|
}
|
|
|
|
void GxsGroupFrameDialog::openInNewTab()
|
|
{
|
|
openGroupInNewTab(mGroupId);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::openGroupInNewTab(const RsGxsGroupId &groupId)
|
|
{
|
|
if (groupId.isNull()) {
|
|
return;
|
|
}
|
|
|
|
/* search exisiting tab */
|
|
GxsMessageFrameWidget *msgWidget = messageWidget(groupId, true);
|
|
if (!msgWidget) {
|
|
/* not found, create new tab */
|
|
msgWidget = createMessageWidget(groupId);
|
|
}
|
|
|
|
ui->messageTabWidget->setCurrentWidget(msgWidget);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::messageTabCloseRequested(int index)
|
|
{
|
|
QWidget *widget = ui->messageTabWidget->widget(index);
|
|
if (!widget) {
|
|
return;
|
|
}
|
|
|
|
GxsMessageFrameWidget *msgWidget = dynamic_cast<GxsMessageFrameWidget*>(widget);
|
|
if (msgWidget && msgWidget == mMessageWidget) {
|
|
/* Don't close single tab */
|
|
return;
|
|
}
|
|
|
|
delete(widget);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::messageTabChanged(int index)
|
|
{
|
|
GxsMessageFrameWidget *msgWidget = dynamic_cast<GxsMessageFrameWidget*>(ui->messageTabWidget->widget(index));
|
|
if (!msgWidget) {
|
|
return;
|
|
}
|
|
|
|
ui->groupTreeWidget->activateId(QString::fromStdString(msgWidget->groupId().toStdString()), false);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::messageTabInfoChanged(QWidget *widget)
|
|
{
|
|
int index = ui->messageTabWidget->indexOf(widget);
|
|
if (index < 0) {
|
|
return;
|
|
}
|
|
|
|
GxsMessageFrameWidget *msgWidget = dynamic_cast<GxsMessageFrameWidget*>(ui->messageTabWidget->widget(index));
|
|
if (!msgWidget) {
|
|
return;
|
|
}
|
|
|
|
ui->messageTabWidget->setTabText(index, msgWidget->groupName(true));
|
|
ui->messageTabWidget->setTabIcon(index, msgWidget->groupIcon());
|
|
}
|
|
|
|
void GxsGroupFrameDialog::messageTabWaitingChanged(QWidget *widget)
|
|
{
|
|
int index = ui->messageTabWidget->indexOf(widget);
|
|
if (index < 0) {
|
|
return;
|
|
}
|
|
|
|
GxsMessageFrameWidget *msgWidget = dynamic_cast<GxsMessageFrameWidget*>(ui->messageTabWidget->widget(index));
|
|
if (!msgWidget) {
|
|
return;
|
|
}
|
|
|
|
ui->groupTreeWidget->setWaiting(QString::fromStdString(msgWidget->groupId().toStdString()), msgWidget->isWaiting());
|
|
}
|
|
|
|
///***** INSERT GROUP LISTS *****/
|
|
void GxsGroupFrameDialog::groupInfoToGroupItemInfo(const RsGroupMetaData &groupInfo, GroupItemInfo &groupItemInfo, const RsUserdata */*userdata*/)
|
|
{
|
|
groupItemInfo.id = QString::fromStdString(groupInfo.mGroupId.toStdString());
|
|
groupItemInfo.name = QString::fromUtf8(groupInfo.mGroupName.c_str());
|
|
groupItemInfo.popularity = groupInfo.mPop;
|
|
groupItemInfo.lastpost = QDateTime::fromTime_t(groupInfo.mLastPost);
|
|
groupItemInfo.subscribeFlags = groupInfo.mSubscribeFlags;
|
|
groupItemInfo.publishKey = IS_GROUP_PUBLISHER(groupInfo.mSubscribeFlags) ;
|
|
groupItemInfo.adminKey = IS_GROUP_ADMIN(groupInfo.mSubscribeFlags) ;
|
|
groupItemInfo.max_visible_posts = groupInfo.mVisibleMsgCount ;
|
|
|
|
#if TOGXS
|
|
if (groupInfo.mGroupFlags & RS_DISTRIB_AUTHEN_REQ) {
|
|
groupItemInfo.name += " (" + tr("AUTHD") + ")";
|
|
groupItemInfo.icon = QIcon(IMAGE_GROUPAUTHD);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
groupItemInfo.icon = QIcon(icon(ICON_DEFAULT));
|
|
}
|
|
}
|
|
|
|
void GxsGroupFrameDialog::insertGroupsData(const std::map<RsGxsGroupId,RsGroupMetaData> &groupList, const RsUserdata *userdata)
|
|
{
|
|
if (!mInitialized) {
|
|
return;
|
|
}
|
|
|
|
mInFill = true;
|
|
|
|
QList<GroupItemInfo> adminList;
|
|
QList<GroupItemInfo> subList;
|
|
QList<GroupItemInfo> popList;
|
|
QList<GroupItemInfo> otherList;
|
|
std::multimap<uint32_t, GroupItemInfo> popMap;
|
|
|
|
for (auto it = groupList.begin(); it != groupList.end(); ++it) {
|
|
/* sort it into Publish (Own), Subscribed, Popular and Other */
|
|
uint32_t flags = it->second.mSubscribeFlags;
|
|
|
|
GroupItemInfo groupItemInfo;
|
|
groupInfoToGroupItemInfo(it->second, groupItemInfo, userdata);
|
|
|
|
if (IS_GROUP_SUBSCRIBED(flags))
|
|
{
|
|
if (IS_GROUP_ADMIN(flags))
|
|
{
|
|
adminList.push_back(groupItemInfo);
|
|
}
|
|
else
|
|
{
|
|
/* subscribed group */
|
|
subList.push_back(groupItemInfo);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//popMap.insert(std::make_pair(it->mPop, groupItemInfo)); /* rate the others by popularity */
|
|
popMap.insert(std::make_pair(it->second.mLastPost, groupItemInfo)); /* rate the others by time of last post */
|
|
}
|
|
}
|
|
|
|
/* iterate backwards through popMap - take the top 5 or 10% of list */
|
|
uint32_t popCount = 5;
|
|
if (popCount < popMap.size() / 10)
|
|
{
|
|
popCount = popMap.size() / 10;
|
|
}
|
|
|
|
uint32_t i = 0;
|
|
std::multimap<uint32_t, GroupItemInfo>::reverse_iterator rit;
|
|
//uint32_t popLimit = 0;
|
|
//for(rit = popMap.rbegin(); ((rit != popMap.rend()) && (i < popCount)); ++rit, ++i) ;
|
|
//if (rit != popMap.rend()) {
|
|
// popLimit = rit->first;
|
|
//}
|
|
|
|
for (rit = popMap.rbegin(); rit != popMap.rend(); ++rit,++i) {
|
|
//if (rit->second.popularity > (int) popLimit) {
|
|
if(i < popCount)
|
|
popList.append(rit->second);
|
|
else
|
|
otherList.append(rit->second);
|
|
}
|
|
|
|
/* now we can add them in as a tree! */
|
|
ui->groupTreeWidget->fillGroupItems(mYourGroups, adminList);
|
|
ui->groupTreeWidget->fillGroupItems(mSubscribedGroups, subList);
|
|
mSubscribedGroups->setText(2, QString::number(mSubscribedGroups->childCount())); // 1 COLUMN_UNREAD 2 COLUMN_POPULARITY
|
|
ui->groupTreeWidget->fillGroupItems(mPopularGroups, popList);
|
|
mPopularGroups->setText(2, QString::number(mPopularGroups->childCount()));
|
|
ui->groupTreeWidget->fillGroupItems(mOtherGroups, otherList);
|
|
mOtherGroups->setText(2, QString::number(mOtherGroups->childCount()));
|
|
|
|
mInFill = false;
|
|
|
|
/* Re-fill group */
|
|
if (!ui->groupTreeWidget->activateId(QString::fromStdString(mGroupId.toStdString()), true)) {
|
|
mGroupId.clear();
|
|
}
|
|
|
|
updateMessageSummaryList(RsGxsGroupId());
|
|
}
|
|
|
|
void GxsGroupFrameDialog::updateMessageSummaryList(RsGxsGroupId groupId)
|
|
{
|
|
if (!mInitialized) {
|
|
return;
|
|
}
|
|
|
|
if (groupId.isNull()) {
|
|
QTreeWidgetItem *items[2] = { mYourGroups, mSubscribedGroups };
|
|
for (int item = 0; item < 2; ++item) {
|
|
int child;
|
|
int childCount = items[item]->childCount();
|
|
for (child = 0; child < childCount; ++child) {
|
|
QTreeWidgetItem *childItem = items[item]->child(child);
|
|
QString childId = ui->groupTreeWidget->itemId(childItem);
|
|
if (childId.isEmpty()) {
|
|
continue;
|
|
}
|
|
|
|
requestGroupStatistics(RsGxsGroupId(childId.toLatin1().constData()));
|
|
}
|
|
}
|
|
} else {
|
|
requestGroupStatistics(groupId);
|
|
}
|
|
}
|
|
|
|
/*********************** **** **** **** ***********************/
|
|
/** Request / Response of Data ********************************/
|
|
/*********************** **** **** **** ***********************/
|
|
|
|
void GxsGroupFrameDialog::requestGroupSummary()
|
|
{
|
|
mStateHelper->setLoading(TOKEN_TYPE_GROUP_SUMMARY, true);
|
|
|
|
#ifdef DEBUG_GROUPFRAMEDIALOG
|
|
std::cerr << "GxsGroupFrameDialog::requestGroupSummary()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
mTokenQueue->cancelActiveRequestTokens(TOKEN_TYPE_GROUP_SUMMARY);
|
|
|
|
RsTokReqOptions opts;
|
|
opts.mReqType = requestGroupSummaryType();
|
|
|
|
uint32_t token;
|
|
mTokenQueue->requestGroupInfo(token, RS_TOKREQ_ANSTYPE_SUMMARY, opts, TOKEN_TYPE_GROUP_SUMMARY);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::loadGroupSummaryToken(const uint32_t &token, std::list<RsGroupMetaData> &groupInfo, RsUserdata *&/*userdata*/)
|
|
{
|
|
/* Default implementation for request type GXS_REQUEST_TYPE_GROUP_META */
|
|
mInterface->getGroupSummary(token, groupInfo);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::loadGroupSummary(const uint32_t &token)
|
|
{
|
|
#ifdef DEBUG_GROUPFRAMEDIALOG
|
|
std::cerr << "GxsGroupFrameDialog::loadGroupSummary()";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
std::list<RsGroupMetaData> groupInfo;
|
|
RsUserdata *userdata = NULL;
|
|
loadGroupSummaryToken(token, groupInfo, userdata);
|
|
|
|
mCachedGroupMetas.clear();
|
|
for(auto it(groupInfo.begin());it!=groupInfo.end();++it)
|
|
mCachedGroupMetas[(*it).mGroupId] = *it;
|
|
|
|
insertGroupsData(mCachedGroupMetas, userdata);
|
|
updateSearchResults();
|
|
|
|
mStateHelper->setLoading(TOKEN_TYPE_GROUP_SUMMARY, false);
|
|
|
|
if (userdata) {
|
|
delete(userdata);
|
|
}
|
|
|
|
if (!mNavigatePendingGroupId.isNull()) {
|
|
/* Navigate pending */
|
|
navigate(mNavigatePendingGroupId, mNavigatePendingMsgId);
|
|
|
|
mNavigatePendingGroupId.clear();
|
|
mNavigatePendingMsgId.clear();
|
|
}
|
|
}
|
|
|
|
/*********************** **** **** **** ***********************/
|
|
/*********************** **** **** **** ***********************/
|
|
|
|
//void GxsGroupFrameDialog::acknowledgeSubscribeChange(const uint32_t &token)
|
|
//{
|
|
//#ifdef DEBUG_GROUPFRAMEDIALOG
|
|
// std::cerr << "GxsGroupFrameDialog::acknowledgeSubscribeChange()";
|
|
// std::cerr << std::endl;
|
|
//#endif
|
|
|
|
// RsGxsGroupId groupId;
|
|
// mInterface->acknowledgeGrp(token, groupId);
|
|
|
|
// fillComplete();
|
|
//}
|
|
|
|
/*********************** **** **** **** ***********************/
|
|
/*********************** **** **** **** ***********************/
|
|
|
|
//void GxsGroupFrameDialog::requestGroupSummary_CurrentGroup(const RsGxsGroupId &groupId)
|
|
//{
|
|
// RsTokReqOptions opts;
|
|
// opts.mReqType = GXS_REQUEST_TYPE_GROUP_META;
|
|
|
|
// std::list<std::string> grpIds;
|
|
// grpIds.push_back(groupId);
|
|
|
|
// std::cerr << "GxsGroupFrameDialog::requestGroupSummary_CurrentGroup(" << groupId << ")";
|
|
// std::cerr << std::endl;
|
|
|
|
// uint32_t token;
|
|
// mInteface->requestGroupInfo(token, RS_TOKREQ_ANSTYPE_SUMMARY, opts, grpIds, TOKEN_TYPE_CURRENTGROUP);
|
|
//}
|
|
|
|
//void GxsGroupFrameDialog::loadGroupSummary_CurrentGroup(const uint32_t &token)
|
|
//{
|
|
// std::cerr << "GxsGroupFrameDialog::loadGroupSummary_CurrentGroup()";
|
|
// std::cerr << std::endl;
|
|
|
|
// std::list<RsGroupMetaData> groupInfo;
|
|
// rsGxsForums->getGroupSummary(token, groupInfo);
|
|
|
|
// if (groupInfo.size() == 1)
|
|
// {
|
|
// RsGroupMetaData fi = groupInfo.front();
|
|
// mSubscribeFlags = fi.mSubscribeFlags;
|
|
// }
|
|
// else
|
|
// {
|
|
// resetData();
|
|
// std::cerr << "GxsGroupFrameDialog::loadGroupSummary_CurrentGroup() ERROR Invalid Number of Groups...";
|
|
// std::cerr << std::endl;
|
|
// }
|
|
|
|
// setValid(true);
|
|
//}
|
|
|
|
/*********************** **** **** **** ***********************/
|
|
/*********************** **** **** **** ***********************/
|
|
|
|
void GxsGroupFrameDialog::requestGroupStatistics(const RsGxsGroupId &groupId)
|
|
{
|
|
uint32_t token;
|
|
mTokenService->requestGroupStatistic(token, groupId);
|
|
mTokenQueue->queueRequest(token, 0, RS_TOKREQ_ANSTYPE_ACK, TOKEN_TYPE_STATISTICS);
|
|
}
|
|
|
|
void GxsGroupFrameDialog::loadGroupStatistics(const uint32_t &token)
|
|
{
|
|
GxsGroupStatistic stats;
|
|
mInterface->getGroupStatistic(token, stats);
|
|
|
|
QTreeWidgetItem *item = ui->groupTreeWidget->getItemFromId(QString::fromStdString(stats.mGrpId.toStdString()));
|
|
if (!item) {
|
|
return;
|
|
}
|
|
|
|
ui->groupTreeWidget->setUnreadCount(item, mCountChildMsgs ? (stats.mNumThreadMsgsUnread + stats.mNumChildMsgsUnread) : stats.mNumThreadMsgsUnread);
|
|
}
|
|
|
|
/*********************** **** **** **** ***********************/
|
|
/*********************** **** **** **** ***********************/
|
|
|
|
void GxsGroupFrameDialog::loadRequest(const TokenQueue *queue, const TokenRequest &req)
|
|
{
|
|
#ifdef DEBUG_GROUPFRAMEDIALOG
|
|
std::cerr << "GxsGroupFrameDialog::loadRequest() UserType: " << req.mUserType;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
if (queue == mTokenQueue)
|
|
{
|
|
/* now switch on req */
|
|
switch(req.mUserType)
|
|
{
|
|
case TOKEN_TYPE_GROUP_SUMMARY:
|
|
loadGroupSummary(req.mToken);
|
|
break;
|
|
|
|
// case TOKEN_TYPE_SUBSCRIBE_CHANGE:
|
|
// acknowledgeSubscribeChange(req.mToken);
|
|
// break;
|
|
|
|
// case TOKEN_TYPE_CURRENTGROUP:
|
|
// loadGroupSummary_CurrentGroup(req.mToken);
|
|
// break;
|
|
|
|
case TOKEN_TYPE_STATISTICS:
|
|
loadGroupStatistics(req.mToken);
|
|
break;
|
|
|
|
default:
|
|
std::cerr << "GxsGroupFrameDialog::loadRequest() ERROR: INVALID TYPE";
|
|
std::cerr << std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
TurtleRequestId GxsGroupFrameDialog::distantSearch(const QString& search_string) // this should be overloaded in the child class
|
|
{
|
|
std::cerr << "Searching for \"" << search_string.toStdString() << "\". Function is not overloaded, so nothing will happen." << std::endl;
|
|
return 0;
|
|
}
|
|
|
|
void GxsGroupFrameDialog::searchNetwork(const QString& search_string)
|
|
{
|
|
if(search_string.isNull())
|
|
return ;
|
|
|
|
uint32_t request_id = distantSearch(search_string);
|
|
|
|
if(request_id == 0)
|
|
return ;
|
|
|
|
mSearchGroupsItems[request_id] = ui->groupTreeWidget->addSearchItem(tr("Search for")+ " \"" + search_string + "\"",(uint32_t)request_id,QIcon(icon(ICON_SEARCH)));
|
|
}
|
|
|
|
void GxsGroupFrameDialog::distantRequestGroupData()
|
|
{
|
|
QAction *action = dynamic_cast<QAction*>(sender()) ;
|
|
|
|
if(!action)
|
|
return ;
|
|
|
|
RsGxsGroupId group_id(action->data().toString().toStdString());
|
|
|
|
if(group_id.isNull())
|
|
{
|
|
std::cerr << "Cannot retrieve group! Group id is null!" << std::endl;
|
|
}
|
|
|
|
std::cerr << "Explicit request for group " << group_id << std::endl;
|
|
checkRequestGroup(group_id) ;
|
|
}
|
|
|