mirror of
https://github.com/RetroShare/RetroShare.git
synced 2025-01-21 12:51:09 -05:00
173dbfc18d
git-svn-id: http://svn.code.sf.net/p/retroshare/code/trunk@3791 b45a01b8-16f6-495d-af2f-9b41ad6348cc
2301 lines
70 KiB
C++
2301 lines
70 KiB
C++
/****************************************************************
|
|
* RetroShare is distributed under the following license:
|
|
*
|
|
* Copyright (C) 2006, crypton
|
|
*
|
|
* 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 <QItemDelegate>
|
|
#include <QPainter>
|
|
#include <QPaintEvent>
|
|
#include <QStandardItemModel>
|
|
#include <QShortcut>
|
|
#include <QTimer>
|
|
#include <QDateTime>
|
|
#include <QPrinter>
|
|
#include <QPrintDialog>
|
|
#include <QTextStream>
|
|
#include <QTextCodec>
|
|
|
|
#include "MessagesDialog.h"
|
|
#include "msgs/MessageComposer.h"
|
|
#include "gui/RetroShareLink.h"
|
|
#include "util/printpreview.h"
|
|
#include "settings/rsharesettings.h"
|
|
#include "util/misc.h"
|
|
#include "common/TagDefs.h"
|
|
#include "common/PeerDefs.h"
|
|
#include "common/RSItemDelegate.h"
|
|
#include "common/Emoticons.h"
|
|
|
|
#include <retroshare/rsinit.h>
|
|
#include <retroshare/rspeers.h>
|
|
#include <retroshare/rsfiles.h>
|
|
|
|
#include <algorithm>
|
|
|
|
/* Images for context menu icons */
|
|
#define IMAGE_MESSAGE ":/images/folder-draft.png"
|
|
#define IMAGE_MESSAGEREPLY ":/images/mail_reply.png"
|
|
#define IMAGE_MESSAGEREPLYALL ":/images/mail_replyall.png"
|
|
#define IMAGE_MESSAGEFORWARD ":/images/mail_forward.png"
|
|
#define IMAGE_MESSAGEREMOVE ":/images/message-mail-imapdelete.png"
|
|
#define IMAGE_DOWNLOAD ":/images/start.png"
|
|
#define IMAGE_DOWNLOADALL ":/images/startall.png"
|
|
|
|
#define COLUMN_COUNT 7
|
|
#define COLUMN_ATTACHEMENTS 0
|
|
#define COLUMN_SUBJECT 1
|
|
#define COLUMN_UNREAD 2
|
|
#define COLUMN_FROM 3
|
|
#define COLUMN_DATE 4
|
|
#define COLUMN_CONTENT 5
|
|
#define COLUMN_TAGS 6
|
|
|
|
#define COLUMN_DATA 0 // column for storing the userdata like msgid and srcid
|
|
|
|
#define ROLE_SORT Qt::UserRole
|
|
#define ROLE_MSGID Qt::UserRole + 1
|
|
#define ROLE_SRCID Qt::UserRole + 2
|
|
#define ROLE_UNREAD Qt::UserRole + 3
|
|
#define ROLE_MSGFLAGS Qt::UserRole + 4
|
|
|
|
#define COLUMN_FILE_NAME 0
|
|
#define COLUMN_FILE_SIZE 1
|
|
#define COLUMN_FILE_HASH 2
|
|
#define COLUMN_FILE_COUNT 3
|
|
|
|
#define ROW_INBOX 0
|
|
#define ROW_OUTBOX 1
|
|
#define ROW_DRAFTBOX 2
|
|
#define ROW_SENTBOX 3
|
|
#define ROW_TRASHBOX 4
|
|
|
|
#define ACTION_TAGSINDEX_SIZE 3
|
|
#define ACTION_TAGSINDEX_TYPE "Type"
|
|
#define ACTION_TAGSINDEX_ID "ID"
|
|
#define ACTION_TAGSINDEX_COLOR "Color"
|
|
|
|
#define ACTION_TAGS_REMOVEALL 0
|
|
#define ACTION_TAGS_TAG 1
|
|
#define ACTION_TAGS_NEWTAG 2
|
|
|
|
class MessagesMenu : public QMenu
|
|
{
|
|
public:
|
|
MessagesMenu(const QString &title, QWidget *parent) : QMenu (title, parent)
|
|
{
|
|
}
|
|
|
|
protected:
|
|
virtual void paintEvent(QPaintEvent *e)
|
|
{
|
|
QMenu::paintEvent(e);
|
|
|
|
QPainter p(this);
|
|
QRegion emptyArea = QRegion(rect());
|
|
|
|
//draw the items with color
|
|
foreach (QAction *pAction, actions()) {
|
|
QRect adjustedActionRect = actionGeometry(pAction);
|
|
if (!e->rect().intersects(adjustedActionRect))
|
|
continue;
|
|
|
|
const QMap<QString, QVariant> &Values = pAction->data().toMap();
|
|
if (Values.size () != ACTION_TAGSINDEX_SIZE) {
|
|
continue;
|
|
}
|
|
if (Values [ACTION_TAGSINDEX_TYPE] != ACTION_TAGS_TAG) {
|
|
continue;
|
|
}
|
|
|
|
//set the clip region to be extra safe (and adjust for the scrollers)
|
|
QRegion adjustedActionReg(adjustedActionRect);
|
|
emptyArea -= adjustedActionReg;
|
|
p.setClipRegion(adjustedActionReg);
|
|
|
|
QStyleOptionMenuItem opt;
|
|
initStyleOption(&opt, pAction);
|
|
|
|
opt.palette.setColor(QPalette::ButtonText, QColor(Values [ACTION_TAGSINDEX_COLOR].toInt()));
|
|
// needed for Cleanlooks
|
|
opt.palette.setColor(QPalette::Text, QColor(Values [ACTION_TAGSINDEX_COLOR].toInt()));
|
|
|
|
opt.rect = adjustedActionRect;
|
|
style()->drawControl(QStyle::CE_MenuItem, &opt, &p, this);
|
|
}
|
|
}
|
|
};
|
|
|
|
MessagesDialog::LockUpdate::LockUpdate (MessagesDialog *pDialog, bool bUpdate)
|
|
{
|
|
m_pDialog = pDialog;
|
|
m_bUpdate = bUpdate;
|
|
|
|
m_pDialog->m_nLockUpdate++;
|
|
}
|
|
|
|
MessagesDialog::LockUpdate::~LockUpdate ()
|
|
{
|
|
if(--m_pDialog->m_nLockUpdate < 0)
|
|
m_pDialog->m_nLockUpdate = 0;
|
|
|
|
if (m_bUpdate && m_pDialog->m_nLockUpdate == 0) {
|
|
m_pDialog->insertMessages();
|
|
}
|
|
}
|
|
|
|
void MessagesDialog::LockUpdate::setUpdate(bool bUpdate)
|
|
{
|
|
m_bUpdate = bUpdate;
|
|
}
|
|
|
|
static int FilterColumnFromComboBox(int nIndex)
|
|
{
|
|
switch (nIndex) {
|
|
case 0:
|
|
return COLUMN_ATTACHEMENTS;
|
|
case 1:
|
|
return COLUMN_SUBJECT;
|
|
case 2:
|
|
return COLUMN_FROM;
|
|
case 3:
|
|
return COLUMN_DATE;
|
|
case 4:
|
|
return COLUMN_CONTENT;
|
|
case 5:
|
|
return COLUMN_TAGS;
|
|
}
|
|
|
|
return COLUMN_SUBJECT;
|
|
}
|
|
|
|
static int FilterColumnToComboBox(int nIndex)
|
|
{
|
|
switch (nIndex) {
|
|
case COLUMN_ATTACHEMENTS:
|
|
return 0;
|
|
case COLUMN_SUBJECT:
|
|
return 1;
|
|
case COLUMN_FROM:
|
|
return 2;
|
|
case COLUMN_DATE:
|
|
return 3;
|
|
case COLUMN_CONTENT:
|
|
return 4;
|
|
case COLUMN_TAGS:
|
|
return 5;
|
|
}
|
|
|
|
return FilterColumnToComboBox(COLUMN_SUBJECT);
|
|
}
|
|
|
|
/** Constructor */
|
|
MessagesDialog::MessagesDialog(QWidget *parent)
|
|
: MainPage(parent)
|
|
{
|
|
/* Invoke the Qt Designer generated object setup routine */
|
|
ui.setupUi(this);
|
|
|
|
m_bProcessSettings = false;
|
|
m_bInChange = false;
|
|
m_nLockUpdate = 0;
|
|
|
|
connect( ui.messagestreeView, SIGNAL( customContextMenuRequested( QPoint ) ), this, SLOT( messageslistWidgetCostumPopupMenu( QPoint ) ) );
|
|
connect( ui.msgList, SIGNAL( customContextMenuRequested( QPoint ) ), this, SLOT( msgfilelistWidgetCostumPopupMenu( QPoint ) ) );
|
|
connect( ui.listWidget, SIGNAL( customContextMenuRequested( QPoint ) ), this, SLOT( folderlistWidgetCostumPopupMenu( QPoint ) ) );
|
|
connect( ui.messagestreeView, SIGNAL(clicked ( const QModelIndex &) ) , this, SLOT( clicked( const QModelIndex & ) ) );
|
|
connect( ui.messagestreeView, SIGNAL(doubleClicked ( const QModelIndex& ) ) , this, SLOT( doubleClicked( const QModelIndex & ) ) );
|
|
connect( ui.listWidget, SIGNAL( currentRowChanged ( int) ), this, SLOT( changeBox ( int) ) );
|
|
connect( ui.tagWidget, SIGNAL( currentRowChanged ( int) ), this, SLOT( changeTag ( int) ) );
|
|
|
|
connect(ui.newmessageButton, SIGNAL(clicked()), this, SLOT(newmessage()));
|
|
connect(ui.removemessageButton, SIGNAL(clicked()), this, SLOT(removemessage()));
|
|
connect(ui.replymessageButton, SIGNAL(clicked()), this, SLOT(replytomessage()));
|
|
connect(ui.replyallmessageButton, SIGNAL(clicked()), this, SLOT(replyallmessage()));
|
|
connect(ui.forwardmessageButton, SIGNAL(clicked()), this, SLOT(forwardmessage()));
|
|
|
|
connect(ui.actionPrint, SIGNAL(triggered()), this, SLOT(print()));
|
|
ui.actionPrint->setDisabled(true);
|
|
connect(ui.actionPrintPreview, SIGNAL(triggered()), this, SLOT(printpreview()));
|
|
ui.actionPrintPreview->setDisabled(true);
|
|
connect(ui.printbutton, SIGNAL(clicked()), this, SLOT(print()));
|
|
|
|
connect(ui.expandFilesButton, SIGNAL(clicked()), this, SLOT(togglefileview()));
|
|
connect(ui.downloadButton, SIGNAL(clicked()), this, SLOT(getcurrentrecommended()));
|
|
|
|
connect( ui.msgText, SIGNAL( anchorClicked(const QUrl &)), SLOT(anchorClicked(const QUrl &)));
|
|
|
|
connect(ui.actionTextBesideIcon, SIGNAL(triggered()), this, SLOT(buttonstextbesideicon()));
|
|
connect(ui.actionIconOnly, SIGNAL(triggered()), this, SLOT(buttonsicononly()));
|
|
connect(ui.actionTextUnderIcon, SIGNAL(triggered()), this, SLOT(buttonstextundericon()));
|
|
|
|
connect(ui.actionSave_as, SIGNAL(triggered()), this, SLOT(fileSaveAs()));
|
|
ui.actionSave_as->setDisabled(true);
|
|
|
|
connect( ui.clearButton, SIGNAL(clicked()), this, SLOT(clearFilter()));
|
|
connect( ui.filterPatternLineEdit, SIGNAL(textChanged(const QString &)), this, SLOT(filterRegExpChanged()));
|
|
|
|
connect(ui.filterColumnComboBox, SIGNAL(currentIndexChanged(int)), this, SLOT(filterColumnChanged()));
|
|
|
|
connect(ui.toText, SIGNAL(linkActivated(QString)), this, SLOT(linkActivated(QString)));
|
|
connect(ui.ccText, SIGNAL(linkActivated(QString)), this, SLOT(linkActivated(QString)));
|
|
connect(ui.bccText, SIGNAL(linkActivated(QString)), this, SLOT(linkActivated(QString)));
|
|
|
|
ui.msgText->setOpenLinks(false);
|
|
|
|
m_eListMode = LIST_NOTHING;
|
|
|
|
mCurrCertId = "";
|
|
mCurrMsgId = "";
|
|
|
|
// Set the QStandardItemModel
|
|
MessagesModel = new QStandardItemModel(0, COLUMN_COUNT);
|
|
MessagesModel->setHeaderData(COLUMN_ATTACHEMENTS, Qt::Horizontal, QIcon(":/images/attachment.png"), Qt::DecorationRole);
|
|
MessagesModel->setHeaderData(COLUMN_SUBJECT, Qt::Horizontal, tr("Subject"));
|
|
MessagesModel->setHeaderData(COLUMN_UNREAD, Qt::Horizontal, QIcon(":/images/message-state-header.png"), Qt::DecorationRole);
|
|
MessagesModel->setHeaderData(COLUMN_FROM, Qt::Horizontal, tr("From"));
|
|
MessagesModel->setHeaderData(COLUMN_DATE, Qt::Horizontal, tr("Date"));
|
|
MessagesModel->setHeaderData(COLUMN_TAGS, Qt::Horizontal, tr("Tags"));
|
|
MessagesModel->setHeaderData(COLUMN_CONTENT, Qt::Horizontal, tr("Content"));
|
|
|
|
proxyModel = new QSortFilterProxyModel(this);
|
|
proxyModel->setDynamicSortFilter(true);
|
|
proxyModel->setSourceModel(MessagesModel);
|
|
proxyModel->setSortRole(ROLE_SORT);
|
|
proxyModel->sort (COLUMN_DATE, Qt::DescendingOrder);
|
|
ui.messagestreeView->setModel(proxyModel);
|
|
ui.messagestreeView->setSelectionBehavior(QTreeView::SelectRows);
|
|
|
|
RSItemDelegate *itemDelegate = new RSItemDelegate(this);
|
|
itemDelegate->removeFocusRect(COLUMN_UNREAD);
|
|
itemDelegate->setSpacing(QSize(0, 2));
|
|
ui.messagestreeView->setItemDelegate(itemDelegate);
|
|
|
|
ui.messagestreeView->setRootIsDecorated(false);
|
|
ui.messagestreeView->setSortingEnabled(true);
|
|
ui.messagestreeView->sortByColumn(COLUMN_DATE, Qt::DescendingOrder);
|
|
|
|
// connect after setting model
|
|
connect( ui.messagestreeView->selectionModel(), SIGNAL(currentChanged ( QModelIndex, QModelIndex ) ) , this, SLOT( currentChanged( const QModelIndex & ) ) );
|
|
|
|
// workaround for Qt bug, should be solved in next Qt release 4.7.0
|
|
// http://bugreports.qt.nokia.com/browse/QTBUG-8270
|
|
QShortcut *Shortcut = new QShortcut(QKeySequence (Qt::Key_Delete), ui.messagestreeView, 0, 0, Qt::WidgetShortcut);
|
|
connect(Shortcut, SIGNAL(activated()), this, SLOT( removemessage ()));
|
|
Shortcut = new QShortcut(QKeySequence (Qt::SHIFT | Qt::Key_Delete), ui.messagestreeView, 0, 0, Qt::WidgetShortcut);
|
|
connect(Shortcut, SIGNAL(activated()), this, SLOT( removemessage ()));
|
|
|
|
/* hide the Tree +/- */
|
|
ui.msgList->setRootIsDecorated( false );
|
|
ui.msgList->setSelectionMode( QAbstractItemView::ExtendedSelection );
|
|
|
|
/* Set header initial section sizes */
|
|
QHeaderView * msgwheader = ui.messagestreeView->header () ;
|
|
msgwheader->resizeSection (COLUMN_ATTACHEMENTS, 24);
|
|
msgwheader->resizeSection (COLUMN_SUBJECT, 250);
|
|
msgwheader->resizeSection (COLUMN_UNREAD, 16);
|
|
msgwheader->resizeSection (COLUMN_FROM, 140);
|
|
msgwheader->resizeSection (COLUMN_DATE, 140);
|
|
|
|
/* Set header resize modes and initial section sizes */
|
|
QHeaderView * msglheader = ui.msgList->header () ;
|
|
msglheader->setResizeMode (COLUMN_FILE_NAME, QHeaderView::Interactive);
|
|
msglheader->setResizeMode (COLUMN_FILE_SIZE, QHeaderView::Interactive);
|
|
msglheader->setResizeMode (COLUMN_FILE_HASH, QHeaderView::Interactive);
|
|
|
|
msglheader->resizeSection (COLUMN_FILE_NAME, 200);
|
|
msglheader->resizeSection (COLUMN_FILE_SIZE, 100);
|
|
msglheader->resizeSection (COLUMN_FILE_HASH, 200);
|
|
|
|
ui.forwardmessageButton->setToolTip(tr("Forward selected Message"));
|
|
ui.replyallmessageButton->setToolTip(tr("Reply to All"));
|
|
|
|
QMenu * printmenu = new QMenu();
|
|
printmenu->addAction(ui.actionPrint);
|
|
printmenu->addAction(ui.actionPrintPreview);
|
|
ui.printbutton->setMenu(printmenu);
|
|
|
|
QMenu * viewmenu = new QMenu();
|
|
viewmenu->addAction(ui.actionTextBesideIcon);
|
|
viewmenu->addAction(ui.actionIconOnly);
|
|
//viewmenu->addAction(ui.actionTextUnderIcon);
|
|
ui.viewtoolButton->setMenu(viewmenu);
|
|
|
|
loadToolButtonsettings();
|
|
|
|
mFont = QFont("Arial", 10, QFont::Bold);
|
|
ui.subjectText->setFont(mFont);
|
|
|
|
ui.filterPatternLineEdit->setMinimumWidth(20);
|
|
|
|
//setting default filter by column as subject
|
|
proxyModel->setFilterKeyColumn(FilterColumnFromComboBox(ui.filterColumnComboBox->currentIndex()));
|
|
|
|
ui.clearButton->hide();
|
|
|
|
ui.bcclabel->setVisible(false);
|
|
ui.bccText->setVisible(false);
|
|
ui.cclabel->setVisible(false);
|
|
ui.ccText->setVisible(false);
|
|
|
|
// load settings
|
|
processSettings(true);
|
|
|
|
/* Set header sizes for the fixed columns and resize modes, must be set after processSettings */
|
|
msgwheader->setResizeMode (COLUMN_ATTACHEMENTS, QHeaderView::Fixed);
|
|
msgwheader->setResizeMode (COLUMN_DATE, QHeaderView::Interactive);
|
|
msgwheader->setResizeMode (COLUMN_UNREAD, QHeaderView::Fixed);
|
|
msgwheader->resizeSection (COLUMN_UNREAD, 24);
|
|
|
|
// fill folder list
|
|
updateMessageSummaryList();
|
|
ui.listWidget->setCurrentRow(ROW_INBOX);
|
|
|
|
// create tag menu
|
|
fillTags();
|
|
|
|
// create timer for navigation
|
|
timer = new QTimer(this);
|
|
timer->setInterval(300);
|
|
timer->setSingleShot(true);
|
|
connect(timer, SIGNAL(timeout()), this, SLOT(updateCurrentMessage()));
|
|
|
|
/* Hide platform specific features */
|
|
#ifdef Q_WS_WIN
|
|
|
|
#endif
|
|
}
|
|
|
|
MessagesDialog::~MessagesDialog()
|
|
{
|
|
// stop and delete timer
|
|
timer->stop();
|
|
delete(timer);
|
|
|
|
// save settings
|
|
processSettings(false);
|
|
}
|
|
|
|
void MessagesDialog::processSettings(bool bLoad)
|
|
{
|
|
m_bProcessSettings = true;
|
|
|
|
QHeaderView *msgwheader = ui.messagestreeView->header () ;
|
|
|
|
Settings->beginGroup(QString("MessageDialog"));
|
|
|
|
if (bLoad) {
|
|
// load settings
|
|
|
|
// expandFiles
|
|
bool bValue = Settings->value("expandFiles", true).toBool();
|
|
ui.expandFilesButton->setChecked(bValue);
|
|
ui.msgList->setVisible(bValue);
|
|
togglefileview_internal();
|
|
|
|
// filterColumn
|
|
int nValue = FilterColumnToComboBox(Settings->value("filterColumn", true).toInt());
|
|
ui.filterColumnComboBox->setCurrentIndex(nValue);
|
|
|
|
// state of message tree
|
|
msgwheader->restoreState(Settings->value("MessageTree").toByteArray());
|
|
|
|
// state of tag list
|
|
bValue = Settings->value("tagList", true).toBool();
|
|
ui.Tags_Button->setChecked(bValue);
|
|
|
|
// state of splitter
|
|
ui.msgSplitter->restoreState(Settings->value("Splitter").toByteArray());
|
|
ui.msgSplitter_2->restoreState(Settings->value("Splitter2").toByteArray());
|
|
ui.listSplitter->restoreState(Settings->value("Splitter3").toByteArray());
|
|
} else {
|
|
// save settings
|
|
|
|
// state of message tree
|
|
Settings->setValue("MessageTree", msgwheader->saveState());
|
|
|
|
// state of tag list
|
|
Settings->setValue("tagList", ui.Tags_Button->isChecked());
|
|
|
|
// state of splitter
|
|
Settings->setValue("Splitter", ui.msgSplitter->saveState());
|
|
Settings->setValue("Splitter2", ui.msgSplitter_2->saveState());
|
|
Settings->setValue("Splitter3", ui.listSplitter->saveState());
|
|
}
|
|
|
|
Settings->endGroup();
|
|
|
|
m_bProcessSettings = false;
|
|
}
|
|
|
|
void MessagesDialog::fillTags()
|
|
{
|
|
MsgTagType Tags;
|
|
rsMsgs->getMessageTagTypes(Tags);
|
|
std::map<uint32_t, std::pair<std::string, uint32_t> >::iterator Tag;
|
|
|
|
// create tag menu
|
|
QMenu *pMenu = new MessagesMenu (tr("Tags"), this);
|
|
connect(pMenu, SIGNAL(triggered (QAction*)), this, SLOT(tagTriggered(QAction*)));
|
|
connect(pMenu, SIGNAL(aboutToShow()), this, SLOT(tagAboutToShow()));
|
|
|
|
bool bUser = false;
|
|
|
|
QString text;
|
|
QAction *pAction;
|
|
QMap<QString, QVariant> Values;
|
|
|
|
if (Tags.types.size()) {
|
|
pAction = new QAction(tr("Remove All Tags"), pMenu);
|
|
Values [ACTION_TAGSINDEX_TYPE] = ACTION_TAGS_REMOVEALL;
|
|
Values [ACTION_TAGSINDEX_ID] = 0;
|
|
Values [ACTION_TAGSINDEX_COLOR] = 0;
|
|
pAction->setData (Values);
|
|
pMenu->addAction(pAction);
|
|
|
|
pMenu->addSeparator();
|
|
|
|
for (Tag = Tags.types.begin(); Tag != Tags.types.end(); Tag++) {
|
|
text = TagDefs::name(Tag->first, Tag->second.first);
|
|
|
|
pAction = new QAction(text, pMenu);
|
|
Values [ACTION_TAGSINDEX_TYPE] = ACTION_TAGS_TAG;
|
|
Values [ACTION_TAGSINDEX_ID] = Tag->first;
|
|
Values [ACTION_TAGSINDEX_COLOR] = QRgb(Tag->second.second);
|
|
pAction->setData (Values);
|
|
pAction->setCheckable(true);
|
|
|
|
if (Tag->first >= RS_MSGTAGTYPE_USER && bUser == false) {
|
|
bUser = true;
|
|
pMenu->addSeparator();
|
|
}
|
|
|
|
pMenu->addAction(pAction);
|
|
}
|
|
|
|
pMenu->addSeparator();
|
|
}
|
|
|
|
pAction = new QAction(tr("New tag ..."), pMenu);
|
|
Values [ACTION_TAGSINDEX_TYPE] = ACTION_TAGS_NEWTAG;
|
|
Values [ACTION_TAGSINDEX_ID] = 0;
|
|
Values [ACTION_TAGSINDEX_COLOR] = 0;
|
|
pAction->setData (Values);
|
|
pMenu->addAction(pAction);
|
|
|
|
ui.tagButton->setMenu(pMenu);
|
|
|
|
// fill tags
|
|
m_bInChange = true;
|
|
|
|
// save current selection
|
|
QListWidgetItem *pItem = ui.tagWidget->currentItem();
|
|
uint32_t nSelectecTagId = 0;
|
|
if (pItem) {
|
|
nSelectecTagId = pItem->data(Qt::UserRole).toInt();
|
|
}
|
|
|
|
QListWidgetItem *pItemToSelect = NULL;
|
|
|
|
ui.tagWidget->clear();
|
|
for (Tag = Tags.types.begin(); Tag != Tags.types.end(); Tag++) {
|
|
text = TagDefs::name(Tag->first, Tag->second.first);
|
|
|
|
pItem = new QListWidgetItem (text, ui.tagWidget);
|
|
pItem->setForeground(QBrush(QColor(Tag->second.second)));
|
|
pItem->setIcon(QIcon(":/images/foldermail.png"));
|
|
pItem->setData(Qt::UserRole, Tag->first);
|
|
pItem->setData(Qt::UserRole + 1, text); // for updateMessageSummaryList
|
|
|
|
if (Tag->first == nSelectecTagId) {
|
|
pItemToSelect = pItem;
|
|
}
|
|
}
|
|
|
|
if (pItemToSelect) {
|
|
ui.tagWidget->setCurrentItem(pItemToSelect);
|
|
}
|
|
|
|
m_bInChange = false;
|
|
|
|
updateMessageSummaryList();
|
|
}
|
|
|
|
// replaced by shortcut
|
|
//void MessagesDialog::keyPressEvent(QKeyEvent *e)
|
|
//{
|
|
// if(e->key() == Qt::Key_Delete)
|
|
// {
|
|
// removemessage() ;
|
|
// e->accept() ;
|
|
// }
|
|
// else
|
|
// MainPage::keyPressEvent(e) ;
|
|
//}
|
|
|
|
int MessagesDialog::getSelectedMsgCount (QList<int> *pRows, QList<int> *pRowsRead, QList<int> *pRowsUnread)
|
|
{
|
|
if (pRowsRead) pRowsRead->clear();
|
|
if (pRowsUnread) pRowsUnread->clear();
|
|
|
|
//To check if the selection has more than one row.
|
|
QList<QModelIndex> selectedIndexList = ui.messagestreeView->selectionModel() -> selectedIndexes ();
|
|
QList<int> rowList;
|
|
for(QList<QModelIndex>::iterator it = selectedIndexList.begin(); it != selectedIndexList.end(); it++)
|
|
{
|
|
int row = it->row();
|
|
if (rowList.contains(row) == false)
|
|
{
|
|
rowList.append(row);
|
|
|
|
if (pRows || pRowsRead || pRowsUnread) {
|
|
int mappedRow = proxyModel->mapToSource(*it).row();
|
|
|
|
if (pRows) pRows->append(mappedRow);
|
|
|
|
if (MessagesModel->item(mappedRow, COLUMN_DATA)->data(ROLE_UNREAD).toBool()) {
|
|
if (pRowsUnread) pRowsUnread->append(mappedRow);
|
|
} else {
|
|
if (pRowsRead) pRowsRead->append(mappedRow);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return rowList.size();
|
|
}
|
|
|
|
bool MessagesDialog::isMessageRead(int nRow)
|
|
{
|
|
QStandardItem *item;
|
|
item = MessagesModel->item(nRow,COLUMN_DATA);
|
|
return !item->data(ROLE_UNREAD).toBool();
|
|
}
|
|
|
|
void MessagesDialog::messageslistWidgetCostumPopupMenu( QPoint point )
|
|
{
|
|
QMenu contextMnu( this );
|
|
|
|
/** Defines the actions for the context menu */
|
|
|
|
QAction *replytomsgAct = new QAction(QIcon(IMAGE_MESSAGEREPLY), tr( "Reply to Message" ), &contextMnu );
|
|
connect( replytomsgAct , SIGNAL( triggered() ), this, SLOT( replytomessage() ) );
|
|
contextMnu.addAction( replytomsgAct);
|
|
|
|
QAction *replyallmsgAct = new QAction(QIcon(IMAGE_MESSAGEREPLYALL), tr( "Reply to All" ), &contextMnu );
|
|
connect( replyallmsgAct , SIGNAL( triggered() ), this, SLOT( replyallmessage() ) );
|
|
contextMnu.addAction( replyallmsgAct);
|
|
|
|
QAction *forwardmsgAct = new QAction(QIcon(IMAGE_MESSAGEFORWARD), tr( "Forward Message" ), &contextMnu );
|
|
connect( forwardmsgAct , SIGNAL( triggered() ), this, SLOT( forwardmessage() ) );
|
|
contextMnu.addAction( forwardmsgAct);
|
|
|
|
QList<int> RowsRead;
|
|
QList<int> RowsUnread;
|
|
int nCount = getSelectedMsgCount (NULL, &RowsRead, &RowsUnread);
|
|
|
|
QAction *editAct = new QAction(tr( "Edit..." ), &contextMnu );
|
|
connect(editAct, SIGNAL(triggered()), this, SLOT(editmessage()));
|
|
contextMnu.addAction(editAct);
|
|
|
|
if (nCount == 1) {
|
|
editAct->setEnabled(true);
|
|
} else {
|
|
editAct->setDisabled(true);
|
|
}
|
|
|
|
contextMnu.addSeparator();
|
|
|
|
QAction *markAsRead = new QAction(QIcon(":/images/message-mail-read.png"), tr( "Mark as read" ), &contextMnu);
|
|
connect(markAsRead , SIGNAL(triggered()), this, SLOT(markAsRead()));
|
|
contextMnu.addAction(markAsRead);
|
|
if (RowsUnread.size() == 0) {
|
|
markAsRead->setDisabled(true);
|
|
}
|
|
|
|
QAction *markAsUnread = new QAction(QIcon(":/images/message-mail.png"), tr( "Mark as unread" ), &contextMnu);
|
|
connect(markAsUnread , SIGNAL(triggered()), this, SLOT(markAsUnread()));
|
|
contextMnu.addAction(markAsUnread);
|
|
if (RowsRead.size() == 0) {
|
|
markAsUnread->setDisabled(true);
|
|
}
|
|
|
|
contextMnu.addSeparator();
|
|
|
|
// add tags
|
|
contextMnu.addMenu(ui.tagButton->menu());
|
|
contextMnu.addSeparator();
|
|
|
|
QAction *removemsgAct;
|
|
if (nCount > 1) {
|
|
removemsgAct = new QAction(QIcon(IMAGE_MESSAGEREMOVE), tr( "Remove Messages" ), &contextMnu );
|
|
} else {
|
|
removemsgAct = new QAction(QIcon(IMAGE_MESSAGEREMOVE), tr( "Remove Message" ), &contextMnu );
|
|
}
|
|
|
|
connect( removemsgAct , SIGNAL( triggered() ), this, SLOT( removemessage() ) );
|
|
contextMnu.addAction( removemsgAct);
|
|
|
|
int listrow = ui.listWidget -> currentRow();
|
|
if (listrow == ROW_TRASHBOX) {
|
|
QAction *undeleteAct = new QAction(tr( "Undelete" ), &contextMnu );
|
|
connect(undeleteAct, SIGNAL(triggered()), this, SLOT(undeletemessage()));
|
|
contextMnu.addAction(undeleteAct);
|
|
|
|
if (nCount) {
|
|
undeleteAct->setEnabled(true);
|
|
} else {
|
|
undeleteAct->setDisabled(true);
|
|
}
|
|
}
|
|
|
|
contextMnu.addAction( ui.actionSave_as);
|
|
contextMnu.addAction( ui.actionPrintPreview);
|
|
contextMnu.addAction( ui.actionPrint);
|
|
contextMnu.addSeparator();
|
|
|
|
QAction *newmsgAct = new QAction(QIcon(IMAGE_MESSAGE), tr( "New Message" ), &contextMnu );
|
|
connect( newmsgAct , SIGNAL( triggered() ), this, SLOT( newmessage() ) );
|
|
contextMnu.addAction( newmsgAct);
|
|
|
|
if (nCount != 1) {
|
|
replytomsgAct->setDisabled(true);
|
|
replyallmsgAct->setDisabled(true);
|
|
forwardmsgAct->setDisabled(true);
|
|
}
|
|
if (nCount == 0) {
|
|
removemsgAct->setDisabled(true);
|
|
}
|
|
|
|
contextMnu.exec(QCursor::pos());
|
|
}
|
|
|
|
|
|
void MessagesDialog::msgfilelistWidgetCostumPopupMenu( QPoint point )
|
|
{
|
|
QMenu contextMnu( this );
|
|
|
|
QAction* getRecAct = new QAction(QIcon(IMAGE_DOWNLOAD), tr( "Download" ), &contextMnu );
|
|
connect( getRecAct , SIGNAL( triggered() ), this, SLOT( getcurrentrecommended() ) );
|
|
|
|
// QAction* getAllRecAct = new QAction(QIcon(IMAGE_DOWNLOADALL), tr( "Download" ), &contextMnu );
|
|
// connect( getAllRecAct , SIGNAL( triggered() ), this, SLOT( getallrecommended() ) );
|
|
|
|
contextMnu.addAction( getRecAct);
|
|
// contextMnu.addAction( getAllRecAct);
|
|
contextMnu.exec(QCursor::pos());
|
|
}
|
|
|
|
void MessagesDialog::folderlistWidgetCostumPopupMenu(QPoint point)
|
|
{
|
|
if (ui.listWidget->currentRow() != ROW_TRASHBOX) {
|
|
/* Context menu only neede for trash box */
|
|
return;
|
|
}
|
|
|
|
QMenu contextMnu(this);
|
|
|
|
contextMnu.addAction(tr("Empty trash"), this, SLOT(emptyTrash()));
|
|
|
|
contextMnu.exec(QCursor::pos());
|
|
}
|
|
|
|
void MessagesDialog::newmessage()
|
|
{
|
|
MessageComposer *nMsgDialog = MessageComposer::newMsg();
|
|
if (nMsgDialog == NULL) {
|
|
return;
|
|
}
|
|
|
|
/* fill it in */
|
|
//std::cerr << "MessagesDialog::newmessage()" << std::endl;
|
|
nMsgDialog->show();
|
|
nMsgDialog->activateWindow();
|
|
|
|
|
|
/* window will destroy itself! */
|
|
}
|
|
|
|
void MessagesDialog::editmessage()
|
|
{
|
|
std::string cid;
|
|
std::string mid;
|
|
|
|
if(!getCurrentMsg(cid, mid))
|
|
return ;
|
|
|
|
MessageComposer *pMsgDialog = MessageComposer::newMsg(mid);
|
|
if (pMsgDialog == NULL) {
|
|
return;
|
|
}
|
|
|
|
pMsgDialog->show();
|
|
pMsgDialog->activateWindow();
|
|
|
|
/* window will destroy itself! */
|
|
}
|
|
|
|
void MessagesDialog::replytomessage()
|
|
{
|
|
/* put msg on msgBoard, and switch to it. */
|
|
|
|
std::string cid;
|
|
std::string mid;
|
|
|
|
if(!getCurrentMsg(cid, mid))
|
|
return ;
|
|
|
|
mCurrCertId = cid;
|
|
mCurrMsgId = mid;
|
|
|
|
MessageComposer *nMsgDialog = MessageComposer::replyMsg(mid, false);
|
|
if (nMsgDialog == NULL) {
|
|
return;
|
|
}
|
|
|
|
nMsgDialog->show();
|
|
nMsgDialog->activateWindow();
|
|
|
|
/* window will destroy itself! */
|
|
}
|
|
|
|
void MessagesDialog::replyallmessage()
|
|
{
|
|
/* put msg on msgBoard, and switch to it. */
|
|
|
|
std::string cid;
|
|
std::string mid;
|
|
|
|
if(!getCurrentMsg(cid, mid))
|
|
return ;
|
|
|
|
mCurrCertId = cid;
|
|
mCurrMsgId = mid;
|
|
|
|
MessageComposer *nMsgDialog = MessageComposer::replyMsg(mid, true);
|
|
if (nMsgDialog == NULL) {
|
|
return;
|
|
}
|
|
|
|
nMsgDialog->show();
|
|
nMsgDialog->activateWindow();
|
|
|
|
/* window will destroy itself! */
|
|
}
|
|
|
|
void MessagesDialog::forwardmessage()
|
|
{
|
|
/* put msg on msgBoard, and switch to it. */
|
|
|
|
std::string cid;
|
|
std::string mid;
|
|
|
|
if(!getCurrentMsg(cid, mid))
|
|
return ;
|
|
|
|
mCurrCertId = cid;
|
|
mCurrMsgId = mid;
|
|
|
|
MessageComposer *nMsgDialog = MessageComposer::forwardMsg(mid);
|
|
if (nMsgDialog == NULL) {
|
|
return;
|
|
}
|
|
|
|
nMsgDialog->show();
|
|
nMsgDialog->activateWindow();
|
|
|
|
/* window will destroy itself! */
|
|
}
|
|
|
|
void MessagesDialog::togglefileview_internal()
|
|
{
|
|
/* if msg header visible -> change icon and tooltip
|
|
* three widgets...
|
|
*/
|
|
|
|
if (ui.expandFilesButton->isChecked()) {
|
|
ui.expandFilesButton->setIcon(QIcon(QString(":/images/edit_remove24.png")));
|
|
ui.expandFilesButton->setToolTip(tr("Hide"));
|
|
} else {
|
|
ui.expandFilesButton->setIcon(QIcon(QString(":/images/edit_add24.png")));
|
|
ui.expandFilesButton->setToolTip(tr("Expand"));
|
|
}
|
|
}
|
|
|
|
void MessagesDialog::togglefileview()
|
|
{
|
|
// save state of files view
|
|
Settings->setValueToGroup("MessageDialog", "expandFiles", ui.expandFilesButton->isChecked());
|
|
|
|
togglefileview_internal();
|
|
}
|
|
|
|
|
|
/* download the recommendations... */
|
|
void MessagesDialog::getcurrentrecommended()
|
|
{
|
|
MessageInfo msgInfo;
|
|
if (!rsMsgs -> getMessage(mCurrMsgId, msgInfo))
|
|
return;
|
|
|
|
std::list<std::string> srcIds;
|
|
srcIds.push_back(msgInfo.srcId);
|
|
|
|
QModelIndexList list = ui.msgList->selectionModel()->selectedIndexes();
|
|
|
|
std::map<int,FileInfo> files ;
|
|
|
|
for(QModelIndexList::const_iterator it(list.begin());it!=list.end();++it)
|
|
{
|
|
FileInfo& f(files[it->row()]) ;
|
|
|
|
switch(it->column())
|
|
{
|
|
case COLUMN_FILE_NAME:
|
|
f.fname = it->data().toString().toStdString() ;
|
|
break ;
|
|
case COLUMN_FILE_SIZE:
|
|
f.size = it->data().toULongLong() ;
|
|
break ;
|
|
case COLUMN_FILE_HASH:
|
|
f.hash = it->data().toString().toStdString() ;
|
|
break ;
|
|
}
|
|
}
|
|
|
|
for(std::map<int,FileInfo>::const_iterator it(files.begin());it!=files.end();++it)
|
|
{
|
|
const FileInfo& f(it->second) ;
|
|
std::cout << "Requesting file " << f.fname << ", size=" << f.size << ", hash=" << f.hash << std::endl ;
|
|
rsFiles -> FileRequest(it->second.fname,it->second.hash,it->second.size, "", RS_FILE_HINTS_NETWORK_WIDE, srcIds);
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
void MessagesDialog::getallrecommended()
|
|
{
|
|
/* get Message */
|
|
MessageInfo msgInfo;
|
|
if (!rsMsgs -> getMessage(mCurrMsgId, msgInfo))
|
|
{
|
|
return;
|
|
}
|
|
|
|
const std::list<FileInfo> &recList = msgInfo.files;
|
|
std::list<FileInfo>::const_iterator it;
|
|
|
|
std::list<std::string> fnames;
|
|
std::list<std::string> hashes;
|
|
std::list<int> sizes;
|
|
|
|
for(it = recList.begin(); it != recList.end(); it++)
|
|
{
|
|
fnames.push_back(it->fname);
|
|
hashes.push_back(it->hash);
|
|
sizes.push_back(it->size);
|
|
}
|
|
|
|
/* now do requests */
|
|
std::list<std::string>::const_iterator fit;
|
|
std::list<std::string>::const_iterator hit;
|
|
std::list<int>::const_iterator sit;
|
|
|
|
for(fit = fnames.begin(), hit = hashes.begin(), sit = sizes.begin(); fit != fnames.end(); fit++, hit++, sit++)
|
|
{
|
|
std::cerr << "MessagesDialog::getallrecommended() Calling File Request";
|
|
std::cerr << std::endl;
|
|
std::list<std::string> srcIds;
|
|
srcIds.push_back(msgInfo.srcId);
|
|
rsFiles -> FileRequest(*fit, *hit, *sit, "", 0, srcIds);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void MessagesDialog::changeBox(int)
|
|
{
|
|
if (m_bInChange) {
|
|
// already in change method
|
|
return;
|
|
}
|
|
|
|
m_bInChange = true;
|
|
|
|
MessagesModel->removeRows (0, MessagesModel->rowCount());
|
|
|
|
ui.tagWidget->setCurrentItem(NULL);
|
|
m_eListMode = LIST_BOX;
|
|
|
|
insertMessages();
|
|
insertMsgTxtAndFiles();
|
|
|
|
m_bInChange = false;
|
|
}
|
|
|
|
void MessagesDialog::changeTag(int)
|
|
{
|
|
if (m_bInChange) {
|
|
// already in change method
|
|
return;
|
|
}
|
|
|
|
m_bInChange = true;
|
|
|
|
MessagesModel->removeRows (0, MessagesModel->rowCount());
|
|
|
|
ui.listWidget->setCurrentItem(NULL);
|
|
m_eListMode = LIST_TAG;
|
|
|
|
insertMessages();
|
|
insertMsgTxtAndFiles();
|
|
|
|
m_bInChange = false;
|
|
}
|
|
|
|
void MessagesDialog::messagesTagsChanged()
|
|
{
|
|
if (m_nLockUpdate) {
|
|
return;
|
|
}
|
|
|
|
fillTags();
|
|
insertMessages();
|
|
}
|
|
|
|
static void InitIconAndFont(QStandardItem *pItem [COLUMN_COUNT])
|
|
{
|
|
QString sText = pItem [COLUMN_SUBJECT]->text();
|
|
QString mid = pItem [COLUMN_DATA]->data(ROLE_MSGID).toString();
|
|
int nFlag = pItem [COLUMN_DATA]->data(ROLE_MSGFLAGS).toInt();
|
|
|
|
// show the real "New" state
|
|
if (nFlag & RS_MSG_NEW) {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-state-new.png"));
|
|
} else {
|
|
if (nFlag & RS_MSG_UNREAD_BY_USER) {
|
|
if ((nFlag & (RS_MSG_REPLIED | RS_MSG_FORWARDED)) == RS_MSG_REPLIED) {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail-replied.png"));
|
|
} else if ((nFlag & (RS_MSG_REPLIED | RS_MSG_FORWARDED)) == RS_MSG_FORWARDED) {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail-forwarded.png"));
|
|
} else if ((nFlag & (RS_MSG_REPLIED | RS_MSG_FORWARDED)) == (RS_MSG_REPLIED | RS_MSG_FORWARDED)) {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail-replied-forw.png"));
|
|
} else {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail.png"));
|
|
}
|
|
} else {
|
|
if ((nFlag & (RS_MSG_REPLIED | RS_MSG_FORWARDED)) == RS_MSG_REPLIED) {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail-replied-read.png"));
|
|
} else if ((nFlag & (RS_MSG_REPLIED | RS_MSG_FORWARDED)) == RS_MSG_FORWARDED) {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail-forwarded-read.png"));
|
|
} else if ((nFlag & (RS_MSG_REPLIED | RS_MSG_FORWARDED)) == (RS_MSG_REPLIED | RS_MSG_FORWARDED)) {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail-replied-forw-read.png"));
|
|
} else {
|
|
pItem[COLUMN_SUBJECT]->setIcon(QIcon(":/images/message-mail-read.png"));
|
|
}
|
|
}
|
|
}
|
|
|
|
bool bNew = nFlag & (RS_MSG_NEW | RS_MSG_UNREAD_BY_USER);
|
|
|
|
// set icon
|
|
if (bNew) {
|
|
pItem[COLUMN_UNREAD]->setIcon(QIcon(":/images/message-state-unread.png"));
|
|
} else {
|
|
pItem[COLUMN_UNREAD]->setIcon(QIcon(":/images/message-state-read.png"));
|
|
}
|
|
|
|
// set font
|
|
for (int i = 0; i < COLUMN_COUNT; i++) {
|
|
QFont qf = pItem[i]->font();
|
|
qf.setBold(bNew);
|
|
pItem[i]->setFont(qf);
|
|
}
|
|
|
|
pItem[COLUMN_DATA]->setData(bNew, ROLE_UNREAD);
|
|
}
|
|
|
|
void MessagesDialog::insertMessages()
|
|
{
|
|
if (m_nLockUpdate) {
|
|
return;
|
|
}
|
|
|
|
std::cerr <<"MessagesDialog::insertMessages called";
|
|
fflush(0);
|
|
|
|
std::list<MsgInfoSummary> msgList;
|
|
std::list<MsgInfoSummary>::const_iterator it;
|
|
MessageInfo msgInfo;
|
|
bool bGotInfo;
|
|
QString text;
|
|
|
|
rsMsgs -> getMessageSummaries(msgList);
|
|
|
|
std::cerr << "MessagesDialog::insertMessages()" << std::endl;
|
|
fflush(0);
|
|
|
|
int nFilterColumn = FilterColumnFromComboBox(ui.filterColumnComboBox->currentIndex());
|
|
|
|
/* check the mode we are in */
|
|
unsigned int msgbox = 0;
|
|
bool bTrash = false;
|
|
bool bFill = true;
|
|
uint32_t nTagId = 0;
|
|
|
|
switch (m_eListMode) {
|
|
case LIST_NOTHING:
|
|
bFill = false;
|
|
break;
|
|
|
|
case LIST_BOX:
|
|
{
|
|
int listrow = ui.listWidget->currentRow();
|
|
|
|
switch (listrow) {
|
|
case ROW_INBOX:
|
|
msgbox = RS_MSG_INBOX;
|
|
ui.tabWidget->setTabIcon ( 0, QIcon(":/images/folder-inbox.png") );
|
|
ui.tabWidget->setTabText ( 0, tr ("Inbox") );
|
|
break;
|
|
case ROW_OUTBOX:
|
|
msgbox = RS_MSG_OUTBOX;
|
|
ui.tabWidget->setTabIcon ( 0, QIcon(":/images/folder-outbox.png") );
|
|
ui.tabWidget->setTabText ( 0, tr ("Outbox") );
|
|
break;
|
|
case ROW_DRAFTBOX:
|
|
msgbox = RS_MSG_DRAFTBOX;
|
|
ui.tabWidget->setTabIcon ( 0, QIcon(":/images/folder-draft.png") );
|
|
ui.tabWidget->setTabText ( 0, tr ("Drafts") );
|
|
break;
|
|
case ROW_SENTBOX:
|
|
msgbox = RS_MSG_SENTBOX;
|
|
ui.tabWidget->setTabIcon ( 0, QIcon(":/images/folder-sent.png") );
|
|
ui.tabWidget->setTabText ( 0, tr ("Sent") );
|
|
break;
|
|
case ROW_TRASHBOX:
|
|
bTrash = true;
|
|
ui.tabWidget->setTabIcon ( 0, QIcon(":/images/folder-trash.png") );
|
|
ui.tabWidget->setTabText ( 0, tr ("Trash") );
|
|
break;
|
|
default:
|
|
bFill = false;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case LIST_TAG:
|
|
{
|
|
QListWidgetItem *pItem = ui.tagWidget->currentItem();
|
|
if (pItem) {
|
|
nTagId = pItem->data (Qt::UserRole).toInt();
|
|
} else {
|
|
bFill = false;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
bFill = false;
|
|
}
|
|
|
|
if (msgbox == RS_MSG_INBOX) {
|
|
MessagesModel->setHeaderData(COLUMN_FROM, Qt::Horizontal, tr("From"));
|
|
} else {
|
|
MessagesModel->setHeaderData(COLUMN_FROM, Qt::Horizontal, tr("To"));
|
|
}
|
|
|
|
if (bFill) {
|
|
MsgTagType Tags;
|
|
rsMsgs->getMessageTagTypes(Tags);
|
|
|
|
/* search messages */
|
|
std::list<MsgInfoSummary> msgToShow;
|
|
for(it = msgList.begin(); it != msgList.end(); it++) {
|
|
if (m_eListMode == LIST_BOX) {
|
|
if (bTrash) {
|
|
if ((it->msgflags & RS_MSG_TRASH) == 0) {
|
|
continue;
|
|
}
|
|
} else {
|
|
if (it->msgflags & RS_MSG_TRASH) {
|
|
continue;
|
|
}
|
|
if ((it->msgflags & RS_MSG_BOXMASK) != msgbox) {
|
|
continue;
|
|
}
|
|
}
|
|
} else if (m_eListMode == LIST_TAG) {
|
|
MsgTagInfo tagInfo;
|
|
rsMsgs->getMessageTag(it->msgId, tagInfo);
|
|
if (std::find(tagInfo.tagIds.begin(), tagInfo.tagIds.end(), nTagId) == tagInfo.tagIds.end()) {
|
|
continue;
|
|
}
|
|
} else {
|
|
continue;
|
|
}
|
|
|
|
msgToShow.push_back(*it);
|
|
}
|
|
|
|
/* remove old items */
|
|
int nRowCount = MessagesModel->rowCount();
|
|
int nRow = 0;
|
|
for (nRow = 0; nRow < nRowCount; ) {
|
|
std::string msgIdFromRow = MessagesModel->item(nRow, COLUMN_DATA)->data(ROLE_MSGID).toString().toStdString();
|
|
for(it = msgToShow.begin(); it != msgToShow.end(); it++) {
|
|
if (it->msgId == msgIdFromRow) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (it == msgToShow.end ()) {
|
|
MessagesModel->removeRow (nRow);
|
|
nRowCount = MessagesModel->rowCount();
|
|
} else {
|
|
nRow++;
|
|
}
|
|
}
|
|
|
|
for(it = msgToShow.begin(); it != msgToShow.end(); it++)
|
|
{
|
|
/* check the message flags, to decide which
|
|
* group it should go in...
|
|
*
|
|
* InBox
|
|
* OutBox
|
|
* Drafts
|
|
* Sent
|
|
*
|
|
* FLAGS = OUTGOING.
|
|
* -> Outbox/Drafts/Sent
|
|
* + SENT -> Sent
|
|
* + IN_PROGRESS -> Draft.
|
|
* + nuffing -> Outbox.
|
|
* FLAGS = INCOMING = (!OUTGOING)
|
|
* -> + NEW -> Bold.
|
|
*
|
|
*/
|
|
|
|
bGotInfo = false;
|
|
msgInfo = MessageInfo(); // clear
|
|
|
|
// search exisisting items
|
|
nRowCount = MessagesModel->rowCount();
|
|
for (nRow = 0; nRow < nRowCount; nRow++) {
|
|
if (it->msgId == MessagesModel->item(nRow, COLUMN_DATA)->data(ROLE_MSGID).toString().toStdString()) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* make a widget per friend */
|
|
|
|
QStandardItem *item [COLUMN_COUNT];
|
|
|
|
bool bInsert = false;
|
|
|
|
if (nRow < nRowCount) {
|
|
for (int i = 0; i < COLUMN_COUNT; i++) {
|
|
item[i] = MessagesModel->item(nRow, i);
|
|
}
|
|
} else {
|
|
for (int i = 0; i < COLUMN_COUNT; i++) {
|
|
item[i] = new QStandardItem();
|
|
}
|
|
bInsert = true;
|
|
}
|
|
|
|
//set this false if you want to expand on double click
|
|
for (int i = 0; i < COLUMN_COUNT; i++) {
|
|
item[i]->setEditable(false);
|
|
}
|
|
|
|
/* So Text should be:
|
|
* (1) Msg / Broadcast
|
|
* (1b) Person / Channel Name
|
|
* (2) Rank
|
|
* (3) Date
|
|
* (4) Title
|
|
* (5) Msg
|
|
* (6) File Count
|
|
* (7) File Total
|
|
*/
|
|
|
|
QString dateString;
|
|
// Date First.... (for sorting)
|
|
{
|
|
QDateTime qdatetime;
|
|
qdatetime.setTime_t(it->ts);
|
|
|
|
// add string to all data
|
|
dateString = qdatetime.toString("_yyyyMMdd_hhmmss");
|
|
|
|
//if the mail is on same date show only time.
|
|
if (qdatetime.daysTo(QDateTime::currentDateTime()) == 0)
|
|
{
|
|
QTime qtime = qdatetime.time();
|
|
QVariant varTime(qtime);
|
|
item[COLUMN_DATE]->setData(varTime, Qt::DisplayRole);
|
|
}
|
|
else
|
|
{
|
|
QVariant varDateTime(qdatetime);
|
|
item[COLUMN_DATE]->setData(varDateTime, Qt::DisplayRole);
|
|
}
|
|
// for sorting
|
|
item[COLUMN_DATE]->setData(qdatetime, ROLE_SORT);
|
|
}
|
|
|
|
// From ....
|
|
{
|
|
if (msgbox == RS_MSG_INBOX || msgbox == RS_MSG_OUTBOX) {
|
|
text = QString::fromStdString(rsPeers->getPeerName(it->srcId));
|
|
} else {
|
|
if (bGotInfo || rsMsgs->getMessage(it->msgId, msgInfo)) {
|
|
bGotInfo = true;
|
|
|
|
text.clear();
|
|
|
|
std::list<std::string>::const_iterator pit;
|
|
for (pit = msgInfo.msgto.begin(); pit != msgInfo.msgto.end(); pit++)
|
|
{
|
|
if (text.isEmpty() == false) {
|
|
text += ", ";
|
|
}
|
|
|
|
std::string peerName = rsPeers->getPeerName(*pit);
|
|
if (peerName.empty()) {
|
|
text += PeerDefs::rsid("", *pit);
|
|
} else {
|
|
text += QString::fromStdString(peerName);
|
|
}
|
|
}
|
|
} else {
|
|
std::cerr << "MessagesDialog::insertMsgTxtAndFiles() Couldn't find Msg" << std::endl;
|
|
}
|
|
}
|
|
item[COLUMN_FROM]->setText(text);
|
|
item[COLUMN_FROM]->setData(text + dateString, ROLE_SORT);
|
|
}
|
|
|
|
// Subject
|
|
text = QString::fromStdWString(it->title);
|
|
item[COLUMN_SUBJECT]->setText(text);
|
|
item[COLUMN_SUBJECT]->setData(text + dateString, ROLE_SORT);
|
|
|
|
// internal data
|
|
QString msgId = QString::fromStdString(it->msgId);
|
|
item[COLUMN_DATA]->setData(QString::fromStdString(it->srcId), ROLE_SRCID);
|
|
item[COLUMN_DATA]->setData(msgId, ROLE_MSGID);
|
|
item[COLUMN_DATA]->setData(it->msgflags, ROLE_MSGFLAGS);
|
|
|
|
// Init icon and font
|
|
InitIconAndFont(item);
|
|
|
|
// Tags
|
|
MsgTagInfo tagInfo;
|
|
rsMsgs->getMessageTag(it->msgId, tagInfo);
|
|
|
|
text.clear();
|
|
|
|
// build tag names
|
|
std::map<uint32_t, std::pair<std::string, uint32_t> >::iterator Tag;
|
|
for (std::list<uint32_t>::iterator tagId = tagInfo.tagIds.begin(); tagId != tagInfo.tagIds.end(); tagId++) {
|
|
if (text.isEmpty() == false) {
|
|
text += ",";
|
|
}
|
|
Tag = Tags.types.find(*tagId);
|
|
if (Tag != Tags.types.end()) {
|
|
text += TagDefs::name(Tag->first, Tag->second.first);
|
|
} else {
|
|
// clean tagId
|
|
rsMsgs->setMessageTag(it->msgId, *tagId, false);
|
|
}
|
|
}
|
|
item[COLUMN_TAGS]->setText(text);
|
|
|
|
// set color
|
|
QBrush Brush; // standard
|
|
if (tagInfo.tagIds.size()) {
|
|
Tag = Tags.types.find(tagInfo.tagIds.front());
|
|
if (Tag != Tags.types.end()) {
|
|
Brush = QBrush(Tag->second.second);
|
|
} else {
|
|
// clean tagId
|
|
rsMsgs->setMessageTag(it->msgId, tagInfo.tagIds.front(), false);
|
|
}
|
|
}
|
|
for (int i = 0; i < COLUMN_COUNT; i++) {
|
|
item[i]->setForeground(Brush);
|
|
}
|
|
|
|
// No of Files.
|
|
{
|
|
std::ostringstream out;
|
|
out << it -> count;
|
|
item[COLUMN_ATTACHEMENTS] -> setText(QString::fromStdString(out.str()));
|
|
item[COLUMN_ATTACHEMENTS] -> setData(item[COLUMN_ATTACHEMENTS]->text() + dateString, ROLE_SORT);
|
|
item[COLUMN_ATTACHEMENTS] -> setTextAlignment(Qt::AlignHCenter);
|
|
}
|
|
|
|
if (nFilterColumn == COLUMN_CONTENT) {
|
|
// need content for filter
|
|
if (bGotInfo || rsMsgs->getMessage(it->msgId, msgInfo)) {
|
|
bGotInfo = true;
|
|
QTextDocument doc;
|
|
doc.setHtml(QString::fromStdWString(msgInfo.msg));
|
|
item[COLUMN_CONTENT]->setText(doc.toPlainText().replace(QString("\n"), QString(" ")));
|
|
} else {
|
|
std::cerr << "MessagesDialog::insertMsgTxtAndFiles() Couldn't find Msg" << std::endl;
|
|
item[COLUMN_CONTENT]->setText("");
|
|
}
|
|
}
|
|
|
|
if (bInsert) {
|
|
/* add to the list */
|
|
QList<QStandardItem *> itemList;
|
|
for (int i = 0; i < COLUMN_COUNT; i++) {
|
|
itemList.append(item[i]);
|
|
}
|
|
MessagesModel->appendRow(itemList);
|
|
}
|
|
}
|
|
} else {
|
|
MessagesModel->removeRows (0, MessagesModel->rowCount());
|
|
}
|
|
|
|
ui.messagestreeView->showColumn(COLUMN_ATTACHEMENTS);
|
|
ui.messagestreeView->showColumn(COLUMN_SUBJECT);
|
|
ui.messagestreeView->showColumn(COLUMN_UNREAD);
|
|
ui.messagestreeView->showColumn(COLUMN_FROM);
|
|
ui.messagestreeView->showColumn(COLUMN_DATE);
|
|
ui.messagestreeView->showColumn(COLUMN_TAGS);
|
|
ui.messagestreeView->hideColumn(COLUMN_CONTENT);
|
|
|
|
updateMessageSummaryList();
|
|
}
|
|
|
|
// current row in messagestreeView has changed
|
|
void MessagesDialog::currentChanged(const QModelIndex &index )
|
|
{
|
|
timer->stop();
|
|
timerIndex = index;
|
|
timer->start();
|
|
}
|
|
|
|
// click in messagestreeView
|
|
void MessagesDialog::clicked(const QModelIndex &index )
|
|
{
|
|
if (index.isValid() == false) {
|
|
return;
|
|
}
|
|
|
|
if (index.column() == COLUMN_UNREAD) {
|
|
int mappedRow = proxyModel->mapToSource(index).row();
|
|
|
|
QList<int> Rows;
|
|
Rows.append(mappedRow);
|
|
setMsgAsReadUnread(Rows, !isMessageRead(mappedRow));
|
|
insertMsgTxtAndFiles(index, false);
|
|
updateMessageSummaryList();
|
|
return;
|
|
}
|
|
|
|
timer->stop();
|
|
timerIndex = index;
|
|
// show current message directly
|
|
updateCurrentMessage();
|
|
}
|
|
|
|
// double click in messagestreeView
|
|
void MessagesDialog::doubleClicked(const QModelIndex &index)
|
|
{
|
|
/* activate row */
|
|
clicked (index);
|
|
|
|
/* edit message */
|
|
editmessage();
|
|
}
|
|
|
|
// show current message directly
|
|
void MessagesDialog::updateCurrentMessage()
|
|
{
|
|
timer->stop();
|
|
insertMsgTxtAndFiles(timerIndex);
|
|
}
|
|
|
|
void MessagesDialog::setMsgAsReadUnread(const QList<int> &Rows, bool bRead)
|
|
{
|
|
LockUpdate Lock (this, false);
|
|
|
|
for (int nRow = 0; nRow < Rows.size(); nRow++) {
|
|
QStandardItem* item[COLUMN_COUNT];
|
|
for(int nCol = 0; nCol < COLUMN_COUNT; nCol++)
|
|
{
|
|
item[nCol] = MessagesModel->item(Rows [nRow], nCol);
|
|
}
|
|
|
|
std::string mid = item[COLUMN_DATA]->data(ROLE_MSGID).toString().toStdString();
|
|
|
|
if (rsMsgs->MessageRead(mid, !bRead)) {
|
|
int nFlag = item[COLUMN_DATA]->data(ROLE_MSGFLAGS).toInt();
|
|
nFlag &= ~RS_MSG_NEW;
|
|
|
|
if (bRead) {
|
|
nFlag &= ~RS_MSG_UNREAD_BY_USER;
|
|
} else {
|
|
nFlag |= RS_MSG_UNREAD_BY_USER;
|
|
}
|
|
|
|
item[COLUMN_DATA]->setData(nFlag, ROLE_MSGFLAGS);
|
|
|
|
InitIconAndFont(item);
|
|
}
|
|
}
|
|
|
|
// LockUpdate
|
|
}
|
|
|
|
void MessagesDialog::markAsRead()
|
|
{
|
|
QList<int> RowsUnread;
|
|
getSelectedMsgCount (NULL, NULL, &RowsUnread);
|
|
|
|
setMsgAsReadUnread (RowsUnread, true);
|
|
updateMessageSummaryList();
|
|
}
|
|
|
|
void MessagesDialog::markAsUnread()
|
|
{
|
|
QList<int> RowsRead;
|
|
getSelectedMsgCount (NULL, &RowsRead, NULL);
|
|
|
|
setMsgAsReadUnread (RowsRead, false);
|
|
updateMessageSummaryList();
|
|
}
|
|
|
|
void MessagesDialog::clearTagLabels()
|
|
{
|
|
/* clear all tags */
|
|
while (tagLabels.size()) {
|
|
delete tagLabels.front();
|
|
tagLabels.pop_front();
|
|
}
|
|
while (ui.taglayout->count()) {
|
|
delete ui.taglayout->takeAt(0);
|
|
}
|
|
|
|
ui.tagslabel->setVisible(false);
|
|
}
|
|
|
|
void MessagesDialog::showTagLabels()
|
|
{
|
|
clearTagLabels();
|
|
|
|
if (mCurrMsgId.empty()) {
|
|
return;
|
|
}
|
|
|
|
MsgTagInfo tagInfo;
|
|
rsMsgs->getMessageTag(mCurrMsgId, tagInfo);
|
|
|
|
if (tagInfo.tagIds.empty() == false) {
|
|
ui.tagslabel->setVisible(true);
|
|
|
|
MsgTagType Tags;
|
|
rsMsgs->getMessageTagTypes(Tags);
|
|
|
|
std::map<uint32_t, std::pair<std::string, uint32_t> >::iterator Tag;
|
|
for (std::list<uint32_t>::iterator tagId = tagInfo.tagIds.begin(); tagId != tagInfo.tagIds.end(); tagId++) {
|
|
Tag = Tags.types.find(*tagId);
|
|
if (Tag != Tags.types.end()) {
|
|
QLabel *tagLabel = new QLabel(TagDefs::name(Tag->first, Tag->second.first), this);
|
|
tagLabel->setStyleSheet(TagDefs::labelStyleSheet(Tag->second.second));
|
|
tagLabels.push_back(tagLabel);
|
|
ui.taglayout->addWidget(tagLabel);
|
|
ui.taglayout->addSpacing(3);
|
|
}
|
|
}
|
|
ui.taglayout->addStretch();
|
|
} else {
|
|
ui.tagslabel->setVisible(false);
|
|
}
|
|
}
|
|
|
|
void MessagesDialog::insertMsgTxtAndFiles(QModelIndex Index, bool bSetToRead)
|
|
{
|
|
std::cerr << "MessagesDialog::insertMsgTxtAndFiles()" << std::endl;
|
|
|
|
/* get its Ids */
|
|
std::string cid;
|
|
std::string mid;
|
|
|
|
QModelIndex currentIndex = proxyModel->mapToSource(Index);
|
|
if (currentIndex.isValid() == false)
|
|
{
|
|
mCurrCertId.clear();
|
|
mCurrMsgId.clear();
|
|
|
|
/* blank it */
|
|
ui.dateText-> setText("");
|
|
ui.toText->setText("");
|
|
ui.fromText->setText("");
|
|
ui.filesText->setText("");
|
|
|
|
ui.cclabel->setVisible(false);
|
|
ui.ccText->setVisible(false);
|
|
ui.ccText->clear();
|
|
|
|
ui.bcclabel->setVisible(false);
|
|
ui.bccText->setVisible(false);
|
|
ui.bccText->clear();
|
|
|
|
ui.subjectText->setText("");
|
|
ui.msgList->clear();
|
|
ui.msgText->clear();
|
|
|
|
ui.actionSave_as->setDisabled(true);
|
|
ui.actionPrintPreview->setDisabled(true);
|
|
ui.actionPrint->setDisabled(true);
|
|
|
|
clearTagLabels();
|
|
|
|
return;
|
|
}
|
|
|
|
QStandardItem *item = MessagesModel->item(currentIndex.row(),COLUMN_DATA);
|
|
if (item == NULL) {
|
|
return;
|
|
}
|
|
cid = item->data(ROLE_SRCID).toString().toStdString();
|
|
mid = item->data(ROLE_MSGID).toString().toStdString();
|
|
|
|
int nCount = getSelectedMsgCount (NULL, NULL, NULL);
|
|
if (nCount == 1) {
|
|
ui.actionSave_as->setEnabled(true);
|
|
ui.actionPrintPreview->setEnabled(true);
|
|
ui.actionPrint->setEnabled(true);
|
|
} else {
|
|
ui.actionSave_as->setDisabled(true);
|
|
ui.actionPrintPreview->setDisabled(true);
|
|
ui.actionPrint->setDisabled(true);
|
|
}
|
|
|
|
if (mCurrMsgId == mid) {
|
|
// message doesn't changed
|
|
return;
|
|
}
|
|
|
|
clearTagLabels();
|
|
|
|
/* Save the Data.... for later */
|
|
|
|
mCurrCertId = cid;
|
|
mCurrMsgId = mid;
|
|
|
|
MessageInfo msgInfo;
|
|
if (!rsMsgs -> getMessage(mid, msgInfo))
|
|
{
|
|
std::cerr << "MessagesDialog::insertMsgTxtAndFiles() Couldn't find Msg" << std::endl;
|
|
return;
|
|
}
|
|
|
|
QList<int> Rows;
|
|
Rows.append(currentIndex.row());
|
|
|
|
bool bSetToReadOnActive = Settings->getMsgSetToReadOnActivate();
|
|
|
|
if (msgInfo.msgflags & RS_MSG_NEW) {
|
|
// set always to read or unread
|
|
if (bSetToReadOnActive == false || bSetToRead == false) {
|
|
// set locally to unread
|
|
setMsgAsReadUnread(Rows, false);
|
|
} else {
|
|
setMsgAsReadUnread(Rows, true);
|
|
}
|
|
updateMessageSummaryList();
|
|
} else {
|
|
if ((msgInfo.msgflags & RS_MSG_UNREAD_BY_USER) && bSetToRead && bSetToReadOnActive) {
|
|
// set to read
|
|
setMsgAsReadUnread(Rows, true);
|
|
updateMessageSummaryList();
|
|
}
|
|
}
|
|
|
|
const std::list<FileInfo> &recList = msgInfo.files;
|
|
std::list<FileInfo>::const_iterator it;
|
|
|
|
/* get a link to the table */
|
|
QTreeWidget *tree = ui.msgList;
|
|
|
|
/* get the MessageInfo */
|
|
|
|
tree->clear();
|
|
|
|
QList<QTreeWidgetItem *> items;
|
|
for(it = recList.begin(); it != recList.end(); it++)
|
|
{
|
|
/* make a widget per person */
|
|
QTreeWidgetItem *item = new QTreeWidgetItem((QTreeWidget*)0);
|
|
/* (0) Filename */
|
|
item -> setText(COLUMN_FILE_NAME, QString::fromStdString(it->fname));
|
|
//std::cerr << "Msg FileItem(" << it->fname.length() << ") :" << it->fname << std::endl;
|
|
|
|
item -> setText(COLUMN_FILE_SIZE, QString::number(it->size)); /* (1) Size */
|
|
item -> setText(COLUMN_FILE_HASH, QString::fromStdString(it->hash));
|
|
|
|
/* add to the list */
|
|
items.append(item);
|
|
}
|
|
|
|
/* add the items in! */
|
|
tree->insertTopLevelItems(0, items);
|
|
|
|
/* iterate through the sources */
|
|
std::list<std::string>::const_iterator pit;
|
|
|
|
QString text;
|
|
QString name;
|
|
QString rsid;
|
|
for(pit = msgInfo.msgto.begin(); pit != msgInfo.msgto.end(); pit++)
|
|
{
|
|
rsid = PeerDefs::rsidFromId(*pit, &name);
|
|
text += "<a style='color: black;'href='" + rsid + "'> " + name + "</a>" + " ";
|
|
}
|
|
ui.toText->setText(text);
|
|
|
|
if (msgInfo.msgcc.size() > 0) {
|
|
ui.cclabel->setVisible(true);
|
|
ui.ccText->setVisible(true);
|
|
|
|
text.clear();
|
|
for(pit = msgInfo.msgcc.begin(); pit != msgInfo.msgcc.end(); pit++)
|
|
{
|
|
rsid = PeerDefs::rsidFromId(*pit, &name);
|
|
text += "<a style='color: black;'href='" + rsid + "'> " + name + "</a>" + " ";
|
|
}
|
|
ui.ccText->setText(text);
|
|
} else {
|
|
ui.cclabel->setVisible(false);
|
|
ui.ccText->setVisible(false);
|
|
ui.ccText->clear();
|
|
}
|
|
|
|
if (msgInfo.msgbcc.size() > 0) {
|
|
ui.bcclabel->setVisible(true);
|
|
ui.bccText->setVisible(true);
|
|
|
|
text.clear();
|
|
for(pit = msgInfo.msgbcc.begin(); pit != msgInfo.msgbcc.end(); pit++)
|
|
{
|
|
rsid = PeerDefs::rsidFromId(*pit, &name);
|
|
text += "<a style='color: black;'href='" + rsid + "'> " + name + "</a>" + " ";
|
|
}
|
|
ui.bccText->setText(text);
|
|
} else {
|
|
ui.bcclabel->setVisible(false);
|
|
ui.bccText->setVisible(false);
|
|
ui.bccText->clear();
|
|
}
|
|
|
|
{
|
|
QDateTime qtime;
|
|
qtime.setTime_t(msgInfo.ts);
|
|
QString timestamp = qtime.toString("dd.MM.yyyy hh:mm:ss");
|
|
ui.dateText-> setText(timestamp);
|
|
}
|
|
|
|
std::string sSrcId;
|
|
if ((msgInfo.msgflags & RS_MSG_BOXMASK) == RS_MSG_OUTBOX) {
|
|
// outgoing message are from me
|
|
sSrcId = rsPeers->getOwnId();
|
|
} else {
|
|
sSrcId = msgInfo.srcId;
|
|
}
|
|
rsid = PeerDefs::rsidFromId(sSrcId, &name);
|
|
text += "<a style='color: black;'href='" + rsid + "'> " + name + "</a>" + " ";
|
|
|
|
ui.fromText->setText("<a style='color: blue;' href='" + rsid + "'> " + name +"</a>");
|
|
ui.fromText->setToolTip(rsid);
|
|
|
|
ui.subjectText->setText(QString::fromStdWString(msgInfo.title));
|
|
|
|
text = QString::fromStdWString(msgInfo.msg);
|
|
Emoticons::formatText(text);
|
|
ui.msgText->setHtml(text);
|
|
|
|
{
|
|
std::ostringstream out;
|
|
out << "(" << msgInfo.count << " Files)";
|
|
ui.filesText->setText(QString::fromStdString(out.str()));
|
|
}
|
|
|
|
showTagLabels();
|
|
|
|
std::cerr << "MessagesDialog::insertMsgTxtAndFiles() Msg Displayed OK!" << std::endl;
|
|
}
|
|
|
|
bool MessagesDialog::getCurrentMsg(std::string &cid, std::string &mid)
|
|
{
|
|
QModelIndex currentIndex = ui.messagestreeView->currentIndex();
|
|
currentIndex = proxyModel->mapToSource(currentIndex);
|
|
int rowSelected = -1;
|
|
|
|
/* get its Ids */
|
|
if (currentIndex.isValid() == false)
|
|
{
|
|
//If no message is selected. assume first message is selected.
|
|
if(MessagesModel->rowCount() == 0)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
rowSelected = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
rowSelected = currentIndex.row();
|
|
}
|
|
|
|
QStandardItem *item = MessagesModel->item(rowSelected,COLUMN_DATA);
|
|
if (item == NULL) {
|
|
return false;
|
|
}
|
|
cid = item->data(ROLE_SRCID).toString().toStdString();
|
|
mid = item->data(ROLE_MSGID).toString().toStdString();
|
|
return true;
|
|
}
|
|
|
|
void MessagesDialog::removemessage()
|
|
{
|
|
LockUpdate Lock (this, true);
|
|
|
|
QList<QModelIndex> selectedIndexList= ui.messagestreeView->selectionModel() -> selectedIndexes ();
|
|
QList<int> rowList;
|
|
QModelIndex selectedIndex;
|
|
|
|
for(QList<QModelIndex>::iterator it = selectedIndexList.begin(); it != selectedIndexList.end(); it++) {
|
|
selectedIndex = proxyModel->mapToSource(*it);
|
|
int row = selectedIndex.row();
|
|
if (rowList.contains(row) == false)
|
|
{
|
|
rowList.append(row);
|
|
}
|
|
}
|
|
|
|
bool bDelete = false;
|
|
int listrow = ui.listWidget->currentRow();
|
|
if (listrow == ROW_TRASHBOX) {
|
|
bDelete = true;
|
|
} else {
|
|
if (QApplication::keyboardModifiers() & Qt::ShiftModifier) {
|
|
bDelete = true;
|
|
}
|
|
}
|
|
|
|
for(QList<int>::const_iterator it1 = rowList.begin(); it1 != rowList.end(); it1++) {
|
|
QStandardItem *pItem = MessagesModel->item((*it1), COLUMN_DATA);
|
|
if (pItem) {
|
|
QString mid = pItem->data(ROLE_MSGID).toString();
|
|
if (bDelete) {
|
|
rsMsgs->MessageDelete(mid.toStdString());
|
|
} else {
|
|
rsMsgs->MessageToTrash(mid.toStdString(), true);
|
|
}
|
|
}
|
|
}
|
|
|
|
// LockUpdate -> insertMessages();
|
|
}
|
|
|
|
void MessagesDialog::undeletemessage()
|
|
{
|
|
LockUpdate Lock (this, true);
|
|
|
|
QList<int> Rows;
|
|
getSelectedMsgCount (&Rows, NULL, NULL);
|
|
for (int nRow = 0; nRow < Rows.size(); nRow++) {
|
|
QString mid = MessagesModel->item (Rows [nRow], COLUMN_DATA)->data(ROLE_MSGID).toString();
|
|
rsMsgs->MessageToTrash(mid.toStdString(), false);
|
|
}
|
|
|
|
// LockUpdate -> insertMessages();
|
|
}
|
|
|
|
void MessagesDialog::print()
|
|
{
|
|
#ifndef QT_NO_PRINTER
|
|
QPrinter printer(QPrinter::HighResolution);
|
|
printer.setFullPage(true);
|
|
QPrintDialog *dlg = new QPrintDialog(&printer, this);
|
|
if (ui.msgText->textCursor().hasSelection())
|
|
dlg->addEnabledOption(QAbstractPrintDialog::PrintSelection);
|
|
dlg->setWindowTitle(tr("Print Document"));
|
|
if (dlg->exec() == QDialog::Accepted) {
|
|
ui.msgText->print(&printer);
|
|
}
|
|
delete dlg;
|
|
#endif
|
|
}
|
|
|
|
void MessagesDialog::printpreview()
|
|
{
|
|
PrintPreview *preview = new PrintPreview(ui.msgText->document(), this);
|
|
preview->setWindowModality(Qt::WindowModal);
|
|
preview->setAttribute(Qt::WA_DeleteOnClose);
|
|
preview->show();
|
|
}
|
|
|
|
void MessagesDialog::anchorClicked (const QUrl& link )
|
|
{
|
|
std::cerr << "MessagesDialog::anchorClicked link.scheme() : " << link.scheme().toStdString() << std::endl;
|
|
|
|
RetroShareLink::processUrl(link, RSLINK_PROCESS_NOTIFY_ALL);
|
|
}
|
|
|
|
bool MessagesDialog::fileSave()
|
|
{
|
|
if (fileName.isEmpty())
|
|
return fileSaveAs();
|
|
|
|
QFile file(fileName);
|
|
if (!file.open(QFile::WriteOnly))
|
|
return false;
|
|
QTextStream ts(&file);
|
|
ts.setCodec(QTextCodec::codecForName("UTF-8"));
|
|
ts << ui.msgText->document()->toHtml("UTF-8");
|
|
ui.msgText->document()->setModified(false);
|
|
return true;
|
|
}
|
|
|
|
bool MessagesDialog::fileSaveAs()
|
|
{
|
|
QString fn = QFileDialog::getSaveFileName(this, tr("Save as..."),
|
|
QString(), tr("HTML-Files (*.htm *.html);;All Files (*)"));
|
|
if (fn.isEmpty())
|
|
return false;
|
|
setCurrentFileName(fn);
|
|
return fileSave();
|
|
}
|
|
|
|
void MessagesDialog::setCurrentFileName(const QString &fileName)
|
|
{
|
|
this->fileName = fileName;
|
|
ui.msgText->document()->setModified(false);
|
|
|
|
setWindowModified(false);
|
|
}
|
|
|
|
void MessagesDialog::setToolbarButtonStyle(Qt::ToolButtonStyle style)
|
|
{
|
|
ui.newmessageButton->setToolButtonStyle(style);
|
|
ui.removemessageButton->setToolButtonStyle(style);
|
|
ui.replymessageButton->setToolButtonStyle(style);
|
|
ui.replyallmessageButton->setToolButtonStyle(style);
|
|
ui.forwardmessageButton->setToolButtonStyle(style);
|
|
ui.tagButton->setToolButtonStyle(style);
|
|
ui.printbutton->setToolButtonStyle(style);
|
|
ui.viewtoolButton->setToolButtonStyle(style);
|
|
}
|
|
|
|
void MessagesDialog::buttonsicononly()
|
|
{
|
|
setToolbarButtonStyle(Qt::ToolButtonIconOnly);
|
|
|
|
Settings->setValueToGroup("MessageDialog", "ToolButon_Stlye", Qt::ToolButtonIconOnly);
|
|
}
|
|
|
|
void MessagesDialog::buttonstextbesideicon()
|
|
{
|
|
setToolbarButtonStyle(Qt::ToolButtonTextBesideIcon);
|
|
|
|
Settings->setValueToGroup("MessageDialog", "ToolButon_Stlye", Qt::ToolButtonTextBesideIcon);
|
|
}
|
|
|
|
void MessagesDialog::buttonstextundericon()
|
|
{
|
|
setToolbarButtonStyle(Qt::ToolButtonTextUnderIcon);
|
|
|
|
Settings->setValueToGroup("MessageDialog", "ToolButon_Stlye", Qt::ToolButtonTextUnderIcon);
|
|
}
|
|
|
|
void MessagesDialog::loadToolButtonsettings()
|
|
{
|
|
Qt::ToolButtonStyle style = (Qt::ToolButtonStyle) Settings->valueFromGroup("MessageDialog", "ToolButon_Stlye", Qt::ToolButtonIconOnly).toInt();
|
|
setToolbarButtonStyle(style);
|
|
}
|
|
|
|
void MessagesDialog::filterRegExpChanged()
|
|
{
|
|
QRegExp regExp(ui.filterPatternLineEdit->text(), Qt::CaseInsensitive , QRegExp::FixedString);
|
|
proxyModel->setFilterRegExp(regExp);
|
|
|
|
QString text = ui.filterPatternLineEdit->text();
|
|
|
|
if (text.isEmpty())
|
|
{
|
|
ui.clearButton->hide();
|
|
}
|
|
else
|
|
{
|
|
ui.clearButton->show();
|
|
}
|
|
}
|
|
|
|
void MessagesDialog::filterColumnChanged()
|
|
{
|
|
if (m_bProcessSettings) {
|
|
return;
|
|
}
|
|
|
|
int nFilterColumn = FilterColumnFromComboBox(ui.filterColumnComboBox->currentIndex());
|
|
if (nFilterColumn == COLUMN_CONTENT) {
|
|
// need content ... refill
|
|
insertMessages();
|
|
}
|
|
proxyModel->setFilterKeyColumn(nFilterColumn);
|
|
|
|
// save index
|
|
Settings->setValueToGroup("MessageDialog", "filterColumn", nFilterColumn);
|
|
}
|
|
|
|
void MessagesDialog::updateMessageSummaryList()
|
|
{
|
|
unsigned int newInboxCount = 0;
|
|
unsigned int newOutboxCount = 0;
|
|
unsigned int newDraftCount = 0;
|
|
unsigned int newSentboxCount = 0;
|
|
unsigned int inboxCount = 0;
|
|
unsigned int trashboxCount = 0;
|
|
|
|
/* calculating the new messages */
|
|
// rsMsgs->getMessageCount (&inboxCount, &newInboxCount, &newOutboxCount, &newDraftCount, &newSentboxCount);
|
|
|
|
std::list<MsgInfoSummary> msgList;
|
|
std::list<MsgInfoSummary>::const_iterator it;
|
|
|
|
rsMsgs->getMessageSummaries(msgList);
|
|
|
|
QMap<int, int> tagCount;
|
|
|
|
/* calculating the new messages */
|
|
for (it = msgList.begin(); it != msgList.end(); it++) {
|
|
/* calcluate tag count */
|
|
MsgTagInfo tagInfo;
|
|
rsMsgs->getMessageTag(it->msgId, tagInfo);
|
|
for (std::list<uint32_t>::iterator tagId = tagInfo.tagIds.begin(); tagId != tagInfo.tagIds.end(); tagId++) {
|
|
int nCount = tagCount [*tagId];
|
|
nCount++;
|
|
tagCount [*tagId] = nCount;
|
|
}
|
|
|
|
/* calculate box */
|
|
if (it->msgflags & RS_MSG_TRASH) {
|
|
trashboxCount++;
|
|
continue;
|
|
}
|
|
|
|
switch (it->msgflags & RS_MSG_BOXMASK) {
|
|
case RS_MSG_INBOX:
|
|
inboxCount++;
|
|
if (it->msgflags & (RS_MSG_NEW | RS_MSG_UNREAD_BY_USER)) {
|
|
newInboxCount++;
|
|
}
|
|
break;
|
|
case RS_MSG_OUTBOX:
|
|
newOutboxCount++;
|
|
break;
|
|
case RS_MSG_DRAFTBOX:
|
|
newDraftCount++;
|
|
break;
|
|
case RS_MSG_SENTBOX:
|
|
newSentboxCount++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
int listrow = ui.listWidget->currentRow();
|
|
QString textTotal;
|
|
|
|
switch (listrow)
|
|
{
|
|
case ROW_INBOX:
|
|
textTotal = tr("Total:") + " " + QString::number(inboxCount);
|
|
ui.total_label->setText(textTotal);
|
|
break;
|
|
case ROW_OUTBOX:
|
|
textTotal = tr("Total:") + " " + QString::number(newOutboxCount);
|
|
ui.total_label->setText(textTotal);
|
|
break;
|
|
case ROW_DRAFTBOX:
|
|
textTotal = tr("Total:") + " " + QString::number(newDraftCount);
|
|
ui.total_label->setText(textTotal);
|
|
break;
|
|
case ROW_SENTBOX:
|
|
textTotal = tr("Total:") + " " + QString::number(newSentboxCount);
|
|
ui.total_label->setText(textTotal);
|
|
break;
|
|
case ROW_TRASHBOX:
|
|
textTotal = tr("Total:") + " " + QString::number(trashboxCount);
|
|
ui.total_label->setText(textTotal);
|
|
break;
|
|
}
|
|
|
|
|
|
QString textItem;
|
|
/*updating the labels in leftcolumn*/
|
|
|
|
//QList<QListWidgetItem *> QListWidget::findItems ( const QString & text, Qt::MatchFlags flags ) const
|
|
QListWidgetItem* item = ui.listWidget->item(ROW_INBOX);
|
|
if (newInboxCount != 0)
|
|
{
|
|
textItem = tr("Inbox") + " (" + QString::number(newInboxCount)+")";
|
|
item->setText(textItem);
|
|
QFont qf = item->font();
|
|
qf.setBold(true);
|
|
item->setFont(qf);
|
|
item->setIcon(QIcon(":/images/folder-inbox-new.png"));
|
|
item->setForeground(QBrush(QColor(49, 106, 197)));
|
|
}
|
|
else
|
|
{
|
|
textItem = tr("Inbox");
|
|
item->setText(textItem);
|
|
QFont qf = item->font();
|
|
qf.setBold(false);
|
|
item->setFont(qf);
|
|
item->setIcon(QIcon(":/images/folder-inbox.png"));
|
|
item->setForeground(QBrush(QColor(0, 0, 0)));
|
|
}
|
|
|
|
//QList<QListWidgetItem *> QListWidget::findItems ( const QString & text, Qt::MatchFlags flags ) const
|
|
item = ui.listWidget->item(ROW_OUTBOX);
|
|
if (newOutboxCount != 0)
|
|
{
|
|
textItem = tr("Outbox") + " (" + QString::number(newOutboxCount)+")";
|
|
item->setText(textItem);
|
|
QFont qf = item->font();
|
|
qf.setBold(true);
|
|
item->setFont(qf);
|
|
}
|
|
else
|
|
{
|
|
textItem = tr("Outbox");
|
|
item->setText(textItem);
|
|
QFont qf = item->font();
|
|
qf.setBold(false);
|
|
item->setFont(qf);
|
|
|
|
}
|
|
|
|
//QList<QListWidgetItem *> QListWidget::findItems ( const QString & text, Qt::MatchFlags flags ) const
|
|
item = ui.listWidget->item(ROW_DRAFTBOX);
|
|
if (newDraftCount != 0)
|
|
{
|
|
textItem = tr("Drafts") + " (" + QString::number(newDraftCount)+")";
|
|
item->setText(textItem);
|
|
QFont qf = item->font();
|
|
qf.setBold(true);
|
|
item->setFont(qf);
|
|
}
|
|
else
|
|
{
|
|
textItem = tr("Drafts");
|
|
item->setText(textItem);
|
|
QFont qf = item->font();
|
|
qf.setBold(false);
|
|
item->setFont(qf);
|
|
|
|
}
|
|
|
|
item = ui.listWidget->item(ROW_TRASHBOX);
|
|
if (trashboxCount != 0)
|
|
{
|
|
textItem = tr("Trash") + " (" + QString::number(trashboxCount)+")";
|
|
item->setText(textItem);
|
|
}
|
|
else
|
|
{
|
|
textItem = tr("Trash");
|
|
item->setText(textItem);
|
|
}
|
|
|
|
|
|
/* set tag counts */
|
|
int nRowCount = ui.tagWidget->count();
|
|
for (int nRow = 0; nRow < nRowCount; nRow++) {
|
|
QListWidgetItem *pItem = ui.tagWidget->item(nRow);
|
|
int nCount = tagCount[pItem->data(Qt::UserRole).toInt()];
|
|
|
|
QString sText = pItem->data(Qt::UserRole + 1).toString();
|
|
if (nCount) {
|
|
sText += " (" + QString::number(nCount) + ")";
|
|
}
|
|
|
|
pItem->setText(sText);
|
|
}
|
|
}
|
|
|
|
/** clear Filter **/
|
|
void MessagesDialog::clearFilter()
|
|
{
|
|
ui.filterPatternLineEdit->clear();
|
|
ui.filterPatternLineEdit->setFocus();
|
|
}
|
|
|
|
void MessagesDialog::tagAboutToShow()
|
|
{
|
|
// activate actions from the first selected row
|
|
MsgTagInfo tagInfo;
|
|
|
|
QList<int> Rows;
|
|
getSelectedMsgCount (&Rows, NULL, NULL);
|
|
|
|
if (Rows.size()) {
|
|
QStandardItem* pItem = MessagesModel->item(Rows [0], COLUMN_DATA);
|
|
std::string msgId = pItem->data(ROLE_MSGID).toString().toStdString();
|
|
|
|
rsMsgs->getMessageTag(msgId, tagInfo);
|
|
}
|
|
|
|
QMenu *pMenu = ui.tagButton->menu();
|
|
|
|
foreach(QObject *pObject, pMenu->children()) {
|
|
QAction *pAction = qobject_cast<QAction*> (pObject);
|
|
if (pAction == NULL) {
|
|
continue;
|
|
}
|
|
|
|
const QMap<QString, QVariant> &Values = pAction->data().toMap();
|
|
if (Values.size () != ACTION_TAGSINDEX_SIZE) {
|
|
continue;
|
|
}
|
|
if (Values [ACTION_TAGSINDEX_TYPE] != ACTION_TAGS_TAG) {
|
|
continue;
|
|
}
|
|
|
|
std::list<uint32_t>::iterator tagId = std::find(tagInfo.tagIds.begin(), tagInfo.tagIds.end(), Values [ACTION_TAGSINDEX_ID]);
|
|
pAction->setChecked(tagId != tagInfo.tagIds.end());
|
|
}
|
|
}
|
|
|
|
void MessagesDialog::tagTriggered(QAction *pAction)
|
|
{
|
|
if (pAction == NULL) {
|
|
return;
|
|
}
|
|
|
|
const QMap<QString, QVariant> &Values = pAction->data().toMap();
|
|
if (Values.size () != ACTION_TAGSINDEX_SIZE) {
|
|
return;
|
|
}
|
|
|
|
LockUpdate Lock (this, false);
|
|
|
|
bool bRemoveAll = false;
|
|
int nId = 0;
|
|
bool bSet = false;
|
|
|
|
if (Values [ACTION_TAGSINDEX_TYPE] == ACTION_TAGS_REMOVEALL) {
|
|
// remove all tags
|
|
bRemoveAll = true;
|
|
} else if (Values [ACTION_TAGSINDEX_TYPE] == ACTION_TAGS_NEWTAG) {
|
|
// new tag
|
|
MsgTagType Tags;
|
|
rsMsgs->getMessageTagTypes(Tags);
|
|
|
|
NewTag TagDlg(Tags);
|
|
if (TagDlg.exec() == QDialog::Accepted && TagDlg.m_nId) {
|
|
std::map<uint32_t, std::pair<std::string, uint32_t> >::iterator Tag = Tags.types.find(TagDlg.m_nId);
|
|
if (Tag == Tags.types.end()) {
|
|
return;
|
|
}
|
|
if (rsMsgs->setMessageTagType(Tag->first, Tag->second.first, Tag->second.second) == false) {
|
|
return;
|
|
}
|
|
fillTags();
|
|
nId = TagDlg.m_nId;
|
|
bSet = true;
|
|
} else {
|
|
return;
|
|
}
|
|
} else if (Values [ACTION_TAGSINDEX_TYPE].toInt() == ACTION_TAGS_TAG) {
|
|
nId = Values [ACTION_TAGSINDEX_ID].toInt();
|
|
if (nId == 0) {
|
|
return;
|
|
}
|
|
bSet = pAction->isChecked();
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
QList<int> Rows;
|
|
getSelectedMsgCount (&Rows, NULL, NULL);
|
|
for (int nRow = 0; nRow < Rows.size(); nRow++) {
|
|
QStandardItem* pItem = MessagesModel->item(Rows [nRow], COLUMN_DATA);
|
|
std::string msgId = pItem->data(ROLE_MSGID).toString().toStdString();
|
|
|
|
if (bRemoveAll) {
|
|
// remove all
|
|
rsMsgs->setMessageTag(msgId, 0, false);
|
|
Lock.setUpdate(true);
|
|
|
|
} else {
|
|
// set or unset tag
|
|
MsgTagInfo tagInfo;
|
|
if (rsMsgs->setMessageTag(msgId, nId, bSet)) {
|
|
Lock.setUpdate(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
showTagLabels();
|
|
|
|
// LockUpdate -> insertMessages();
|
|
}
|
|
|
|
void MessagesDialog::linkActivated(QString link)
|
|
{
|
|
if (link.isEmpty() == false) {
|
|
std::string id = PeerDefs::idFromRsid(link, false);
|
|
|
|
RsPeerDetails detail;
|
|
if (rsPeers->getPeerDetails(id, detail) && detail.accept_connection) {
|
|
MessageComposer::msgFriend(detail.id, false);
|
|
}
|
|
}
|
|
}
|
|
|
|
void MessagesDialog::emptyTrash()
|
|
{
|
|
LockUpdate Lock (this, true);
|
|
|
|
std::list<MsgInfoSummary> msgList;
|
|
rsMsgs->getMessageSummaries(msgList);
|
|
|
|
std::list<MsgInfoSummary>::const_iterator it;
|
|
for (it = msgList.begin(); it != msgList.end(); it++) {
|
|
if (it->msgflags & RS_MSG_TRASH) {
|
|
rsMsgs->MessageDelete(it->msgId);
|
|
}
|
|
}
|
|
|
|
// LockUpdate -> insertMessages();
|
|
}
|