2018-11-14 15:14:40 -05:00
|
|
|
/*******************************************************************************
|
|
|
|
* retroshare-gui/src/gui/gxs/GxsIdDetails.cpp *
|
|
|
|
* *
|
|
|
|
* Copyright 2012-2013 by 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/>. *
|
|
|
|
* *
|
|
|
|
*******************************************************************************/
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
#include <QApplication>
|
|
|
|
#include <QThread>
|
|
|
|
#include <QTimerEvent>
|
|
|
|
#include <QMutexLocker>
|
|
|
|
|
2014-09-13 10:07:59 -04:00
|
|
|
#include <math.h>
|
2019-06-03 17:52:29 -04:00
|
|
|
#include <util/rsdir.h>
|
|
|
|
#include "gui/common/AvatarDialog.h"
|
2014-02-02 06:25:11 -05:00
|
|
|
#include "GxsIdDetails.h"
|
2015-03-09 16:41:14 -04:00
|
|
|
#include "retroshare-gui/RsAutoUpdatePage.h"
|
2014-02-02 06:25:11 -05:00
|
|
|
|
|
|
|
#include <retroshare/rspeers.h>
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
#include <QPainter>
|
|
|
|
|
|
|
|
/* Images for tag icons */
|
|
|
|
#define IMAGE_LOADING ":/images/folder-draft.png"
|
2020-02-22 07:24:30 -05:00
|
|
|
#define IMAGE_PGPKNOWN ":/images/contact22.png"
|
2014-04-08 09:28:58 -04:00
|
|
|
#define IMAGE_PGPUNKNOWN ":/images/tags/pgp-unknown.png"
|
|
|
|
#define IMAGE_ANON ":/images/tags/anon.png"
|
2018-12-06 17:04:53 -05:00
|
|
|
#define IMAGE_BANNED ":/icons/biohazard_red.png"
|
2014-04-08 09:28:58 -04:00
|
|
|
|
|
|
|
#define IMAGE_DEV_AMBASSADOR ":/images/tags/dev-ambassador.png"
|
|
|
|
#define IMAGE_DEV_CONTRIBUTOR ":/images/tags/vote_down.png"
|
|
|
|
#define IMAGE_DEV_TRANSLATOR ":/images/tags/dev-translator.png"
|
|
|
|
#define IMAGE_DEV_PATCHER ":/images/tags/dev-patcher.png"
|
|
|
|
#define IMAGE_DEV_DEVELOPER ":/images/tags/developer.png"
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2016-12-24 08:13:02 -05:00
|
|
|
#define REPUTATION_LOCALLY_POSITIVE_ICON ":/icons/bullet_green_yellow_star_128.png"
|
|
|
|
#define REPUTATION_REMOTELY_POSITIVE_ICON ":/icons/bullet_green_128.png"
|
|
|
|
#define REPUTATION_NEUTRAL_ICON ":/icons/bullet_grey_128.png"
|
2017-01-02 19:37:16 -05:00
|
|
|
#define REPUTATION_REMOTELY_NEGATIVE_ICON ":/icons/biohazard_yellow.png"
|
|
|
|
#define REPUTATION_LOCALLY_NEGATIVE_ICON ":/icons/biohazard_red.png"
|
2016-12-24 09:04:08 -05:00
|
|
|
#define REPUTATION_VOID ":/icons/void_128.png"
|
2016-12-24 08:13:02 -05:00
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
#define TIMER_INTERVAL 500
|
|
|
|
#define MAX_ATTEMPTS 10
|
|
|
|
#define MAX_PROCESS_COUNT_PER_TIMER 50
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2017-07-15 12:54:21 -04:00
|
|
|
//const int kRecognTagClass_DEVELOPMENT = 1;
|
|
|
|
//
|
|
|
|
//const int kRecognTagType_Dev_Ambassador = 1;
|
|
|
|
//const int kRecognTagType_Dev_Contributor = 2;
|
|
|
|
//const int kRecognTagType_Dev_Translator = 3;
|
|
|
|
//const int kRecognTagType_Dev_Patcher = 4;
|
|
|
|
//const int kRecognTagType_Dev_Developer = 5;
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2019-03-21 18:36:18 -04:00
|
|
|
uint32_t GxsIdDetails::mImagesAllocated = 0;
|
|
|
|
time_t GxsIdDetails::mLastIconCacheCleaning = time(NULL);
|
2019-06-04 05:49:26 -04:00
|
|
|
std::map<RsGxsId,std::pair<time_t,QPixmap>[4] > GxsIdDetails::mDefaultIconCache ;
|
2019-03-21 18:36:18 -04:00
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
QMutex GxsIdDetails::mMutex;
|
|
|
|
QMutex GxsIdDetails::mIconCacheMutex;
|
|
|
|
|
2019-06-05 13:47:06 -04:00
|
|
|
#define ICON_CACHE_STORAGE_TIME 240
|
|
|
|
#define DELAY_BETWEEN_ICON_CACHE_CLEANING 120
|
2016-12-24 09:04:08 -05:00
|
|
|
|
|
|
|
void ReputationItemDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
|
|
{
|
|
|
|
Q_ASSERT(index.isValid());
|
|
|
|
|
|
|
|
QStyleOptionViewItemV4 opt = option;
|
|
|
|
initStyleOption(&opt, index);
|
|
|
|
// disable default icon
|
|
|
|
opt.icon = QIcon();
|
|
|
|
// draw default item
|
|
|
|
QApplication::style()->drawControl(QStyle::CE_ItemViewItem, &opt, painter, 0);
|
|
|
|
|
|
|
|
const QRect r = option.rect;
|
|
|
|
|
|
|
|
// get pixmap
|
|
|
|
unsigned int icon_index = qvariant_cast<unsigned int>(index.data(Qt::DecorationRole));
|
|
|
|
|
|
|
|
if(icon_index > mMaxLevelToDisplay)
|
|
|
|
return ;
|
|
|
|
|
2019-02-16 09:42:22 -05:00
|
|
|
QIcon icon = GxsIdDetails::getReputationIcon(
|
|
|
|
RsReputationLevel(icon_index), 0xff );
|
2016-12-24 09:04:08 -05:00
|
|
|
|
|
|
|
QPixmap pix = icon.pixmap(r.size());
|
|
|
|
|
|
|
|
// draw pixmap at center of item
|
|
|
|
const QPoint p = QPoint((r.width() - pix.width())/2, (r.height() - pix.height())/2);
|
|
|
|
painter->drawPixmap(r.topLeft() + p, pix);
|
|
|
|
}
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
/* The global object */
|
2015-02-04 18:05:56 -05:00
|
|
|
GxsIdDetails *GxsIdDetails::mInstance = NULL ;
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
GxsIdDetails::GxsIdDetails()
|
|
|
|
: QObject()
|
|
|
|
{
|
|
|
|
mCheckTimerId = 0;
|
2015-06-26 07:44:45 -04:00
|
|
|
mProcessDisableCount = 0;
|
2016-04-11 23:30:42 -04:00
|
|
|
mPendingDataIterator = mPendingData.end() ;
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
connect(this, SIGNAL(startTimerFromThread()), this, SLOT(doStartTimer()));
|
|
|
|
}
|
|
|
|
|
|
|
|
GxsIdDetails::~GxsIdDetails()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-04-20 18:28:19 -04:00
|
|
|
void GxsIdDetails::initialize()
|
|
|
|
{
|
|
|
|
if (mInstance) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
mInstance = new GxsIdDetails;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GxsIdDetails::cleanup()
|
|
|
|
{
|
|
|
|
if (!mInstance) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(mInstance);
|
|
|
|
mInstance = NULL;
|
|
|
|
}
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
void GxsIdDetails::objectDestroyed(QObject *object)
|
|
|
|
{
|
|
|
|
if (!object) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QMutexLocker lock(&mMutex);
|
|
|
|
|
|
|
|
/* Object is about to be destroyed, remove it from pending list */
|
|
|
|
QList<CallbackData>::iterator dataIt;
|
2016-04-11 23:30:42 -04:00
|
|
|
|
|
|
|
QMap<QObject*,CallbackData>::iterator it = mPendingData.find(object) ;
|
|
|
|
|
|
|
|
if(it != mPendingData.end())
|
|
|
|
{
|
|
|
|
if(it == mPendingDataIterator)
|
|
|
|
mPendingDataIterator = mPendingData.erase(it) ;
|
|
|
|
else
|
|
|
|
mPendingData.erase(it) ;
|
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void GxsIdDetails::connectObject_locked(QObject *object, bool doConnect)
|
|
|
|
{
|
|
|
|
if (!object) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Search Object in pending list */
|
2016-04-11 23:30:42 -04:00
|
|
|
|
2016-04-12 10:06:01 -04:00
|
|
|
if(mPendingData.find(object) == mPendingData.end()) // force disconnect when not in the list
|
|
|
|
doConnect = false ;
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
if (doConnect) {
|
|
|
|
connect(object, SIGNAL(destroyed(QObject*)), this, SLOT(objectDestroyed(QObject*)));
|
|
|
|
} else {
|
|
|
|
disconnect(object, SIGNAL(destroyed(QObject*)), this, SLOT(objectDestroyed(QObject*)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GxsIdDetails::timerEvent(QTimerEvent *event)
|
|
|
|
{
|
|
|
|
if (event->timerId() == mCheckTimerId) {
|
2015-03-09 16:41:14 -04:00
|
|
|
if (!RsAutoUpdatePage::eventsLocked()) {
|
|
|
|
/* Stop timer */
|
|
|
|
killTimer(mCheckTimerId);
|
|
|
|
mCheckTimerId = 0;
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
if (rsIdentity)
|
|
|
|
{
|
2015-03-09 16:41:14 -04:00
|
|
|
QMutexLocker lock(&mMutex);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
if (mProcessDisableCount == 0) {
|
|
|
|
if (!mPendingData.empty()) {
|
|
|
|
/* Check pending id's */
|
|
|
|
int processed = qMin(MAX_PROCESS_COUNT_PER_TIMER, mPendingData.size());
|
2016-04-11 23:30:42 -04:00
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
while (!mPendingData.isEmpty()) {
|
|
|
|
if (processed-- <= 0) {
|
|
|
|
break;
|
|
|
|
}
|
2016-04-11 23:30:42 -04:00
|
|
|
|
|
|
|
if(mPendingDataIterator == mPendingData.end())
|
|
|
|
mPendingDataIterator = mPendingData.begin() ;
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2016-04-11 23:30:42 -04:00
|
|
|
CallbackData &pendingData = *mPendingDataIterator;
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
RsIdentityDetails details;
|
|
|
|
if (rsIdentity->getIdDetails(pendingData.mId, details)) {
|
|
|
|
/* Got details */
|
|
|
|
pendingData.mCallback(GXS_ID_DETAILS_TYPE_DONE, details, pendingData.mObject, pendingData.mData);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
QObject *object = pendingData.mObject;
|
|
|
|
connectObject_locked(object, false);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2016-04-12 10:06:01 -04:00
|
|
|
mPendingDataIterator = mPendingData.erase(mPendingDataIterator);
|
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
continue;
|
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
if (++pendingData.mAttempt > MAX_ATTEMPTS) {
|
|
|
|
/* Max attempts reached, stop trying */
|
|
|
|
details.mId = pendingData.mId;
|
|
|
|
pendingData.mCallback(GXS_ID_DETAILS_TYPE_FAILED, details, pendingData.mObject, pendingData.mData);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
QObject *object = pendingData.mObject;
|
|
|
|
connectObject_locked(object, false);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2016-04-12 10:06:01 -04:00
|
|
|
mPendingDataIterator = mPendingData.erase(mPendingDataIterator);
|
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
continue;
|
|
|
|
}
|
2016-04-11 23:30:42 -04:00
|
|
|
|
|
|
|
++mPendingDataIterator ;
|
|
|
|
|
|
|
|
//mPendingData.move(0, mPendingData.size() - 1);
|
2015-06-26 07:44:45 -04:00
|
|
|
}
|
2015-03-09 16:41:14 -04:00
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
bool empty = false ;
|
|
|
|
{
|
|
|
|
QMutexLocker lock(&mMutex);
|
|
|
|
empty = mPendingData.empty();
|
|
|
|
}
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
if (!empty) /* Start timer */
|
2015-03-09 16:41:14 -04:00
|
|
|
doStartTimer();
|
2014-11-26 19:55:48 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QObject::timerEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GxsIdDetails::doStartTimer()
|
|
|
|
{
|
|
|
|
if (mCheckTimerId) {
|
|
|
|
/* Timer is running */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
mCheckTimerId = startTimer(TIMER_INTERVAL);
|
|
|
|
}
|
|
|
|
|
2015-06-26 07:44:45 -04:00
|
|
|
void GxsIdDetails::enableProcess(bool enable)
|
|
|
|
{
|
|
|
|
if (!mInstance) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
QMutexLocker lock(&mMutex);
|
2015-06-26 07:44:45 -04:00
|
|
|
|
|
|
|
if (enable) {
|
|
|
|
--mInstance->mProcessDisableCount;
|
|
|
|
if (mInstance->mProcessDisableCount < 0) {
|
|
|
|
mInstance->mProcessDisableCount = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
++mInstance->mProcessDisableCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
bool GxsIdDetails::process(const RsGxsId &id, GxsIdDetailsCallbackFunction callback, QObject *object, const QVariant &data)
|
|
|
|
{
|
|
|
|
if (!callback) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsIdentityDetails details;
|
|
|
|
if (id.isNull()) {
|
|
|
|
callback(GXS_ID_DETAILS_TYPE_EMPTY, details, object, data);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-04-11 23:30:42 -04:00
|
|
|
// remove any existing call for this object. This is needed for when the same widget is used to display IDs that vary in time.
|
|
|
|
{
|
2019-06-07 09:03:48 -04:00
|
|
|
QMutexLocker lock(&mMutex);
|
2016-04-11 23:30:42 -04:00
|
|
|
|
|
|
|
// check if a pending request is not already on its way. If so, replace it.
|
|
|
|
|
|
|
|
QMap<QObject*,CallbackData>::iterator it = mInstance->mPendingData.find(object) ;
|
|
|
|
|
|
|
|
if(it != mInstance->mPendingData.end())
|
|
|
|
{
|
|
|
|
mInstance->connectObject_locked(object, false);
|
2016-04-13 18:41:49 -04:00
|
|
|
|
|
|
|
if(mInstance->mPendingDataIterator == it)
|
|
|
|
mInstance->mPendingDataIterator = mInstance->mPendingData.erase(it) ;
|
|
|
|
else
|
|
|
|
mInstance->mPendingData.erase(it) ;
|
2016-04-11 23:30:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Connect signal "destroy" */
|
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
/* Try to get the information */
|
2015-02-11 02:00:52 -05:00
|
|
|
// the idea behind this was, to call the callback directly when the identity is already loaded in librs
|
|
|
|
// without one timer tick, but it causes the use of Pixmap in avatars within a threat that is different than
|
|
|
|
// the GUI thread, which is not allowed by Qt => some avatars fail to load.
|
|
|
|
|
2015-02-22 10:06:27 -05:00
|
|
|
bool isGuiThread = (QThread::currentThread() == qApp->thread());
|
2015-03-16 08:57:12 -04:00
|
|
|
if (isGuiThread && !RsAutoUpdatePage::eventsLocked() && rsIdentity && rsIdentity->getIdDetails(id, details)) {
|
2015-02-22 10:06:27 -05:00
|
|
|
callback(GXS_ID_DETAILS_TYPE_DONE, details, object, data);
|
|
|
|
return true;
|
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
details.mId = id;
|
|
|
|
|
|
|
|
/* Add id to the pending list */
|
|
|
|
if (!mInstance) {
|
2015-04-20 18:28:19 -04:00
|
|
|
/* GxsIdDetails not initialized. Please use ::initialize */
|
|
|
|
callback(GXS_ID_DETAILS_TYPE_FAILED, details, object, data);
|
|
|
|
return false;
|
2014-11-26 19:55:48 -05:00
|
|
|
}
|
2015-04-20 18:28:19 -04:00
|
|
|
|
|
|
|
callback(GXS_ID_DETAILS_TYPE_LOADING, details, object, data);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
CallbackData pendingData;
|
|
|
|
pendingData.mId = id;
|
|
|
|
pendingData.mCallback = callback;
|
|
|
|
pendingData.mObject = object;
|
|
|
|
pendingData.mData = data;
|
|
|
|
|
|
|
|
{
|
2019-06-07 09:03:48 -04:00
|
|
|
QMutexLocker lock(&mMutex);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2016-04-11 23:30:42 -04:00
|
|
|
// check if a pending request is not already on its way. If so, replace it.
|
|
|
|
|
|
|
|
mInstance->mPendingData[object] = pendingData;
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
/* Connect signal "destroy" */
|
|
|
|
mInstance->connectObject_locked(object, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Start timer */
|
2015-02-22 10:06:27 -05:00
|
|
|
if (isGuiThread) {
|
2014-11-26 19:55:48 -05:00
|
|
|
/* Start timer directly */
|
|
|
|
mInstance->doStartTimer();
|
|
|
|
} else {
|
|
|
|
/* Send signal to start timer in main thread */
|
|
|
|
emit mInstance->startTimerFromThread();
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool findTagIcon(int tag_class, int /*tag_type*/, QIcon &icon)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
|
|
|
switch(tag_class)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
case 0:
|
|
|
|
icon = QIcon(IMAGE_DEV_AMBASSADOR);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
icon = QIcon(IMAGE_DEV_CONTRIBUTOR);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-02-02 16:59:44 -05:00
|
|
|
/**
|
|
|
|
* @brief GxsIdDetails::makeIdentIcon
|
|
|
|
* @param id: RsGxsId to compute
|
|
|
|
* @return QImage representing an IndentIcon cf:
|
|
|
|
* http://en.wikipedia.org/wiki/Identicon
|
|
|
|
* Bring the source code from this adaptation:
|
|
|
|
* http://francisshanahan.com/identicon5/test.html
|
|
|
|
*/
|
2019-06-04 05:49:26 -04:00
|
|
|
const QPixmap GxsIdDetails::makeDefaultIcon(const RsGxsId& id, AvatarSize size)
|
2015-02-04 18:05:56 -05:00
|
|
|
{
|
2019-06-03 17:52:29 -04:00
|
|
|
checkCleanImagesCache();
|
|
|
|
|
2019-03-21 18:36:18 -04:00
|
|
|
// We use a cache for images. QImage has its own smart pointer system, but it does not prevent
|
|
|
|
// the same image to be allocated many times. We do this using a cache. The cache is also cleaned-up
|
|
|
|
// on a regular time basis so as to get rid of unused images.
|
|
|
|
|
|
|
|
time_t now = time(NULL);
|
|
|
|
|
2019-06-03 17:52:29 -04:00
|
|
|
// now look for the icon
|
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
QMutexLocker lock(&mIconCacheMutex);
|
2019-06-04 05:49:26 -04:00
|
|
|
auto& it = mDefaultIconCache[id];
|
2019-06-03 17:52:29 -04:00
|
|
|
|
2019-06-04 05:49:26 -04:00
|
|
|
if(it[(int)size].second.width() > 0)
|
2019-06-03 17:52:29 -04:00
|
|
|
{
|
2019-06-04 05:49:26 -04:00
|
|
|
it[(int)size].first = now;
|
|
|
|
return it[(int)size].second;
|
2019-06-03 17:52:29 -04:00
|
|
|
}
|
|
|
|
|
2019-06-04 05:49:26 -04:00
|
|
|
int S =0;
|
2019-06-03 17:52:29 -04:00
|
|
|
|
2019-06-04 05:49:26 -04:00
|
|
|
switch(size)
|
|
|
|
{
|
|
|
|
case SMALL: S = 16*3 ; break;
|
|
|
|
default:
|
|
|
|
case MEDIUM: S = 32*3 ; break;
|
|
|
|
case ORIGINAL:
|
|
|
|
case LARGE: S = 64*3 ; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPixmap image = drawIdentIcon(QString::fromStdString(id.toStdString()),S,true);
|
|
|
|
|
|
|
|
it[(int)size] = std::make_pair(now,image);
|
2019-06-03 17:52:29 -04:00
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GxsIdDetails::checkCleanImagesCache()
|
|
|
|
{
|
|
|
|
time_t now = time(NULL);
|
|
|
|
|
2019-03-21 18:36:18 -04:00
|
|
|
// cleanup the cache every 10 mins
|
|
|
|
|
|
|
|
if(mLastIconCacheCleaning + DELAY_BETWEEN_ICON_CACHE_CLEANING < now)
|
|
|
|
{
|
|
|
|
std::cerr << "(II) Cleaning the icons cache." << std::endl;
|
|
|
|
int nb_deleted = 0;
|
2019-06-04 07:01:35 -04:00
|
|
|
uint32_t size_deleted = 0;
|
|
|
|
uint32_t total_size = 0;
|
2019-03-21 18:36:18 -04:00
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
QMutexLocker lock(&mIconCacheMutex);
|
2019-06-06 08:21:19 -04:00
|
|
|
|
2019-03-21 18:36:18 -04:00
|
|
|
for(auto it(mDefaultIconCache.begin());it!=mDefaultIconCache.end();)
|
2019-06-04 05:49:26 -04:00
|
|
|
{
|
|
|
|
bool all_empty = true ;
|
|
|
|
|
|
|
|
for(int i=0;i<4;++i)
|
|
|
|
if(it->second[i].first + ICON_CACHE_STORAGE_TIME < now && it->second[i].second.isDetached())
|
|
|
|
{
|
2019-06-04 07:01:35 -04:00
|
|
|
int s = it->second[i].second.width()*it->second[i].second.height()*4;
|
|
|
|
|
|
|
|
std::cerr << "Deleting pixmap " << it->first << " size " << i << " " << s << " bytes." << std::endl;
|
2019-06-04 05:49:26 -04:00
|
|
|
|
2019-06-04 07:01:35 -04:00
|
|
|
it->second[i].second = QPixmap();
|
2019-06-04 05:49:26 -04:00
|
|
|
++nb_deleted;
|
2019-06-04 07:01:35 -04:00
|
|
|
size_deleted += s;
|
2019-06-04 05:49:26 -04:00
|
|
|
}
|
|
|
|
else
|
2019-06-04 07:01:35 -04:00
|
|
|
{
|
2019-06-04 05:49:26 -04:00
|
|
|
all_empty = false;
|
2019-06-04 07:01:35 -04:00
|
|
|
total_size += it->second[i].second.width()*it->second[i].second.height()*4;
|
|
|
|
}
|
2019-06-04 05:49:26 -04:00
|
|
|
|
|
|
|
if(all_empty)
|
2019-03-21 18:36:18 -04:00
|
|
|
it = mDefaultIconCache.erase(it);
|
|
|
|
else
|
|
|
|
++it;
|
2019-06-04 05:49:26 -04:00
|
|
|
}
|
2019-03-21 18:36:18 -04:00
|
|
|
|
|
|
|
mLastIconCacheCleaning = now;
|
2019-06-04 07:01:35 -04:00
|
|
|
std::cerr << "(II) Removed " << nb_deleted << " (" << size_deleted << " bytes) unused icons. Cache contains " << mDefaultIconCache.size() << " icons (" << total_size << " bytes)"<< std::endl;
|
2019-03-21 18:36:18 -04:00
|
|
|
}
|
2019-06-03 17:52:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-04 05:49:26 -04:00
|
|
|
bool GxsIdDetails::loadPixmapFromData(const unsigned char *data,size_t data_len,QPixmap& pixmap, AvatarSize size)
|
2019-06-03 17:52:29 -04:00
|
|
|
{
|
|
|
|
// The trick below converts the data into an Id that can be read in the image cache. Because this method is mainly dedicated to loading
|
|
|
|
// avatars, we could also use the GxsId as id, but the avatar may change in time, so we actually need to make the id from the data itself.
|
|
|
|
|
|
|
|
assert(Sha1CheckSum::SIZE_IN_BYTES >= RsGxsId::SIZE_IN_BYTES);
|
|
|
|
|
|
|
|
Sha1CheckSum chksum = RsDirUtil::sha1sum(data,data_len);
|
|
|
|
RsGxsId id(chksum.toByteArray());
|
|
|
|
|
|
|
|
// We use a cache for images. QImage has its own smart pointer system, but it does not prevent
|
|
|
|
// the same image to be allocated many times. We do this using a cache. The cache is also cleaned-up
|
|
|
|
// on a regular time basis so as to get rid of unused images.
|
|
|
|
|
|
|
|
checkCleanImagesCache();
|
2019-03-21 18:36:18 -04:00
|
|
|
|
|
|
|
// now look for the icon
|
|
|
|
|
2019-06-07 09:03:48 -04:00
|
|
|
QMutexLocker lock(&mIconCacheMutex);
|
2019-06-06 08:21:19 -04:00
|
|
|
|
2019-06-03 17:52:29 -04:00
|
|
|
time_t now = time(NULL);
|
2019-06-04 05:49:26 -04:00
|
|
|
auto& it = mDefaultIconCache[id];
|
2019-03-21 18:36:18 -04:00
|
|
|
|
2019-06-04 05:49:26 -04:00
|
|
|
if(it[(int)size].second.width() > 0)
|
2019-03-21 18:36:18 -04:00
|
|
|
{
|
2019-06-04 05:49:26 -04:00
|
|
|
it[(int)size].first = now;
|
|
|
|
pixmap = it[(int)size].second;
|
2019-06-03 17:52:29 -04:00
|
|
|
|
|
|
|
return true;
|
2019-03-21 18:36:18 -04:00
|
|
|
}
|
|
|
|
|
2019-06-03 17:52:29 -04:00
|
|
|
if(! pixmap.loadFromData(data,data_len))
|
|
|
|
return false;
|
2019-03-21 18:36:18 -04:00
|
|
|
|
2019-06-03 17:52:29 -04:00
|
|
|
// This resize is here just to prevent someone to explicitely add a huge blank image to screw up the UI
|
2019-03-21 18:36:18 -04:00
|
|
|
|
2019-06-04 05:49:26 -04:00
|
|
|
int wanted_S=0;
|
|
|
|
|
|
|
|
switch(size)
|
|
|
|
{
|
|
|
|
case ORIGINAL: wanted_S = 0 ;break;
|
|
|
|
case SMALL: wanted_S = 32 ;break;
|
|
|
|
default:
|
|
|
|
case MEDIUM: wanted_S = 64 ;break;
|
|
|
|
case LARGE: wanted_S = 128 ;break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(wanted_S > 0)
|
|
|
|
pixmap = pixmap.scaled(wanted_S,wanted_S,Qt::IgnoreAspectRatio,Qt::SmoothTransformation);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2019-06-04 05:49:26 -04:00
|
|
|
mDefaultIconCache[id][(int)size] = std::make_pair(now,pixmap);
|
2019-06-05 13:47:06 -04:00
|
|
|
#ifdef DEBUG
|
2019-06-04 05:49:26 -04:00
|
|
|
std::cerr << "Allocated new icon " << id << " size " << (int)size << std::endl;
|
2019-06-05 13:47:06 -04:00
|
|
|
#endif
|
2019-06-03 17:52:29 -04:00
|
|
|
return true;
|
|
|
|
}
|
2015-02-02 16:59:44 -05:00
|
|
|
/**
|
|
|
|
* @brief GxsIdDetails::getSprite
|
|
|
|
* @param shapeType: type of shape (0 to 15)
|
|
|
|
* @param size: Size for shape
|
|
|
|
* @return QList<qreal> for all point for path drawing (shape)
|
|
|
|
*/
|
|
|
|
QList<qreal> GxsIdDetails::getSprite(quint8 shapeType, quint16 size)
|
2014-11-26 19:55:48 -05:00
|
|
|
{
|
2015-02-02 16:59:44 -05:00
|
|
|
QList<qreal> sprite;
|
|
|
|
switch (shapeType) {
|
|
|
|
case 0: // Triangle
|
|
|
|
sprite.append(0.5);sprite.append(1.0);
|
|
|
|
sprite.append(1.0);sprite.append(0.0);
|
|
|
|
sprite.append(1.0);sprite.append(1.0);
|
|
|
|
break;
|
|
|
|
case 1: // Parallelogram
|
|
|
|
sprite.append(0.5);sprite.append(0.0);
|
|
|
|
sprite.append(1.0);sprite.append(0.0);
|
|
|
|
sprite.append(0.5);sprite.append(1.0);
|
|
|
|
sprite.append(0.0);sprite.append(1.0);
|
|
|
|
break;
|
|
|
|
case 2: // mouse ears
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 3: // Ribbon
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 4: // Sails
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 5: // Fins
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 6: // Beak
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
case 7: // Chevron
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 8: // Fish
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 9: // Kite
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
case 10: // Trough
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
case 11: // Rays
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(1);sprite.append(0.75);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0.25);
|
|
|
|
break;
|
|
|
|
case 12: // Double rhombus
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
case 13: // Crown
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0.25);
|
|
|
|
sprite.append(0.5);sprite.append(0.75);
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0.25);
|
|
|
|
break;
|
|
|
|
case 14: // Radioactive
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
default: // Tiles
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scale up
|
|
|
|
for (quint8 i = 0; i < sprite.size(); ++i) {
|
|
|
|
sprite[i] = sprite[i] * size;
|
2014-11-26 19:55:48 -05:00
|
|
|
}
|
|
|
|
|
2015-02-02 16:59:44 -05:00
|
|
|
return sprite;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief GxsIdDetails::getCenter
|
|
|
|
* @param shapeType: type of shape (0 to 15)
|
|
|
|
* @param size: Size for shape
|
|
|
|
* @return QList<qreal> for all point for path drawing (shape) in center
|
|
|
|
*/
|
|
|
|
QList<qreal> GxsIdDetails::getCenter(quint8 shapeType, quint16 size)
|
|
|
|
{
|
|
|
|
QList<qreal> sprite;
|
|
|
|
switch (shapeType) {
|
|
|
|
case 0: // Empty
|
|
|
|
sprite.clear();
|
|
|
|
break;
|
|
|
|
case 1: // Fill
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
case 2: // Diamond
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 3: // Reverse diamond
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 4: // Cross
|
|
|
|
sprite.append(0.25);sprite.append(0);
|
|
|
|
sprite.append(0.75);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0.25);
|
|
|
|
sprite.append(1);sprite.append(0.75);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0.75);sprite.append(1);
|
|
|
|
sprite.append(0.25);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(0.75);
|
|
|
|
sprite.append(0);sprite.append(0.25);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 5: // Morning star
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.25);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(0.75);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.75);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
sprite.append(0.25);sprite.append(0.5);
|
|
|
|
break;
|
|
|
|
case 6: // Small square
|
|
|
|
sprite.append(0.33);sprite.append(0.33);
|
|
|
|
sprite.append(0.67);sprite.append(0.33);
|
|
|
|
sprite.append(0.67);sprite.append(0.67);
|
|
|
|
sprite.append(0.33);sprite.append(0.67);
|
|
|
|
break;
|
|
|
|
case 7: // Checkerboard
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(0.33);sprite.append(0);
|
|
|
|
sprite.append(0.33);sprite.append(0.33);
|
|
|
|
sprite.append(0.66);sprite.append(0.33);
|
|
|
|
sprite.append(0.67);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0.33);
|
|
|
|
sprite.append(0.67);sprite.append(0.33);
|
|
|
|
sprite.append(0.67);sprite.append(0.67);
|
|
|
|
sprite.append(1);sprite.append(0.67);
|
|
|
|
sprite.append(1);sprite.append(1);
|
|
|
|
sprite.append(0.67);sprite.append(1);
|
|
|
|
sprite.append(0.67);sprite.append(0.67);
|
|
|
|
sprite.append(0.33);sprite.append(0.67);
|
|
|
|
sprite.append(0.33);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
sprite.append(0);sprite.append(0.67);
|
|
|
|
sprite.append(0.33);sprite.append(0.67);
|
|
|
|
sprite.append(0.33);sprite.append(0.33);
|
|
|
|
sprite.append(0);sprite.append(0.33);
|
|
|
|
break;
|
|
|
|
default: // Tiles
|
|
|
|
sprite.append(0);sprite.append(0);
|
|
|
|
sprite.append(1);sprite.append(0);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(0);
|
|
|
|
sprite.append(0);sprite.append(0.5);
|
|
|
|
sprite.append(1);sprite.append(0.5);
|
|
|
|
sprite.append(0.5);sprite.append(1);
|
|
|
|
sprite.append(0.5);sprite.append(0.5);
|
|
|
|
sprite.append(0);sprite.append(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* apply ratios */
|
|
|
|
for (quint8 i = 0; i < sprite.size(); ++i) {
|
|
|
|
sprite[i] = sprite[i] * size;
|
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2015-02-02 16:59:44 -05:00
|
|
|
return sprite;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief GxsIdDetails::drawRotatedPolygon
|
|
|
|
* @param pixmap: The pixmap write to draw
|
|
|
|
* @param sprite: path to follow
|
|
|
|
* @param x: Offset to start
|
|
|
|
* @param y: Offset to start
|
|
|
|
* @param shapeangle: Angle of shape to draw
|
|
|
|
* @param angle: Angle (where 3x3 in picture)
|
|
|
|
* @param size: Size of shape
|
|
|
|
* @param fillColor: Color to fill shape
|
|
|
|
*/
|
|
|
|
void GxsIdDetails::drawRotatedPolygon( QPixmap *pixmap,
|
|
|
|
QList<qreal> sprite,
|
|
|
|
quint16 x, quint16 y,
|
|
|
|
qreal shapeangle, qreal angle,
|
|
|
|
quint16 size, QColor fillColor)
|
|
|
|
{
|
|
|
|
qreal halfSize = size / 2;
|
|
|
|
QPainter painter (pixmap);
|
|
|
|
painter.save();
|
|
|
|
painter.setBrush(fillColor);
|
|
|
|
painter.setPen(fillColor);
|
|
|
|
|
|
|
|
painter.translate(x, y);
|
|
|
|
painter.rotate(angle);
|
|
|
|
painter.save();
|
|
|
|
painter.translate(halfSize, halfSize);
|
|
|
|
QList<qreal> tmpSprite;
|
|
|
|
for (quint8 p = 0; p < sprite.size(); ++p) {
|
|
|
|
tmpSprite.append(sprite[p] - halfSize);
|
|
|
|
}
|
|
|
|
painter.rotate(shapeangle);
|
|
|
|
if (tmpSprite.size() >= 2) {
|
|
|
|
QPainterPath path;
|
|
|
|
path.setFillRule(Qt::WindingFill);
|
|
|
|
path.moveTo(tmpSprite[0], tmpSprite[1]);
|
|
|
|
for (int i = 2; i < tmpSprite.size(); ++++i) {
|
|
|
|
path.lineTo(tmpSprite[i], tmpSprite[i + 1]);
|
2014-11-26 19:55:48 -05:00
|
|
|
}
|
2015-02-02 16:59:44 -05:00
|
|
|
//path.lineTo(tmpSprite[0], tmpSprite[1]);
|
|
|
|
painter.drawPath(path);
|
|
|
|
}
|
|
|
|
// Black outline for debugging
|
|
|
|
//painter.drawRect(-halfSize, -halfSize, size, size);
|
|
|
|
painter.restore();
|
|
|
|
painter.restore();
|
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2015-02-02 16:59:44 -05:00
|
|
|
/**
|
|
|
|
* @brief GxsIdDetails::drawIdentIcon
|
|
|
|
* @param hash: Hash to compute (min 20 char)
|
|
|
|
* @param width: Size of picture to get
|
|
|
|
* @param rotate: If the shapes could be rotated
|
|
|
|
* @return QImage of computed hash
|
|
|
|
*/
|
2019-06-03 17:52:29 -04:00
|
|
|
QPixmap GxsIdDetails::drawIdentIcon( QString hash, quint16 width, bool rotate)
|
2015-02-02 16:59:44 -05:00
|
|
|
{
|
|
|
|
bool ok;
|
|
|
|
quint8 csh = hash.mid(0, 1).toInt(&ok,16);// Corner sprite shape
|
|
|
|
quint8 ssh = hash.mid(1, 1).toInt(&ok, 16); // Side sprite shape
|
|
|
|
quint8 xsh = hash.mid(2, 1).toInt(&ok, 16) & 7; // Center sprite shape
|
|
|
|
|
|
|
|
qreal halfPi = 90;// M_PI/2;
|
|
|
|
qreal cro = rotate ? halfPi * (hash.mid(3, 1).toInt(&ok, 16) & 3) : 0; // Corner sprite rotation
|
|
|
|
qreal sro = rotate ? halfPi * (hash.mid(4, 1).toInt(&ok, 16) & 3) : 0; // Side sprite rotation
|
|
|
|
quint8 xbg = hash.mid(5, 1).toInt(&ok, 16) % 2; // Center sprite background
|
|
|
|
|
|
|
|
// Corner sprite foreground color
|
|
|
|
quint8 cfr = hash.mid(6, 2).toInt(&ok, 16);
|
|
|
|
quint8 cfg = hash.mid(8, 2).toInt(&ok, 16);
|
|
|
|
quint8 cfb = hash.mid(10, 2).toInt(&ok, 16);
|
|
|
|
|
|
|
|
// Side sprite foreground color
|
|
|
|
quint8 sfr = hash.mid(12, 2).toInt(&ok, 16);
|
|
|
|
quint8 sfg = hash.mid(14, 2).toInt(&ok, 16);
|
|
|
|
quint8 sfb = hash.mid(16, 2).toInt(&ok, 16);
|
|
|
|
|
|
|
|
// Final angle of rotation
|
|
|
|
// not used
|
|
|
|
//int angle = hash.mid(18, 2).toInt(&ok, 16);
|
|
|
|
|
|
|
|
/* Size of each sprite */
|
|
|
|
quint16 size = width / 3;
|
|
|
|
quint16 totalsize = width;
|
|
|
|
|
|
|
|
/// start with blank 3x3 identicon
|
|
|
|
QPixmap pixmap = QPixmap(totalsize, totalsize);
|
2015-02-03 17:16:21 -05:00
|
|
|
pixmap.fill(QColor::fromRgb(230,230,230));
|
2015-02-02 16:59:44 -05:00
|
|
|
|
|
|
|
// Generate corner sprites
|
|
|
|
QList<qreal> corner = getSprite(csh, size);
|
|
|
|
QColor fillCorner = QColor( cfr, cfg, cfb );
|
|
|
|
drawRotatedPolygon(&pixmap, corner, 0, 0, cro, 0, size, fillCorner);
|
|
|
|
drawRotatedPolygon(&pixmap, corner, totalsize, 0, cro, 90, size, fillCorner);
|
|
|
|
drawRotatedPolygon(&pixmap, corner, totalsize, totalsize, cro, 180, size, fillCorner);
|
|
|
|
drawRotatedPolygon(&pixmap, corner, 0, totalsize, cro, 270, size, fillCorner);
|
|
|
|
|
|
|
|
// Draw sides
|
|
|
|
QList<qreal> side = getSprite(ssh, size);
|
|
|
|
QColor fillSide = QColor( sfr, sfg, sfb);
|
|
|
|
drawRotatedPolygon(&pixmap, side, 0, size, sro, 0, size, fillSide);
|
|
|
|
drawRotatedPolygon(&pixmap, side, 2 * size, 0, sro, 90, size, fillSide);
|
|
|
|
drawRotatedPolygon(&pixmap, side, 3 * size, 2 * size, sro, 180, size, fillSide);
|
|
|
|
drawRotatedPolygon(&pixmap, side, size, 3 * size, sro, 270, size, fillSide);
|
|
|
|
|
|
|
|
// Draw center
|
|
|
|
QList<qreal> center = getCenter(xsh, size);
|
|
|
|
// Make sure there's enough contrast before we use background color of side sprite
|
|
|
|
QColor fillCenter;
|
|
|
|
if (xbg > 0 && (abs(cfr - sfr) > 127 || abs(cfg - sfg) > 127 || abs(cfb - sfb) > 127)) {
|
|
|
|
fillCenter = QColor( sfr, sfg, sfb);
|
|
|
|
} else {
|
|
|
|
fillCenter = QColor( cfr, cfg, cfb);
|
|
|
|
}
|
|
|
|
drawRotatedPolygon(&pixmap, center, size, size, 0, 0, size, fillCenter);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2019-06-03 17:52:29 -04:00
|
|
|
return pixmap;
|
2014-09-13 10:07:59 -04:00
|
|
|
}
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
//static bool CreateIdIcon(const RsGxsId &id, QIcon &idIcon)
|
|
|
|
//{
|
|
|
|
// QPixmap image(IconSize, IconSize);
|
|
|
|
// QPainter painter(&image);
|
|
|
|
|
|
|
|
// painter.fillRect(0, 0, IconSize, IconSize, Qt::black);
|
|
|
|
|
|
|
|
// int len = id.SIZE_IN_BYTES;
|
|
|
|
// for(int i = 0; i<len; ++i)
|
|
|
|
// {
|
|
|
|
// unsigned char hex = id.toByteArray()[i];
|
|
|
|
// int x = hex & 0xf ;
|
|
|
|
// int y = (hex & 0xf0) >> 4 ;
|
|
|
|
// painter.fillRect(x, y, x+1, y+1, Qt::green);
|
|
|
|
// }
|
|
|
|
// idIcon = QIcon(image);
|
|
|
|
// return true;
|
|
|
|
//}
|
|
|
|
|
|
|
|
QString GxsIdDetails::getLoadingText(const RsGxsId &id)
|
|
|
|
{
|
|
|
|
return QString("%1... %2").arg(QApplication::translate("GxsIdDetails", "Loading"), QString::fromStdString(id.toStdString().substr(0, 5)));
|
|
|
|
}
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
QString GxsIdDetails::getFailedText(const RsGxsId &id)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
2014-11-26 19:55:48 -05:00
|
|
|
return QString("%1 ...[%2]").arg(QApplication::translate("GxsIdDetails", "Not found"), QString::fromStdString(id.toStdString().substr(0, 5)));
|
|
|
|
}
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
QString GxsIdDetails::getEmptyIdText()
|
|
|
|
{
|
2018-08-30 05:48:37 -04:00
|
|
|
return QApplication::translate("GxsIdDetails", "[None]");
|
2014-11-26 19:55:48 -05:00
|
|
|
}
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
QString GxsIdDetails::getNameForType(GxsIdDetailsType type, const RsIdentityDetails &details)
|
|
|
|
{
|
|
|
|
switch (type) {
|
|
|
|
case GXS_ID_DETAILS_TYPE_EMPTY:
|
|
|
|
return getEmptyIdText();
|
|
|
|
|
|
|
|
case GXS_ID_DETAILS_TYPE_LOADING:
|
|
|
|
return getLoadingText(details.mId);
|
|
|
|
|
|
|
|
case GXS_ID_DETAILS_TYPE_DONE:
|
|
|
|
return getName(details);
|
|
|
|
|
2015-10-10 21:25:06 -04:00
|
|
|
case GXS_ID_DETAILS_TYPE_BANNED:
|
|
|
|
return tr("[Banned]") ;
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
case GXS_ID_DETAILS_TYPE_FAILED:
|
|
|
|
return getFailedText(details.mId);
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
QIcon GxsIdDetails::getLoadingIcon(const RsGxsId &/*id*/)
|
|
|
|
{
|
|
|
|
return QIcon(IMAGE_LOADING);
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
|
|
|
|
2018-11-23 17:17:12 -05:00
|
|
|
bool GxsIdDetails::MakeIdDesc(const RsGxsId &id, bool doIcons, QString &str, QList<QIcon> &icons, QString& comment,uint32_t icon_types)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
|
|
|
RsIdentityDetails details;
|
2014-11-26 19:55:48 -05:00
|
|
|
|
2014-02-02 06:25:11 -05:00
|
|
|
if (!rsIdentity->getIdDetails(id, details))
|
|
|
|
{
|
2014-11-26 19:55:48 -05:00
|
|
|
// std::cerr << "GxsIdTreeWidget::MakeIdDesc() FAILED TO GET ID " << id;
|
2014-09-16 15:53:23 -04:00
|
|
|
//std::cerr << std::endl;
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
str = getLoadingText(id);
|
2014-02-02 06:25:11 -05:00
|
|
|
|
|
|
|
if (!doIcons)
|
|
|
|
{
|
2014-11-26 19:55:48 -05:00
|
|
|
icons.push_back(getLoadingIcon(id));
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
str = getName(details);
|
|
|
|
|
|
|
|
comment += getComment(details);
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
if (doIcons)
|
2018-11-23 17:17:12 -05:00
|
|
|
getIcons(details, icons,icon_types);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
// Cyril: I disabled these three which I believe to have been put for testing purposes.
|
|
|
|
//
|
|
|
|
// icons.push_back(QIcon(IMAGE_ANON));
|
|
|
|
// icons.push_back(QIcon(IMAGE_ANON));
|
|
|
|
// icons.push_back(QIcon(IMAGE_ANON));
|
|
|
|
|
|
|
|
// std::cerr << "GxsIdTreeWidget::MakeIdDesc() ID Ok. Comment: " << comment.toStdString() ;
|
|
|
|
// std::cerr << std::endl;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString GxsIdDetails::getName(const RsIdentityDetails &details)
|
|
|
|
{
|
2019-02-16 09:42:22 -05:00
|
|
|
if( details.mReputation.mOverallReputationLevel ==
|
|
|
|
RsReputationLevel::LOCALLY_NEGATIVE )
|
|
|
|
return tr("[Banned]");
|
2015-10-10 21:25:06 -04:00
|
|
|
|
|
|
|
QString name = QString::fromUtf8(details.mNickname.c_str()).left(RSID_MAXIMUM_NICKNAME_SIZE);
|
2014-11-26 19:55:48 -05:00
|
|
|
|
|
|
|
std::list<RsRecognTag>::const_iterator it;
|
|
|
|
for (it = details.mRecognTags.begin(); it != details.mRecognTags.end(); ++it)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
2014-11-26 19:55:48 -05:00
|
|
|
name += QString(" (%1 %2)").arg(it->tag_class).arg(it->tag_type);
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString GxsIdDetails::getComment(const RsIdentityDetails &details)
|
|
|
|
{
|
|
|
|
QString comment;
|
2015-10-10 21:25:06 -04:00
|
|
|
QString nickname ;
|
|
|
|
|
2019-02-16 09:42:22 -05:00
|
|
|
bool banned = ( details.mReputation.mOverallReputationLevel ==
|
|
|
|
RsReputationLevel::LOCALLY_NEGATIVE );
|
2015-10-10 21:25:06 -04:00
|
|
|
|
|
|
|
if(details.mNickname.empty())
|
|
|
|
nickname = tr("[Unknown]") ;
|
|
|
|
else if(banned)
|
|
|
|
nickname = tr("[Banned]") ;
|
|
|
|
else
|
|
|
|
nickname = QString::fromUtf8(details.mNickname.c_str()).left(RSID_MAXIMUM_NICKNAME_SIZE) ;
|
|
|
|
|
|
|
|
|
2015-02-04 18:05:56 -05:00
|
|
|
comment = QString("%1:%2<br/>%3:%4").arg(QApplication::translate("GxsIdDetails", "Identity name"),
|
2015-03-07 09:27:32 -05:00
|
|
|
nickname,
|
2015-02-04 18:05:56 -05:00
|
|
|
QApplication::translate("GxsIdDetails", "Identity Id"),
|
2014-11-26 19:55:48 -05:00
|
|
|
QString::fromStdString(details.mId.toStdString()));
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2015-11-19 21:14:32 -05:00
|
|
|
if (details.mFlags & RS_IDENTITY_FLAGS_PGP_LINKED)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
2018-11-26 17:07:05 -05:00
|
|
|
comment += QString("<br/>%1: ").arg(QApplication::translate("GxsIdDetails", "Node"));
|
2014-09-13 10:07:59 -04:00
|
|
|
|
2015-11-19 21:14:32 -05:00
|
|
|
if (details.mFlags & RS_IDENTITY_FLAGS_PGP_KNOWN)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
|
|
|
/* look up real name */
|
2014-03-17 16:56:06 -04:00
|
|
|
std::string authorName = rsPeers->getGPGName(details.mPgpId);
|
2015-10-10 21:25:06 -04:00
|
|
|
|
|
|
|
comment += QString("%1 [%2]").arg(QString::fromUtf8(authorName.c_str()), QString::fromStdString(details.mPgpId.toStdString()));
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
2014-09-25 16:14:21 -04:00
|
|
|
else
|
2014-11-26 19:55:48 -05:00
|
|
|
comment += QApplication::translate("GxsIdDetails", "unknown Key");
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
2018-11-26 17:07:05 -05:00
|
|
|
//else
|
|
|
|
// comment += QString("<br/>%1: %2").arg(QApplication::translate("GxsIdDetails", "Node:"), QApplication::translate("GxsIdDetails", "anonymous"));
|
2017-01-07 15:29:24 -05:00
|
|
|
|
2017-01-11 14:44:27 -05:00
|
|
|
if(details.mReputation.mFriendsPositiveVotes || details.mReputation.mFriendsNegativeVotes)
|
|
|
|
{
|
|
|
|
comment += "<br/>Votes:";
|
|
|
|
if(details.mReputation.mFriendsPositiveVotes > 0) comment += " <b>+" + QString::number(details.mReputation.mFriendsPositiveVotes) + "</b>";
|
|
|
|
if(details.mReputation.mFriendsNegativeVotes > 0) comment += " <b>-" + QString::number(details.mReputation.mFriendsNegativeVotes) + "</b>";
|
|
|
|
}
|
2014-11-26 19:55:48 -05:00
|
|
|
return comment;
|
|
|
|
}
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2019-02-16 09:42:22 -05:00
|
|
|
QIcon GxsIdDetails::getReputationIcon(
|
|
|
|
RsReputationLevel icon_index, uint32_t min_reputation )
|
2016-12-24 08:13:02 -05:00
|
|
|
{
|
2019-02-16 09:42:22 -05:00
|
|
|
if( static_cast<uint32_t>(icon_index) >= min_reputation )
|
|
|
|
return QIcon(REPUTATION_VOID);
|
2016-12-24 09:04:08 -05:00
|
|
|
|
2016-12-24 08:13:02 -05:00
|
|
|
switch(icon_index)
|
|
|
|
{
|
2019-02-16 09:42:22 -05:00
|
|
|
case RsReputationLevel::LOCALLY_NEGATIVE:
|
|
|
|
return QIcon(REPUTATION_LOCALLY_NEGATIVE_ICON);
|
|
|
|
case RsReputationLevel::LOCALLY_POSITIVE:
|
|
|
|
return QIcon(REPUTATION_LOCALLY_POSITIVE_ICON);
|
|
|
|
case RsReputationLevel::REMOTELY_POSITIVE:
|
|
|
|
return QIcon(REPUTATION_REMOTELY_POSITIVE_ICON);
|
|
|
|
case RsReputationLevel::REMOTELY_NEGATIVE:
|
|
|
|
return QIcon(REPUTATION_REMOTELY_NEGATIVE_ICON);
|
|
|
|
case RsReputationLevel::NEUTRAL:
|
|
|
|
return QIcon(REPUTATION_NEUTRAL_ICON);
|
|
|
|
default:
|
|
|
|
std::cerr << "Asked for unidentified icon index "
|
|
|
|
<< static_cast<uint32_t>(icon_index) << std::endl;
|
2016-12-24 08:13:02 -05:00
|
|
|
return QIcon(); // dont draw anything
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-24 09:04:08 -05:00
|
|
|
void GxsIdDetails::getIcons(const RsIdentityDetails &details, QList<QIcon> &icons,uint32_t icon_types,uint32_t minimal_required_reputation)
|
2014-11-26 19:55:48 -05:00
|
|
|
{
|
2015-01-25 17:09:12 -05:00
|
|
|
QPixmap pix ;
|
|
|
|
|
2019-02-16 09:42:22 -05:00
|
|
|
if( details.mReputation.mOverallReputationLevel ==
|
|
|
|
RsReputationLevel::LOCALLY_NEGATIVE )
|
2015-10-10 21:25:06 -04:00
|
|
|
{
|
|
|
|
icons.clear() ;
|
|
|
|
icons.push_back(QIcon(IMAGE_BANNED)) ;
|
|
|
|
return ;
|
|
|
|
}
|
2016-12-24 08:13:02 -05:00
|
|
|
|
|
|
|
if(icon_types & ICON_TYPE_REPUTATION)
|
2016-12-24 09:04:08 -05:00
|
|
|
icons.push_back(getReputationIcon(details.mReputation.mOverallReputationLevel,minimal_required_reputation)) ;
|
2016-12-24 08:13:02 -05:00
|
|
|
|
2015-03-14 15:19:34 -04:00
|
|
|
if(icon_types & ICON_TYPE_AVATAR)
|
|
|
|
{
|
2019-06-03 17:52:29 -04:00
|
|
|
if(details.mAvatar.mSize == 0 || !GxsIdDetails::loadPixmapFromData(details.mAvatar.mData, details.mAvatar.mSize, pix))
|
2015-02-13 14:34:38 -05:00
|
|
|
#if QT_VERSION < 0x040700
|
2019-06-03 17:52:29 -04:00
|
|
|
pix = makeDefaultIcon(details.mId);
|
2015-02-13 14:34:38 -05:00
|
|
|
#else
|
2019-06-03 17:52:29 -04:00
|
|
|
pix = makeDefaultIcon(details.mId);
|
2015-02-13 14:34:38 -05:00
|
|
|
#endif
|
2015-01-25 17:09:12 -05:00
|
|
|
|
|
|
|
|
2015-03-14 15:19:34 -04:00
|
|
|
QIcon idIcon(pix);
|
|
|
|
//CreateIdIcon(id, idIcon);
|
|
|
|
icons.push_back(idIcon);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(icon_types & ICON_TYPE_PGP)
|
|
|
|
{
|
|
|
|
// ICON Logic.
|
|
|
|
QIcon baseIcon;
|
2015-11-19 21:14:32 -05:00
|
|
|
if (details.mFlags & RS_IDENTITY_FLAGS_PGP_LINKED)
|
2015-03-14 15:19:34 -04:00
|
|
|
{
|
2015-11-19 21:14:32 -05:00
|
|
|
if (details.mFlags & RS_IDENTITY_FLAGS_PGP_KNOWN)
|
2015-03-14 15:19:34 -04:00
|
|
|
baseIcon = QIcon(IMAGE_PGPKNOWN);
|
|
|
|
else
|
|
|
|
baseIcon = QIcon(IMAGE_PGPUNKNOWN);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
baseIcon = QIcon(IMAGE_ANON);
|
|
|
|
|
|
|
|
icons.push_back(baseIcon);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(icon_types & ICON_TYPE_RECOGN)
|
|
|
|
{
|
|
|
|
// Add In RecognTags Icons.
|
|
|
|
std::list<RsRecognTag>::const_iterator it;
|
|
|
|
for (it = details.mRecognTags.begin(); it != details.mRecognTags.end(); ++it)
|
|
|
|
{
|
|
|
|
QIcon tagIcon;
|
|
|
|
if (findTagIcon(it->tag_class, it->tag_type, tagIcon))
|
|
|
|
{
|
|
|
|
icons.push_back(tagIcon);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
|
|
|
|
2015-02-22 06:52:33 -05:00
|
|
|
void GxsIdDetails::GenerateCombinedPixmap(QPixmap &pixmap, const QList<QIcon> &icons, int iconSize)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
|
|
|
int count = icons.size();
|
2015-02-22 06:52:33 -05:00
|
|
|
if (count == 0) {
|
|
|
|
pixmap = QPixmap();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pixmap = QPixmap(iconSize * count, iconSize);
|
|
|
|
pixmap.fill(Qt::transparent);
|
|
|
|
QPainter painter(&pixmap);
|
2014-02-02 06:25:11 -05:00
|
|
|
|
2014-11-26 19:55:48 -05:00
|
|
|
QList<QIcon>::const_iterator it;
|
2014-02-02 06:25:11 -05:00
|
|
|
int i = 0;
|
2014-10-21 18:33:02 -04:00
|
|
|
for(it = icons.begin(); it != icons.end(); ++it, ++i)
|
2014-02-02 06:25:11 -05:00
|
|
|
{
|
2015-02-22 06:52:33 -05:00
|
|
|
it->paint(&painter, iconSize * i, 0, iconSize, iconSize);
|
2014-02-02 06:25:11 -05:00
|
|
|
}
|
|
|
|
}
|