/******************************************************************************* * gui/common/FriendSelectionWidget.cpp * * * * Copyright (C) 2012, Retroshare Team * * * * 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 . * * * *******************************************************************************/ #include "gui/common/FilesDefs.h" #include #include #include "FriendSelectionWidget.h" #include "ui_FriendSelectionWidget.h" #include "gui/gxs/GxsIdDetails.h" #include #include "gui/notifyqt.h" #include "gui/common/RSTreeWidgetItem.h" #include "gui/common/StatusDefs.h" #include "util/qtthreadsutils.h" #include "gui/common/PeerDefs.h" #include "gui/common/GroupDefs.h" #include "rshare.h" #include #include #include #include #define COLUMN_NAME 0 #define COLUMN_CHECK 0 #define COLUMN_DATA 0 #define COLUMN_COUNT 1 #define IDDIALOG_IDLIST 1 #define ROLE_ID Qt::UserRole #define ROLE_SORT_GROUP Qt::UserRole + 1 #define ROLE_SORT_STANDARD_GROUP Qt::UserRole + 2 #define ROLE_SORT_NAME Qt::UserRole + 3 #define ROLE_SORT_STATE Qt::UserRole + 4 #define ROLE_FILTER_REASON Qt::UserRole + 5 #define IMAGE_GROUP16 ":/images/user/group16.png" #define IMAGE_FRIENDINFO ":/images/peerdetails_16x16.png" static bool isSelected(FriendSelectionWidget::Modus modus, QTreeWidgetItem *item) { switch (modus) { case FriendSelectionWidget::MODUS_SINGLE: case FriendSelectionWidget::MODUS_MULTI: return item->isSelected(); case FriendSelectionWidget::MODUS_CHECK: return (item->checkState(COLUMN_CHECK) == Qt::Checked); } return false; } static void setSelected(FriendSelectionWidget::Modus modus, QTreeWidgetItem *item, bool select) { switch (modus) { case FriendSelectionWidget::MODUS_SINGLE: case FriendSelectionWidget::MODUS_MULTI: item->setSelected(select); break; case FriendSelectionWidget::MODUS_CHECK: item->setCheckState(COLUMN_CHECK, select ? Qt::Checked : Qt::Unchecked); break; } } FriendSelectionWidget::FriendSelectionWidget(QWidget *parent) : QWidget(parent), ui(new Ui::FriendSelectionWidget) { ui->setupUi(this); mStarted = false; mListModus = MODUS_SINGLE; mShowTypes = SHOW_GROUP | SHOW_SSL; mInGroupItemChanged = false; mInGpgItemChanged = false; mInSslItemChanged = false; mInFillList = false; connect(ui->friendList, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(contextMenuRequested(QPoint))); connect(ui->friendList, SIGNAL(itemDoubleClicked(QTreeWidgetItem*,int)), this, SLOT(itemDoubleClicked(QTreeWidgetItem*,int))); connect(ui->friendList, SIGNAL(itemChanged(QTreeWidgetItem*,int)), this, SLOT(itemChanged(QTreeWidgetItem*,int))); connect(ui->friendList, SIGNAL(itemSelectionChanged()), this, SIGNAL(itemSelectionChanged())); connect(ui->filterLineEdit, SIGNAL(textChanged(QString)), this, SLOT(filterItems(QString))); connect(NotifyQt::getInstance(), SIGNAL(groupsChanged(int)), this, SLOT(groupsChanged(int))); connect(NotifyQt::getInstance(), SIGNAL(peerStatusChanged(const QString&,int)), this, SLOT(peerStatusChanged(const QString&,int))); mCompareRole = new RSTreeWidgetItemCompareRole; mActionSortByState = new QAction(tr("Sort by state"), this); mActionSortByState->setCheckable(true); connect(mActionSortByState, SIGNAL(toggled(bool)), this, SLOT(sortByState(bool))); ui->friendList->addContextMenuAction(mActionSortByState); mActionFilterConnected = new QAction(tr("Filter only connected"), this); mActionFilterConnected->setCheckable(true); connect(mActionFilterConnected, SIGNAL(toggled(bool)), this, SLOT(filterConnected(bool))); ui->friendList->addContextMenuAction(mActionFilterConnected); /* initialize list */ ui->friendList->setColumnCount(COLUMN_COUNT); ui->friendList->headerItem()->setText(COLUMN_NAME, tr("Name")); ui->friendList->setFilterReasonRole(ROLE_FILTER_REASON); /* sort list by name ascending */ ui->friendList->sortItems(COLUMN_NAME, Qt::AscendingOrder); sortByState(false); ui->filterLineEdit->setPlaceholderText(tr("Search Friends")); ui->filterLineEdit->showFilterIcon(); /* Refresh style to have the correct text color */ Rshare::refreshStyleSheet(this, false); } FriendSelectionWidget::~FriendSelectionWidget() { delete ui; } void FriendSelectionWidget::changeEvent(QEvent *e) { QWidget::changeEvent(e); switch (e->type()) { case QEvent::StyleChange: fillList(); break; default: // remove compiler warnings break; } } void FriendSelectionWidget::setHeaderText(const QString &text) { ui->friendList->headerItem()->setText(COLUMN_NAME, text); } void FriendSelectionWidget::setModus(Modus modus) { mListModus = modus; switch (mListModus) { case MODUS_SINGLE: case MODUS_CHECK: ui->friendList->setSelectionMode(QAbstractItemView::SingleSelection); break; case MODUS_MULTI: ui->friendList->setSelectionMode(QAbstractItemView::ExtendedSelection); break; } fillList(); } void FriendSelectionWidget::setShowType(ShowTypes types) { mShowTypes = types; fillList(); } int FriendSelectionWidget::addColumn(const QString &title) { int column = ui->friendList->columnCount(); ui->friendList->setColumnCount(column + 1); ui->friendList->headerItem()->setText(column, title); return column; } void FriendSelectionWidget::showEvent(QShowEvent *e) { if(gxsIds.empty()) loadIdentities(); } void FriendSelectionWidget::start() { mStarted = true; secured_fillList(); for (int i = 0; i < ui->friendList->columnCount(); ++i) { ui->friendList->resizeColumnToContents(i); } } static void initSslItem(QTreeWidgetItem *item, const RsPeerDetails &detail, const std::list &statusInfo, QColor textColorOnline) { QString name = PeerDefs::nameWithLocation(detail); item->setText(COLUMN_NAME, name); int state = RS_STATUS_OFFLINE; if (detail.state & RS_PEER_STATE_CONNECTED) { std::list::const_iterator it; for (it = statusInfo.begin(); it != statusInfo.end() ; ++it) { if (it->id == detail.id) { state = it->status; break; } } } if (state != (int) RS_STATUS_OFFLINE) { item->setData(COLUMN_NAME, Qt::ForegroundRole, textColorOnline); } item->setIcon(COLUMN_NAME, FilesDefs::getIconFromQtResourcePath(StatusDefs::imageUser(state))); item->setData(COLUMN_DATA, ROLE_ID, QString::fromStdString(detail.id.toStdString())); item->setData(COLUMN_NAME, ROLE_SORT_GROUP, 1); item->setData(COLUMN_NAME, ROLE_SORT_STANDARD_GROUP, 0); item->setData(COLUMN_NAME, ROLE_SORT_NAME, name); item->setData(COLUMN_NAME, ROLE_SORT_STATE, state); } void FriendSelectionWidget::fillList() { if (!mStarted) { return; } if(!isVisible()) return ; if(RsAutoUpdatePage::eventsLocked()) return ; secured_fillList() ; } void FriendSelectionWidget::loadIdentities() { // store all IDs locally, and call fillList() ; RsThread::async([this]() { std::list ids_meta; if(!rsIdentity->getIdentitiesSummaries(ids_meta)) { RS_ERR("failed to retrieve identities group info for all identities"); return; } auto ids = std::make_unique>(); for(auto& meta: ids_meta) ids->push_back(meta.mGroupId); RsQThreadUtils::postToObject( [ids = std::move(ids), this]() { // We do that is the GUI thread. Dont try it on another thread! gxsIds = *ids; /* TODO: To furter optimize away a copy gxsIds could be a unique_ptr * too */ fillList(); }, this ); }); } void FriendSelectionWidget::secured_fillList() { mInFillList = true; // get selected items std::set sslIdsSelected; if (mShowTypes & SHOW_SSL) { if(!ui->friendList->topLevelItemCount()) // if not loaded yet, use the existing list. for(auto& s:mPreSelectedIds) sslIdsSelected.insert(RsPeerId(s)); selectedIds(sslIdsSelected,true); } std::set groupIdsSelected; if (mShowTypes & SHOW_GROUP) { selectedIds(groupIdsSelected,true); if(!ui->friendList->topLevelItemCount()) // if not loaded yet, use the existing list. for(auto& s:mPreSelectedIds) groupIdsSelected.insert(RsNodeGroupId(s)); } std::set gpgIdsSelected; if (mShowTypes & (SHOW_GPG | SHOW_NON_FRIEND_GPG)) { selectedIds(gpgIdsSelected,true); if(!ui->friendList->topLevelItemCount()) // if not loaded yet, use the existing list. for(auto& s:mPreSelectedIds) gpgIdsSelected.insert(RsPgpId(s)); } std::set gxsIdsSelected; if (mShowTypes & SHOW_GXS) { selectedIds(gxsIdsSelected,true); if(!ui->friendList->topLevelItemCount()) // if not loaded yet, use the existing list. for(auto& s:mPreSelectedIds) gxsIdsSelected.insert(RsGxsId(s)); } std::set gxsIdsSelected2; if (mShowTypes & SHOW_CONTACTS) selectedIds(gxsIdsSelected2,true); // remove old items ui->friendList->clear(); // get existing groups std::list groupInfoList; std::list::iterator groupIt; rsPeers->getGroupInfoList(groupInfoList); std::list gpgIds; std::list::iterator gpgIt; if(mShowTypes & SHOW_NON_FRIEND_GPG) rsPeers->getGPGAllList(gpgIds); else rsPeers->getGPGAcceptedList(gpgIds); // add own pgp id to the list gpgIds.push_back(rsPeers->getGPGOwnId()) ; std::list sslIds; std::list::iterator sslIt; if ((mShowTypes & (SHOW_SSL | SHOW_GPG)) == SHOW_SSL) { rsPeers->getFriendList(sslIds); } std::list statusInfo; std::list::iterator statusIt; rsStatus->getStatusList(statusInfo); std::list filledIds; // gpg or ssl id // start with groups groupIt = groupInfoList.begin(); while (true) { QTreeWidgetItem *groupItem = NULL; QTreeWidgetItem *gpgItem = NULL; QTreeWidgetItem *gxsItem = NULL; RsGroupInfo *groupInfo = NULL; if ((mShowTypes & SHOW_GROUP) && groupIt != groupInfoList.end()) { groupInfo = &(*groupIt); if (groupInfo->peerIds.empty()) { // don't show empty groups ++groupIt; continue; } // add group item groupItem = new RSTreeWidgetItem(mCompareRole, IDTYPE_GROUP); // Add item to the list ui->friendList->addTopLevelItem(groupItem); groupItem->setFlags(Qt::ItemIsUserCheckable | groupItem->flags()); groupItem->setChildIndicatorPolicy(QTreeWidgetItem::DontShowIndicatorWhenChildless); groupItem->setTextAlignment(COLUMN_NAME, Qt::AlignLeft | Qt::AlignVCenter); groupItem->setIcon(COLUMN_NAME, FilesDefs::getIconFromQtResourcePath(IMAGE_GROUP16)); groupItem->setData(COLUMN_DATA, ROLE_ID, QString::fromStdString(groupInfo->id.toStdString())); groupItem->setExpanded(true); QString groupName = GroupDefs::name(*groupInfo); groupItem->setText(COLUMN_NAME, groupName); groupItem->setData(COLUMN_NAME, ROLE_SORT_GROUP, 0); groupItem->setData(COLUMN_NAME, ROLE_SORT_STANDARD_GROUP, (groupInfo->flag & RS_GROUP_FLAG_STANDARD) ? 0 : 1); groupItem->setData(COLUMN_NAME, ROLE_SORT_NAME, groupName); groupItem->setData(COLUMN_NAME, ROLE_SORT_STATE, 0); if (mListModus == MODUS_CHECK) { groupItem->setCheckState(0, Qt::Unchecked); } emit itemAdded(IDTYPE_GROUP, QString::fromStdString(groupInfo->id.toStdString()), groupItem); if (std::find(groupIdsSelected.begin(), groupIdsSelected.end(), groupInfo->id) != groupIdsSelected.end()) { setSelected(mListModus, groupItem, true); } } if (mShowTypes & (SHOW_GPG | SHOW_NON_FRIEND_GPG)) { // iterate through gpg ids for (gpgIt = gpgIds.begin(); gpgIt != gpgIds.end(); ++gpgIt) { if (groupInfo) { // we fill a group, check if gpg id is assigned if (std::find(groupInfo->peerIds.begin(), groupInfo->peerIds.end(), *gpgIt) == groupInfo->peerIds.end()) { continue; } } else { // we fill the not assigned gpg ids if (std::find(filledIds.begin(), filledIds.end(), (*gpgIt).toStdString()) != filledIds.end()) { continue; } } // add equal too, its no problem filledIds.push_back((*gpgIt).toStdString()); RsPeerDetails detail; if (!rsPeers->getGPGDetails(*gpgIt, detail)) { continue; /* BAD */ } // make a widget per friend gpgItem = new RSTreeWidgetItem(mCompareRole, IDTYPE_GPG); QString name = QString::fromUtf8(detail.name.c_str()); gpgItem->setText(COLUMN_NAME, name + " ("+QString::fromStdString( (*gpgIt).toStdString() )+")"); sslIds.clear(); rsPeers->getAssociatedSSLIds(*gpgIt, sslIds); int state = RS_STATUS_OFFLINE; for (statusIt = statusInfo.begin(); statusIt != statusInfo.end() ; ++statusIt) { if (std::find(sslIds.begin(), sslIds.end(), statusIt->id) != sslIds.end()) { if (statusIt->status != RS_STATUS_OFFLINE) { state = RS_STATUS_ONLINE; break; } } } if (state != (int) RS_STATUS_OFFLINE) { gpgItem->setData(COLUMN_NAME, Qt::ForegroundRole, textColorOnline()); } gpgItem->setFlags(Qt::ItemIsUserCheckable | gpgItem->flags()); gpgItem->setIcon(COLUMN_NAME, FilesDefs::getIconFromQtResourcePath(StatusDefs::imageUser(state))); gpgItem->setData(COLUMN_DATA, ROLE_ID, QString::fromStdString(detail.gpg_id.toStdString())); gpgItem->setData(COLUMN_NAME, ROLE_SORT_GROUP, 1); gpgItem->setData(COLUMN_NAME, ROLE_SORT_STANDARD_GROUP, 0); gpgItem->setData(COLUMN_NAME, ROLE_SORT_NAME, name); gpgItem->setData(COLUMN_NAME, ROLE_SORT_STATE, state); if (mListModus == MODUS_CHECK) { gpgItem->setCheckState(0, Qt::Unchecked); } // add to the list if (groupItem) { groupItem->addChild(gpgItem); } else { ui->friendList->addTopLevelItem(gpgItem); } gpgItem->setExpanded(true); emit itemAdded(IDTYPE_GPG, QString::fromStdString(detail.gpg_id.toStdString()), gpgItem); if (mShowTypes & SHOW_SSL) { // iterate through associated ssl ids for (sslIt = sslIds.begin(); sslIt != sslIds.end(); ++sslIt) { RsPeerDetails detail; if (!rsPeers->getPeerDetails(*sslIt, detail)) { continue; /* BAD */ } // make a widget per friend QTreeWidgetItem *item = new RSTreeWidgetItem(mCompareRole, IDTYPE_SSL); item->setFlags(Qt::ItemIsUserCheckable | item->flags()); initSslItem(item, detail, statusInfo, textColorOnline()); if (mListModus == MODUS_CHECK) { item->setCheckState(0, Qt::Unchecked); } // add to the list gpgItem->addChild(item); emit itemAdded(IDTYPE_SSL, QString::fromStdString(detail.id.toStdString()), item); if (std::find(sslIdsSelected.begin(), sslIdsSelected.end(), detail.id) != sslIdsSelected.end()) { setSelected(mListModus, item, true); } } } if (std::find(gpgIdsSelected.begin(), gpgIdsSelected.end(), detail.gpg_id) != gpgIdsSelected.end()) { setSelected(mListModus, gpgItem, true); } } } else { // iterate through ssl ids for (sslIt = sslIds.begin(); sslIt != sslIds.end(); ++sslIt) { RsPeerDetails detail; if (!rsPeers->getPeerDetails(*sslIt, detail)) { continue; /* BAD */ } if (groupInfo) { // we fill a group, check if gpg id is assigned if (std::find(groupInfo->peerIds.begin(), groupInfo->peerIds.end(), detail.gpg_id) == groupInfo->peerIds.end()) { continue; } } else { // we fill the not assigned ssl ids if (std::find(filledIds.begin(), filledIds.end(), (*sslIt).toStdString()) != filledIds.end()) { continue; } } // add equal too, its no problem filledIds.push_back(detail.id.toStdString()); // make a widget per friend QTreeWidgetItem *item = new RSTreeWidgetItem(mCompareRole, IDTYPE_SSL); initSslItem(item, detail, statusInfo, textColorOnline()); item->setFlags(Qt::ItemIsUserCheckable | item->flags()); if (mListModus == MODUS_CHECK) { item->setCheckState(0, Qt::Unchecked); } // add to the list if (groupItem) { groupItem->addChild(item); } else { ui->friendList->addTopLevelItem(item); } emit itemAdded(IDTYPE_SSL, QString::fromStdString(detail.id.toStdString()), item); if (std::find(sslIdsSelected.begin(), sslIdsSelected.end(), detail.id) != sslIdsSelected.end()) { setSelected(mListModus, item, true); } } } if(mShowTypes & SHOW_GXS) { // iterate through gpg ids for (std::vector::const_iterator gxsIt = gxsIds.begin(); gxsIt != gxsIds.end(); ++gxsIt) { // we fill the not assigned gpg ids if (std::find(filledIds.begin(), filledIds.end(), (*gxsIt).toStdString()) != filledIds.end()) continue; // add equal too, its no problem filledIds.push_back((*gxsIt).toStdString()); RsIdentityDetails detail; if (!rsIdentity->getIdDetails(RsGxsId(*gxsIt), detail)) continue; /* BAD */ QList icons ; GxsIdDetails::getIcons(detail,icons,GxsIdDetails::ICON_TYPE_AVATAR) ; QIcon identicon = icons.front() ; // make a widget per friend gxsItem = new RSTreeWidgetItem(mCompareRole, IDTYPE_GXS); QString name = QString::fromUtf8(detail.mNickname.c_str()); gxsItem->setText(COLUMN_NAME, name + " ("+QString::fromStdString( (*gxsIt).toStdString() )+")"); //gxsItem->setData(COLUMN_NAME, Qt::ForegroundRole, textColorOnline()); gxsItem->setFlags(Qt::ItemIsUserCheckable | gxsItem->flags()); gxsItem->setIcon(COLUMN_NAME, identicon); gxsItem->setData(COLUMN_DATA, ROLE_ID, QString::fromStdString(detail.mId.toStdString())); gxsItem->setData(COLUMN_NAME, ROLE_SORT_GROUP, 1); gxsItem->setData(COLUMN_NAME, ROLE_SORT_STANDARD_GROUP, 0); gxsItem->setData(COLUMN_NAME, ROLE_SORT_NAME, name); //TODO: online state for gxs items gxsItem->setData(COLUMN_NAME, ROLE_SORT_STATE, 1); if (mListModus == MODUS_CHECK) gxsItem->setCheckState(0, Qt::Unchecked); ui->friendList->addTopLevelItem(gxsItem); gxsItem->setExpanded(true); emit itemAdded(IDTYPE_GXS, QString::fromStdString(detail.mId.toStdString()), gxsItem); if (std::find(gxsIdsSelected.begin(), gxsIdsSelected.end(), detail.mId) != gxsIdsSelected.end()) setSelected(mListModus, gxsItem, true); } } if(mShowTypes & SHOW_CONTACTS) { // iterate through gpg ids for (std::vector::const_iterator gxsIt = gxsIds.begin(); gxsIt != gxsIds.end(); ++gxsIt) { // we fill the not assigned gpg ids if (std::find(filledIds.begin(), filledIds.end(), (*gxsIt).toStdString()) != filledIds.end()) continue; // add equal too, its no problem filledIds.push_back((*gxsIt).toStdString()); RsIdentityDetails detail; if (!rsIdentity->getIdDetails(RsGxsId(*gxsIt), detail)) continue; /* BAD */ QList icons ; GxsIdDetails::getIcons(detail,icons,GxsIdDetails::ICON_TYPE_AVATAR) ; QIcon identicon = icons.front() ; if(detail.mFlags & RS_IDENTITY_FLAGS_IS_A_CONTACT) { // make a widget per friend gxsItem = new RSTreeWidgetItem(mCompareRole, IDTYPE_GXS); QString name = QString::fromUtf8(detail.mNickname.c_str()); gxsItem->setText(COLUMN_NAME, name + " ("+QString::fromStdString( (*gxsIt).toStdString() )+")"); //gxsItem->setData(COLUMN_NAME, Qt::ForegroundRole, textColorOnline()); gxsItem->setFlags(Qt::ItemIsUserCheckable | gxsItem->flags()); gxsItem->setIcon(COLUMN_NAME, identicon); gxsItem->setData(COLUMN_DATA, ROLE_ID, QString::fromStdString(detail.mId.toStdString())); gxsItem->setData(COLUMN_NAME, ROLE_SORT_GROUP, 1); gxsItem->setData(COLUMN_NAME, ROLE_SORT_STANDARD_GROUP, 0); gxsItem->setData(COLUMN_NAME, ROLE_SORT_NAME, name); //TODO: online state for gxs items gxsItem->setData(COLUMN_NAME, ROLE_SORT_STATE, 1); if (mListModus == MODUS_CHECK) gxsItem->setCheckState(0, Qt::Unchecked); ui->friendList->addTopLevelItem(gxsItem); gxsItem->setExpanded(true); emit itemAdded(IDTYPE_GXS, QString::fromStdString(detail.mId.toStdString()), gxsItem); if (std::find(gxsIdsSelected.begin(), gxsIdsSelected.end(), detail.mId) != gxsIdsSelected.end()) setSelected(mListModus, gxsItem, true); } } } if (groupIt != groupInfoList.end()) { ++groupIt; } else { // all done break; } } if (ui->filterLineEdit->text().isEmpty() == false) { filterItems(ui->filterLineEdit->text()); } ui->friendList->update(); /* update display */ mInFillList = false; ui->friendList->resort(); filterConnected(isFilterConnected()); emit contentChanged(); } void FriendSelectionWidget::updateDisplay(bool) { loadIdentities() ; } // This call is inlined so that there's no linking conflict with MinGW on Windows template<> inline void FriendSelectionWidget::setSelectedIds(const std::set& ids, bool add) { if(!add) mPreSelectedIds.clear(); for(auto& gxsId:ids) mPreSelectedIds.insert(gxsId.toStdString()); loadIdentities(); } void FriendSelectionWidget::groupsChanged(int /*type*/) { if (mShowTypes & SHOW_GROUP) { fillList(); } } void FriendSelectionWidget::peerStatusChanged(const QString& peerId, int status) { if(!isVisible()) return ; if(RsAutoUpdatePage::eventsLocked()) return ; RsPeerId peerid(peerId.toStdString()) ; QString gpgId; int gpgStatus = RS_STATUS_OFFLINE; if (mShowTypes & (SHOW_GPG | SHOW_NON_FRIEND_GPG)) { /* need gpg id and online state */ RsPeerDetails detail; if (rsPeers->getPeerDetails(peerid, detail)) { gpgId = QString::fromStdString(detail.gpg_id.toStdString()); if (status == (int) RS_STATUS_OFFLINE) { /* try other nodes */ std::list sslIds; rsPeers->getAssociatedSSLIds(detail.gpg_id, sslIds); std::list statusInfo; std::list::iterator statusIt; rsStatus->getStatusList(statusInfo); for (statusIt = statusInfo.begin(); statusIt != statusInfo.end() ; ++statusIt) { if (std::find(sslIds.begin(), sslIds.end(), statusIt->id) != sslIds.end()) { if (statusIt->status != RS_STATUS_OFFLINE) { gpgStatus = RS_STATUS_ONLINE; break; } } } } else { /* one node is online */ gpgStatus = RS_STATUS_ONLINE; } } } QTreeWidgetItemIterator itemIterator(ui->friendList); QTreeWidgetItem *item; while ((item = *itemIterator) != NULL) { ++itemIterator; bool bFoundGPG = false; bool bFoundSSL = false; switch (idTypeFromItem(item)) { case IDTYPE_NONE: case IDTYPE_GROUP: case IDTYPE_GXS: break; case IDTYPE_GPG: { if (item->data(COLUMN_DATA, ROLE_ID).toString() == gpgId) { if (status != (int) RS_STATUS_OFFLINE) { item->setData(COLUMN_NAME, Qt::ForegroundRole, textColorOnline()); } else { item->setData(COLUMN_NAME, Qt::ForegroundRole, QVariant()); } item->setIcon(COLUMN_NAME, FilesDefs::getIconFromQtResourcePath(StatusDefs::imageUser(gpgStatus))); item->setData(COLUMN_NAME, ROLE_SORT_STATE, gpgStatus); bFoundGPG = true; } } break; case IDTYPE_SSL: { if (item->data(COLUMN_DATA, ROLE_ID).toString() == peerId) { if (status != (int) RS_STATUS_OFFLINE) { item->setData(COLUMN_NAME, Qt::ForegroundRole, textColorOnline()); } else { item->setData(COLUMN_NAME, Qt::ForegroundRole, QVariant()); } item->setIcon(COLUMN_NAME, FilesDefs::getIconFromQtResourcePath(StatusDefs::imageUser(status))); item->setData(COLUMN_NAME, ROLE_SORT_STATE, status); bFoundSSL = true; } } break; } if (bFoundGPG) { if (mShowTypes & SHOW_GROUP) { // a friend can be assigned to more than one group } else { if (mShowTypes & SHOW_SSL) { // search for ssl id } else { break; } } } if (bFoundSSL) { if (mShowTypes & SHOW_GROUP) { // a friend can be assigned to more than one group } else { break; } } } ui->friendList->resort(); filterConnected(isFilterConnected()); } void FriendSelectionWidget::addContextMenuAction(QAction *action) { mContextMenuActions.push_back(action); } void FriendSelectionWidget::contextMenuRequested(const QPoint &/*pos*/) { QMenu *contextMenu = new QMenu(this); if (mListModus == MODUS_MULTI) { contextMenu->addAction(QIcon(), tr("Mark all"), this, SLOT(selectAll())); contextMenu->addAction(QIcon(), tr("Mark none"), this, SLOT(deselectAll())); } if (!mContextMenuActions.isEmpty()) { bool addSeparator = false; if (!contextMenu->isEmpty()) { // Check for visible action foreach (QAction *action, mContextMenuActions) { if (action->isVisible()) { addSeparator = true; break; } } } if (addSeparator) { contextMenu->addSeparator(); } contextMenu->addActions(mContextMenuActions); } contextMenu = ui->friendList->createStandardContextMenu(contextMenu); if (!contextMenu->isEmpty()) { contextMenu->exec(QCursor::pos()); } delete contextMenu; } void FriendSelectionWidget::itemDoubleClicked(QTreeWidgetItem *item, int /*column*/) { if (!item) { return; } emit doubleClicked(idTypeFromItem(item), item->data(COLUMN_DATA, ROLE_ID).toString()); } void FriendSelectionWidget::itemChanged(QTreeWidgetItem *item, int column) { if (mInFillList) { return; } if (column != COLUMN_CHECK) { emit itemChanged(idTypeFromItem(item), item->data(COLUMN_DATA, ROLE_ID).toString(), item, column); return; } if (mListModus != MODUS_CHECK) { return; } switch (idTypeFromItem(item)) { case IDTYPE_NONE: break; case IDTYPE_GROUP: { if (mInGroupItemChanged || mInGpgItemChanged || mInSslItemChanged) { break; } mInGroupItemChanged = true; bool selected = isSelected(mListModus, item); int childCount = item->childCount(); for (int i = 0; i < childCount; ++i) { setSelected(mListModus, item->child(i), selected); } mInGroupItemChanged = false; } break; case IDTYPE_GPG: case IDTYPE_GXS: { if (mInGpgItemChanged) { break; } mInGpgItemChanged = true; if (!mInSslItemChanged) { bool selected = isSelected(mListModus, item); int childCount = item->childCount(); for (int i = 0; i < childCount; ++i) { setSelected(mListModus, item->child(i), selected); } } if (!mInGroupItemChanged) { QTreeWidgetItem *itemParent = item->parent(); if (itemParent) { int childCount = itemParent->childCount(); bool foundUnselected = false; for (int index = 0; index < childCount; ++index) { if (!isSelected(mListModus, itemParent->child(index))) { foundUnselected = true; break; } } setSelected(mListModus, itemParent, !foundUnselected); } } mInGpgItemChanged = false; } break; case IDTYPE_SSL: { if (mInGroupItemChanged || mInGpgItemChanged || mInSslItemChanged) { break; } mInSslItemChanged = true; QTreeWidgetItem *itemParent = item->parent(); if (itemParent) { int childCount = itemParent->childCount(); bool foundUnselected = false; for (int index = 0; index < childCount; ++index) { if (!isSelected(mListModus, itemParent->child(index))) { foundUnselected = true; break; } } setSelected(mListModus, itemParent, !foundUnselected); } mInSslItemChanged = false; } break; } } void FriendSelectionWidget::filterItems(const QString& text) { ui->friendList->filterItems(COLUMN_NAME, text); ui->friendList->resort(); filterConnected(isFilterConnected()); } int FriendSelectionWidget::selectedItemCount() { return ui->friendList->selectedItems().count(); } std::string FriendSelectionWidget::selectedId(IdType &idType) { QTreeWidgetItem *item = ui->friendList->currentItem(); if (!item) { idType = IDTYPE_NONE; return ""; } idType = idTypeFromItem(item); return idFromItem(item); } void FriendSelectionWidget::selectedIds_internal(IdType idType, std::set &ids, bool onlyDirectSelected) { QTreeWidgetItemIterator itemIterator(ui->friendList); QTreeWidgetItem *item; while ((item = *itemIterator) != NULL) { ++itemIterator; std::string id; switch (idTypeFromItem(item)) { case IDTYPE_NONE: break; case IDTYPE_GROUP: if (idType == IDTYPE_GROUP) { if (isSelected(mListModus, item)) { id = idFromItem(item); } } break; case IDTYPE_GPG: case IDTYPE_GXS: if (idType == IDTYPE_GPG || idType == IDTYPE_GXS) { if (isSelected(mListModus, item)) { id = idFromItem(item); } else { if (!onlyDirectSelected) { QTreeWidgetItem *itemParent = item; while ((itemParent = itemParent->parent()) != NULL) { if (isSelected(mListModus, itemParent)) { id = idFromItem(item); break; } } } } } break; case IDTYPE_SSL: if (idType == IDTYPE_SSL) { if (isSelected(mListModus, item)) { id = idFromItem(item); } else { if (!onlyDirectSelected) { QTreeWidgetItem *itemParent = item; while ((itemParent = itemParent->parent()) != NULL) { if (isSelected(mListModus, itemParent)) { id = idFromItem(item); break; } } } } } break; } if (!id.empty() && std::find(ids.begin(), ids.end(), id) == ids.end()) { ids.insert(id); } } } void FriendSelectionWidget::deselectAll() { for(QTreeWidgetItemIterator itemIterator(ui->friendList);*itemIterator!=NULL;++itemIterator) setSelected(mListModus, *itemIterator, false); } void FriendSelectionWidget::selectAll() { for(QTreeWidgetItemIterator itemIterator(ui->friendList);*itemIterator!=NULL;++itemIterator) setSelected(mListModus, *itemIterator, true); } void FriendSelectionWidget::setSelectedIdsFromString(IdType type, const std::set& ids, bool add) { setSelectedIds_internal(type,ids,add); } void FriendSelectionWidget::setSelectedIds_internal(IdType idType, const std::set &ids, bool add) { mPreSelectedIds = ids; // if items are already loaded, check them QTreeWidgetItemIterator itemIterator(ui->friendList); QTreeWidgetItem *item; while ((item = *itemIterator) != NULL) { ++itemIterator; std::string id = idFromItem(item); IdType itemType = idTypeFromItem(item); switch (itemType) { case IDTYPE_NONE: break; case IDTYPE_GROUP: case IDTYPE_GPG: case IDTYPE_SSL: case IDTYPE_GXS: if (idType == itemType) { if (ids.find(id) != ids.end()) { setSelected(mListModus, item, true); break; } } if (!add) { setSelected(mListModus, item, false); } break; } } } void FriendSelectionWidget::itemsFromId(IdType idType, const std::string &id, QList &items) { QTreeWidgetItemIterator itemIterator(ui->friendList); QTreeWidgetItem *item; while ((item = *itemIterator) != NULL) { ++itemIterator; if (idType == idTypeFromItem(item) && idFromItem(item) == id) { items.push_back(item); } } } void FriendSelectionWidget::items(QList &_items, IdType idType) { QTreeWidgetItemIterator itemIterator(ui->friendList); QTreeWidgetItem *item; while ((item = *itemIterator) != NULL) { ++itemIterator; if (idType == IDTYPE_NONE || idType == idTypeFromItem(item)) { _items.push_back(item); } } } FriendSelectionWidget::IdType FriendSelectionWidget::idTypeFromItem(QTreeWidgetItem *item) { if (!item) { return IDTYPE_NONE; } return (IdType) item->type(); } std::string FriendSelectionWidget::idFromItem(QTreeWidgetItem *item) { if (!item) { return ""; } return item->data(COLUMN_DATA, ROLE_ID).toString().toStdString(); } void FriendSelectionWidget::sortByState(bool sort) { mCompareRole->setRole(COLUMN_NAME, ROLE_SORT_GROUP); mCompareRole->addRole(COLUMN_NAME, ROLE_SORT_STANDARD_GROUP); if (sort) { mCompareRole->addRole(COLUMN_NAME, ROLE_SORT_STATE); mCompareRole->addRole(COLUMN_NAME, ROLE_SORT_NAME); } else { mCompareRole->addRole(COLUMN_NAME, ROLE_SORT_NAME); mCompareRole->addRole(COLUMN_NAME, ROLE_SORT_STATE); } mActionSortByState->setChecked(sort); ui->friendList->resort(); filterConnected(isFilterConnected()); } bool FriendSelectionWidget::isSortByState() { return mActionSortByState->isChecked(); } void FriendSelectionWidget::filterConnected(bool filter) { ui->friendList->filterMinValItems(COLUMN_NAME, filter ? RS_STATUS_AWAY : RS_STATUS_OFFLINE, ROLE_SORT_STATE); mActionFilterConnected->setChecked(filter); ui->friendList->resort(); } bool FriendSelectionWidget::isFilterConnected() { return mActionFilterConnected->isChecked(); }