/******************************************************************************* * retroshare-gui/src/gui/gxs/GxsIdChooser.cpp * * * * Copyright 2012-2013 by Robert Fernie * * * * 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 "GxsIdChooser.h" #include "GxsIdDetails.h" #include "RsGxsUpdateBroadcastBase.h" #include "gui/Identity/IdEditDialog.h" #include "util/misc.h" #include #include #include #include #include #define ROLE_SORT Qt::UserRole + 1 // Qt::UserRole is reserved for data #define ROLE_TYPE Qt::UserRole + 2 // /* Used for sorting too */ #define TYPE_NO_ID 1 #define TYPE_FOUND_ID 2 #define TYPE_UNKNOWN_ID 3 #define TYPE_CREATE_ID 4 #define BANNED_ICON ":/icons/yellow_biohazard64.png" #define IDCHOOSER_REFRESH 1 //#define IDCHOOSER_DEBUG /** Constructor */ GxsIdChooser::GxsIdChooser(QWidget *parent) : QComboBox(parent), mFlags(IDCHOOSER_ANON_DEFAULT) { // mBase = new RsGxsUpdateBroadcastBase(rsIdentity, this); // connect(mBase, SIGNAL(fillDisplay(bool)), this, SLOT(fillDisplay(bool))); /* Initialize ui */ setSizeAdjustPolicy(QComboBox::AdjustToContents); mFirstLoad = true; mAllowedCount = 0 ; mDefaultId.clear() ; /* Enable sort with own role */ QSortFilterProxyModel *proxy = new QSortFilterProxyModel(this); proxy->setSourceModel(model()); proxy->setDynamicSortFilter(false); model()->setParent(proxy); setModel(proxy); proxy->setSortRole(ROLE_SORT); /* Connect signals */ connect(this, SIGNAL(currentIndexChanged(int)), this, SLOT(myCurrentIndexChanged(int))); connect(this, SIGNAL(activated(int)), this, SLOT(indexActivated(int))); } void GxsIdChooser::setFlags(uint32_t flags) { mFlags = flags ; updateDisplay(true); } GxsIdChooser::~GxsIdChooser() { } void GxsIdChooser::fillDisplay(bool complete) { updateDisplay(complete); update(); // Qt flush } void GxsIdChooser::showEvent(QShowEvent *event) { // mBase->showEvent(event); updateDisplay(true); QComboBox::showEvent(event); } void GxsIdChooser::setIdConstraintSet(const std::set& s) { mConstraintIdsSet = s ; updateDisplay(true); update(); // Qt flush } void GxsIdChooser::loadIds(uint32_t chooserFlags, const RsGxsId &defId) { mFlags = chooserFlags; mDefaultId = defId; clear(); mFirstLoad = true; updateDisplay(true); update(); // Qt flush } void GxsIdChooser::setDefaultId(const RsGxsId &defId) { mDefaultId = defId; } static void loadPrivateIdsCallback(GxsIdDetailsType type, const RsIdentityDetails &details, QObject *object, const QVariant &/*data*/) { GxsIdChooser *chooser = dynamic_cast(object); if (!chooser) return; // this prevents the objects that depend on what's in the combo-box to activate and // perform any change.Only user-changes should cause this. chooser->blockSignals(true) ; QString text = GxsIdDetails::getNameForType(type, details); QString id = QString::fromStdString(details.mId.toStdString()); /* Find and replace text of exisiting item */ int index = chooser->findData(id); if (index >= 0) { chooser->setItemText(index, text); } else { /* Add new item */ chooser->addItem(text, id); index = chooser->count() - 1; } QList icons; switch (type) { case GXS_ID_DETAILS_TYPE_EMPTY: case GXS_ID_DETAILS_TYPE_FAILED: break; case GXS_ID_DETAILS_TYPE_LOADING: icons.push_back(GxsIdDetails::getLoadingIcon(details.mId)); break; case GXS_ID_DETAILS_TYPE_DONE: GxsIdDetails::getIcons(details, icons, GxsIdDetails::ICON_TYPE_AVATAR); break; case GXS_ID_DETAILS_TYPE_BANNED: icons.push_back(QIcon(BANNED_ICON)) ; break; } chooser->setItemData(index, QString("%1_%2").arg((type == GXS_ID_DETAILS_TYPE_DONE) ? TYPE_FOUND_ID : TYPE_UNKNOWN_ID).arg(text), ROLE_SORT); chooser->setItemData(index, (type == GXS_ID_DETAILS_TYPE_DONE) ? TYPE_FOUND_ID : TYPE_UNKNOWN_ID, ROLE_TYPE); chooser->setItemIcon(index, icons.empty() ? QIcon() : icons[0]); //std::cerr << "ID=" << details.mId << ", chooser->flags()=" << chooser->flags() << ", flags=" << details.mFlags ; if((chooser->flags() & IDCHOOSER_NON_ANONYMOUS) && !(details.mFlags & RS_IDENTITY_FLAGS_PGP_LINKED)) { //std::cerr << " - disabling ID - entry = " << index << std::endl; chooser->setEntryEnabled(index,false) ; } if(!chooser->isInConstraintSet(details.mId)) chooser->setEntryEnabled(index,false) ; chooser->model()->sort(0); // now restore the current item. Problem is, we cannot use the ID position because it may have changed. #ifdef IDCHOOSER_DEBUG std::cerr << "GxsIdChooser: default ID = " << chooser->defaultId() << std::endl; #endif if(!chooser->defaultId().isNull()) { for(int indx=0;indxcount();++indx) if(RsGxsId(chooser->itemData(indx).toString().toStdString()) == chooser->defaultId()) { chooser->setCurrentIndex(indx); #ifdef IDCHOOSER_DEBUG std::cerr << "GxsIdChooser-003 " << (void*)chooser << " setting current index to " << indx << " because it has ID=" << chooser->defaultId() << std::endl; #endif break; } } else { RsGxsId id; GxsIdChooser::ChosenId_Ret cid = chooser->getChosenId(id) ; if(cid == GxsIdChooser::UnKnowId || cid == GxsIdChooser::KnowId) chooser->setDefaultId(id) ; } chooser->blockSignals(false) ; } bool GxsIdChooser::isInConstraintSet(const RsGxsId& id) const { if(mConstraintIdsSet.empty()) // special case: empty set means no constraint return true ; return mConstraintIdsSet.find(id) != mConstraintIdsSet.end() ; } void GxsIdChooser::setEntryEnabled(int indx,bool /*enabled*/) { removeItem(indx) ; } uint32_t GxsIdChooser::countEnabledEntries() const { return count() ; } void GxsIdChooser::loadPrivateIds() { if (mFirstLoad) { //whileBlocking doesn't work here. bool prev = this->blockSignals(true); clear(); this->blockSignals(prev); } std::list ids; rsIdentity->getOwnIds(ids); //rsIdentity->getDefaultId(defId); // Prefer to use an application specific default??? if (mFirstLoad) { if (!(mFlags & IDCHOOSER_ID_REQUIRED)) { /* add No Signature option */ QString str = tr("No Signature"); QString id = ""; addItem(str, id); setItemData(count() - 1, QString("%1_%2").arg(TYPE_NO_ID).arg(str), ROLE_SORT); setItemData(count() - 1, TYPE_NO_ID, ROLE_TYPE); } } else { for (int idx = 0; idx < count(); ++idx) { QVariant type = itemData(idx, ROLE_TYPE); switch (type.toInt()) { case TYPE_NO_ID: case TYPE_CREATE_ID: break; case TYPE_FOUND_ID: case TYPE_UNKNOWN_ID: { QVariant var = itemData(idx); RsGxsId gxsId = RsGxsId(var.toString().toStdString()); std::list::iterator lit = std::find(ids.begin(), ids.end(), gxsId); if (lit == ids.end()) { removeItem(idx); idx--; } } } } } for (std::list::iterator it = ids.begin(); it != ids.end(); ++it) { GxsIdDetails::process(*it, loadPrivateIdsCallback, this); /* add to Chooser */ } if (mFirstLoad) { if (!(mFlags & IDCHOOSER_NO_CREATE)) { /* add Create Identity option */ QString str = tr("Create new Identity"); QString id = ""; addItem(QIcon(":/icons/png/add-identity.png"), str, id); setItemData(count() - 1, QString("%1_%2").arg(TYPE_CREATE_ID).arg(str), ROLE_SORT); setItemData(count() - 1, TYPE_CREATE_ID, ROLE_TYPE); } setDefaultItem(); emit idsLoaded(); } mFirstLoad = false; } void GxsIdChooser::setDefaultItem() { int def = -1; if ((mFlags & IDCHOOSER_ANON_DEFAULT) && !(mFlags & IDCHOOSER_ID_REQUIRED)) { def = findData(TYPE_NO_ID, ROLE_TYPE); } else { if (!mDefaultId.isNull()) { QString id = QString::fromStdString(mDefaultId.toStdString()); def = findData(id); } } if (def >= 0) { setCurrentIndex(def); #ifdef IDCHOOSER_DEBUG std::cerr << "GxsIdChooser-002" << (void*)this << " setting current index to " << def << std::endl; #endif } } bool GxsIdChooser::setChosenId(const RsGxsId &gxsId) { QString id = QString::fromStdString(gxsId.toStdString()); /* Find text of exisiting item */ int index = findData(id); if (index >= 0) { setCurrentIndex(index); #ifdef IDCHOOSER_DEBUG std::cerr << "GxsIdChooser-001" << (void*)this << " setting current index to " << index << std::endl; #endif return true; } return false; } GxsIdChooser::ChosenId_Ret GxsIdChooser::getChosenId(RsGxsId &gxsId) { if (count() < 1) { return None; } int idx = currentIndex(); QVariant var = itemData(idx); gxsId = RsGxsId(var.toString().toStdString()); QVariant type = itemData(idx, ROLE_TYPE); switch (type.toInt()) { case TYPE_NO_ID: return NoId; case TYPE_FOUND_ID: return KnowId; case TYPE_UNKNOWN_ID: return UnKnowId; case TYPE_CREATE_ID: break; } return None; } void GxsIdChooser::myCurrentIndexChanged(int index) { Q_UNUSED(index); QFontMetrics fm = QFontMetrics(font()); QString text = currentText(); if (width() < fm.boundingRect(text).width()) { setToolTip(text); } else { setToolTip(""); } QVariant var = itemData(index); RsGxsId gxsId(var.toString().toStdString()); if(!gxsId.isNull()) mDefaultId = gxsId; } void GxsIdChooser::indexActivated(int index) { int type = itemData(index, ROLE_TYPE).toInt(); if (type == TYPE_CREATE_ID) { IdEditDialog dlg(this); dlg.setupNewId(false, !(mFlags & IDCHOOSER_NON_ANONYMOUS)); if (dlg.exec() == QDialog::Accepted) { setDefaultId(RsGxsId(dlg.groupId())); } } } void GxsIdChooser::updateDisplay(bool reset) { if(reset) mFirstLoad = true ; /* Update identity list */ loadPrivateIds(); }