/**************************************************************** * 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 #include #include #include "rshare.h" #include "GxsChannelPostItem.h" #include "FeedHolder.h" #include "SubFileItem.h" #include "gui/notifyqt.h" #include "util/misc.h" #include "gui/RetroShareLink.h" #include "util/HandleRichText.h" #include "util/DateTime.h" #include /**** * #define DEBUG_ITEM 1 ****/ #define COLOR_NORMAL QColor(248, 248, 248) #define COLOR_NEW QColor(220, 236, 253) #define SELF_LOAD 1 #define DATA_PROVIDED 2 /** Constructor */ GxsChannelPostItem::GxsChannelPostItem(FeedHolder *parent, uint32_t feedId, const std::string &chanId, const std::string &msgId, bool isHome) :QWidget(NULL), mParent(parent), mFeedId(feedId), mChanId(chanId), mMsgId(msgId), mIsHome(isHome) { mMode = SELF_LOAD; mSubscribeFlags = 0; setup(); // THIS IS JUST SO WE REMEMBER TO DO IT. updateItemStatic(); updateItem(); } /** Constructor */ GxsChannelPostItem::GxsChannelPostItem(FeedHolder *parent, uint32_t feedId, const RsGxsChannelPost &post, uint32_t subFlags, bool isHome) :QWidget(NULL), mParent(parent), mFeedId(feedId), mChanId(post.mMeta.mGroupId), mMsgId(post.mMeta.mMsgId), mIsHome(isHome) { mMode = DATA_PROVIDED; mSubscribeFlags = subFlags; setup(); loadPost(post); updateItem(); } void GxsChannelPostItem::setup() { /* Invoke the Qt Designer generated object setup routine */ setupUi(this); setAttribute ( Qt::WA_DeleteOnClose, true ); m_inUpdateItemStatic = false; /* general ones */ connect( expandButton, SIGNAL( clicked( void ) ), this, SLOT( toggle ( void ) ) ); connect( clearButton, SIGNAL( clicked( void ) ), this, SLOT( removeItem ( void ) ) ); /* specific */ connect(readAndClearButton, SIGNAL(clicked()), this, SLOT(readAndClearItem())); connect( unsubscribeButton, SIGNAL( clicked( void ) ), this, SLOT( unsubscribeChannel ( void ) ) ); connect( downloadButton, SIGNAL( clicked( void ) ), this, SLOT( download ( void ) ) ); // HACK FOR NOW. connect( commentButton, SIGNAL( clicked( void ) ), this, SLOT( comments ( void ) ) ); connect( playButton, SIGNAL( clicked( void ) ), this, SLOT( play ( void ) ) ); connect( copyLinkButton, SIGNAL( clicked( void ) ), this, SLOT( copyLink ( void ) ) ); connect( readButton, SIGNAL( toggled(bool) ), this, SLOT( readToggled(bool) ) ); connect( NotifyQt::getInstance(), SIGNAL(channelMsgReadSatusChanged(QString,QString,int)), this, SLOT(channelMsgReadSatusChanged(QString,QString,int)), Qt::QueuedConnection); downloadButton->hide(); playButton->hide(); warn_image_label->hide(); warning_label->hide(); titleLabel->setMinimumWidth(100); subjectLabel->setMinimumWidth(100); warning_label->setMinimumWidth(100); frame->setProperty("state", ""); QPalette palette = frame->palette(); palette.setColor(frame->backgroundRole(), COLOR_NORMAL); frame->setPalette(palette); expandFrame->hide(); } void GxsChannelPostItem::updateItemStatic() { #if 0 /* fill in */ #ifdef DEBUG_ITEM std::cerr << "GxsChannelPostItem::updateItemStatic()"; std::cerr << std::endl; #endif ChannelMsgInfo cmi; if (!rsChannels) return; if (!rsChannels->getChannelMessage(mChanId, mMsgId, cmi)) return; m_inUpdateItemStatic = true; QString title; ChannelInfo ci; rsChannels->getChannelInfo(mChanId, ci); if (!mIsHome) { title = tr("Channel Feed") + ": "; RetroShareLink link; link.createChannel(ci.channelId, ""); title += link.toHtml(); titleLabel->setText(title); RetroShareLink msgLink; msgLink.createChannel(cmi.channelId, cmi.msgId); subjectLabel->setText(msgLink.toHtml()); if ((ci.channelFlags & RS_DISTRIB_SUBSCRIBED) || (ci.channelFlags & RS_DISTRIB_ADMIN)) { unsubscribeButton->setEnabled(true); } else { unsubscribeButton->setEnabled(false); } readButton->hide(); newLabel->hide(); copyLinkButton->hide(); } else { /* subject */ titleLabel->setText(QString::fromStdWString(cmi.subject)); subjectLabel->setText(RsHtml().formatText(NULL, QString::fromStdWString(cmi.msg), RSHTML_FORMATTEXT_EMBED_SMILEYS | RSHTML_FORMATTEXT_EMBED_LINKS)); /* disable buttons: deletion facility not enabled with cache services yet */ clearButton->setEnabled(false); unsubscribeButton->setEnabled(false); clearButton->hide(); readAndClearButton->hide(); unsubscribeButton->hide(); copyLinkButton->show(); if ((ci.channelFlags & RS_DISTRIB_SUBSCRIBED) || (ci.channelFlags & RS_DISTRIB_ADMIN)) { readButton->setVisible(true); uint32_t status = 0; rsChannels->getMessageStatus(mChanId, mMsgId, status); if ((status & CHANNEL_MSG_STATUS_READ) == 0 || (status & CHANNEL_MSG_STATUS_UNREAD_BY_USER)) { readButton->setChecked(true); readButton->setIcon(QIcon(":/images/message-state-unread.png")); } else { readButton->setChecked(false); readButton->setIcon(QIcon(":/images/message-state-read.png")); } bool newState; QColor color; if (status & CHANNEL_MSG_STATUS_READ) { newLabel->setVisible(false); newState = false; color = COLOR_NORMAL; } else { newLabel->setVisible(true); newState = true; color = COLOR_NEW; } /* unpolish widget to clear the stylesheet's palette cache */ frame->style()->unpolish(frame); QPalette palette = frame->palette(); palette.setColor(frame->backgroundRole(), color); frame->setPalette(palette); frame->setProperty("new", newState); Rshare::refreshStyleSheet(frame, false); } else { readButton->setVisible(false); newLabel->setVisible(false); } } msgLabel->setText(RsHtml().formatText(NULL, QString::fromStdWString(cmi.msg), RSHTML_FORMATTEXT_EMBED_SMILEYS | RSHTML_FORMATTEXT_EMBED_LINKS)); msgFrame->setVisible(!cmi.msg.empty()); datetimelabel->setText(DateTime::formatLongDateTime(cmi.ts)); filelabel->setText(QString("(%1 %2) %3").arg(cmi.count).arg(tr("Files")).arg(misc::friendlyUnit(cmi.size))); if (mFileItems.empty() == false) { std::list::iterator it; for(it = mFileItems.begin(); it != mFileItems.end(); it++) { delete(*it); } mFileItems.clear(); } std::list::iterator it; for(it = cmi.files.begin(); it != cmi.files.end(); it++) { /* add file */ SubFileItem *fi = new SubFileItem(it->hash, it->fname, it->path, it->size, SFI_STATE_REMOTE | SFI_TYPE_CHANNEL, ""); mFileItems.push_back(fi); /* check if the file is a media file */ if (!misc::isPreviewable(QFileInfo(QString::fromUtf8(it->fname.c_str())).suffix())) fi->mediatype(); QLayout *layout = expandFrame->layout(); layout->addWidget(fi); } if(cmi.thumbnail.image_thumbnail != NULL) { QPixmap thumbnail; thumbnail.loadFromData(cmi.thumbnail.image_thumbnail, cmi.thumbnail.im_thumbnail_size, "PNG"); logoLabel->setPixmap(thumbnail); } m_inUpdateItemStatic = false; #endif } void GxsChannelPostItem::loadPost(const RsGxsChannelPost &post) { /* fill in */ #ifdef DEBUG_ITEM std::cerr << "GxsChannelPostItem::loadPost()"; std::cerr << std::endl; #endif QString title; if (!mIsHome) { title = tr("Channel Feed") + ": "; RetroShareLink link; link.createChannel(post.mMeta.mGroupId, ""); title += link.toHtml(); titleLabel->setText(title); RetroShareLink msgLink; msgLink.createChannel(post.mMeta.mGroupId, post.mMeta.mMsgId); subjectLabel->setText(msgLink.toHtml()); if (IS_GROUP_SUBSCRIBED(mSubscribeFlags) || IS_GROUP_ADMIN(mSubscribeFlags)) { unsubscribeButton->setEnabled(true); } else { unsubscribeButton->setEnabled(false); } readButton->hide(); //newLabel->hide(); copyLinkButton->hide(); } else { /* subject */ titleLabel->setText(QString::fromUtf8(post.mMeta.mMsgName.c_str())); subjectLabel->setText(RsHtml().formatText(NULL, QString::fromUtf8(post.mMsg.c_str()), RSHTML_FORMATTEXT_EMBED_SMILEYS | RSHTML_FORMATTEXT_EMBED_LINKS)); /* disable buttons: deletion facility not enabled with cache services yet */ clearButton->setEnabled(false); unsubscribeButton->setEnabled(false); clearButton->hide(); readAndClearButton->hide(); unsubscribeButton->hide(); copyLinkButton->show(); if (IS_GROUP_SUBSCRIBED(mSubscribeFlags) || IS_GROUP_ADMIN(mSubscribeFlags)) { readButton->setVisible(true); uint32_t status = 0; rsChannels->getMessageStatus(mChanId, mMsgId, status); if (IS_MSG_UNREAD(post.mMeta.mMsgStatus) || IS_MSG_NEW(post.mMeta.mMsgStatus)) { readButton->setChecked(true); readButton->setIcon(QIcon(":/images/message-state-unread.png")); } else { readButton->setChecked(false); readButton->setIcon(QIcon(":/images/message-state-read.png")); } bool newState; QColor color; if (!IS_MSG_UNREAD(post.mMeta.mMsgStatus)) { //newLabel->setVisible(false); newState = false; color = COLOR_NORMAL; } else { //newLabel->setVisible(true); newState = true; color = COLOR_NEW; } /* unpolish widget to clear the stylesheet's palette cache */ frame->style()->unpolish(frame); QPalette palette = frame->palette(); palette.setColor(frame->backgroundRole(), color); frame->setPalette(palette); frame->setProperty("new", newState); Rshare::refreshStyleSheet(frame, false); } else { readButton->setVisible(false); //newLabel->setVisible(false); } } msgLabel->setText(RsHtml().formatText(NULL, QString::fromUtf8(post.mMsg.c_str()), RSHTML_FORMATTEXT_EMBED_SMILEYS | RSHTML_FORMATTEXT_EMBED_LINKS)); msgFrame->setVisible(!post.mMsg.empty()); datetimelabel->setText(DateTime::formatLongDateTime(post.mMeta.mPublishTs)); filelabel->setText(QString("(%1 %2) %3").arg(post.mCount).arg(tr("Files")).arg(misc::friendlyUnit(post.mSize))); if (mFileItems.empty() == false) { std::list::iterator it; for(it = mFileItems.begin(); it != mFileItems.end(); it++) { delete(*it); } mFileItems.clear(); } std::list::const_iterator it; for(it = post.mFiles.begin(); it != post.mFiles.end(); it++) { /* add file */ std::string path; SubFileItem *fi = new SubFileItem(it->mHash, it->mName, path, it->mSize, SFI_STATE_REMOTE | SFI_TYPE_CHANNEL, ""); mFileItems.push_back(fi); /* check if the file is a media file */ if (!misc::isPreviewable(QFileInfo(QString::fromUtf8(it->mName.c_str())).suffix())) fi->mediatype(); QLayout *layout = expandFrame->layout(); layout->addWidget(fi); } if(post.mThumbnail.mData != NULL) { QPixmap thumbnail; thumbnail.loadFromData(post.mThumbnail.mData, post.mThumbnail.mSize, "PNG"); logoLabel->setPixmap(thumbnail); } } void GxsChannelPostItem::setFileCleanUpWarning(uint32_t time_left) { int hours = (int)time_left/3600; int minutes = (time_left - hours*3600)%60; warning_label->setText(tr("Warning! You have less than %1 hours and %2 minute before this file is deleted Consider saving it.").arg( QString::number(hours)).arg(QString::number(minutes))); QFont warnFont = warning_label->font(); warnFont.setBold(true); warning_label->setFont(warnFont); warn_image_label->setVisible(true); warning_label->setVisible(true); return; } void GxsChannelPostItem::updateItem() { /* fill in */ #ifdef DEBUG_ITEM std::cerr << "GxsChannelPostItem::updateItem()"; std::cerr << std::endl; #endif int msec_rate = 10000; int downloadCount = 0; int downloadStartable = 0; int playCount = 0; int playStartable = 0; bool startable; bool loopAgain = false; /* Very slow Tick to check when all files are downloaded */ std::list::iterator it; for(it = mFileItems.begin(); it != mFileItems.end(); it++) { SubFileItem *item = *it; if (item->isDownloadable(startable)) { downloadCount++; if (startable) { downloadStartable++; } } if (item->isPlayable(startable)) { playCount++; if (startable) { playStartable++; } } if (!item->done()) { /* loop again */ loopAgain = true; } } if (downloadCount) { downloadButton->show(); if (downloadStartable) { downloadButton->setEnabled(true); } else { downloadButton->setEnabled(false); } } else { downloadButton->hide(); } if (playCount) { /* one file is playable */ playButton->show(); if (playStartable == 1) { playButton->setEnabled(true); } else { playButton->setEnabled(false); } } else { playButton->hide(); } if (loopAgain) { QTimer::singleShot( msec_rate, this, SLOT(updateItem( void ) )); } // HACK TO DISPLAY COMMENT BUTTON FOR NOW. downloadButton->show(); downloadButton->setEnabled(true); } void GxsChannelPostItem::toggle() { mParent->lockLayout(this, true); if (expandFrame->isHidden()) { expandFrame->show(); expandButton->setIcon(QIcon(QString(":/images/edit_remove24.png"))); expandButton->setToolTip(tr("Hide")); readToggled(false); } else { expandFrame->hide(); expandButton->setIcon(QIcon(QString(":/images/edit_add24.png"))); expandButton->setToolTip(tr("Expand")); } mParent->lockLayout(this, false); } void GxsChannelPostItem::removeItem() { #ifdef DEBUG_ITEM std::cerr << "GxsChannelPostItem::removeItem()"; std::cerr << std::endl; #endif mParent->lockLayout(this, true); hide(); mParent->lockLayout(this, false); if (mParent) { mParent->deleteFeedItem(this, mFeedId); } } void GxsChannelPostItem::comments() { #ifdef DEBUG_ITEM std::cerr << "GxsChannelPostItem::comments()"; std::cerr << std::endl; #endif if (mParent) { mParent->openComments(mFeedId, mChanId, mMsgId); } } /*********** SPECIFIC FUNCTIONS ***********************/ void GxsChannelPostItem::readAndClearItem() { #ifdef DEBUG_ITEM std::cerr << "GxsChannelPostItem::readAndClearItem()"; std::cerr << std::endl; #endif readToggled(false); removeItem(); } void GxsChannelPostItem::unsubscribeChannel() { #ifdef DEBUG_ITEM std::cerr << "GxsChannelPostItem::unsubscribeChannel()"; std::cerr << std::endl; #endif if (rsChannels) { rsChannels->channelSubscribe(mChanId, false, false); } updateItemStatic(); } void GxsChannelPostItem::download() { std::list::iterator it; for(it = mFileItems.begin(); it != mFileItems.end(); it++) { (*it)->download(); } updateItem(); } void GxsChannelPostItem::play() { std::list::iterator it; for(it = mFileItems.begin(); it != mFileItems.end(); it++) { bool startable; if ((*it)->isPlayable(startable) && startable) { (*it)->play(); } } } void GxsChannelPostItem::readToggled(bool checked) { if (m_inUpdateItemStatic) { return; } /* set always as read ... */ uint32_t statusNew = CHANNEL_MSG_STATUS_READ; if (checked) { /* ... and as unread by user */ statusNew |= CHANNEL_MSG_STATUS_UNREAD_BY_USER; } else { /* ... and as read by user */ statusNew &= ~CHANNEL_MSG_STATUS_UNREAD_BY_USER; } if (!mIsHome) { disconnect( NotifyQt::getInstance(), SIGNAL(channelMsgReadSatusChanged(QString,QString,int)), this, SLOT(channelMsgReadSatusChanged(QString,QString,int))); } rsChannels->setMessageStatus(mChanId, mMsgId, statusNew, CHANNEL_MSG_STATUS_READ | CHANNEL_MSG_STATUS_UNREAD_BY_USER); if (!mIsHome) { connect( NotifyQt::getInstance(), SIGNAL(channelMsgReadSatusChanged(QString,QString,int)), this, SLOT(channelMsgReadSatusChanged(QString,QString,int)), Qt::QueuedConnection); } } void GxsChannelPostItem::channelMsgReadSatusChanged(const QString& channelId, const QString& msgId, int status) { if (channelId.toStdString() == mChanId && msgId.toStdString() == mMsgId) { if (!mIsHome) { if (status & CHANNEL_MSG_STATUS_READ) { close(); return; } } updateItemStatic(); } } void GxsChannelPostItem::copyLink() { if (mChanId.empty() || mMsgId.empty()) { return; } ChannelMsgInfo cmi; if (rsChannels->getChannelMessage(mChanId, mMsgId, cmi)) { RetroShareLink link; if (link.createChannel(cmi.channelId, cmi.msgId)) { QList urls; urls.push_back(link); RSLinkClipboard::copyLinks(urls); } } }