2012-09-26 19:32:53 -04:00
|
|
|
/****************************************************************
|
|
|
|
* This file is distributed under the following license:
|
|
|
|
*
|
|
|
|
* Copyright (c) 2010, RetroShare Team
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
****************************************************************/
|
|
|
|
|
2015-09-02 18:27:14 -04:00
|
|
|
#include <QApplication>
|
|
|
|
#include <QClipboard>
|
2012-09-26 19:32:53 -04:00
|
|
|
#include <QMimeData>
|
|
|
|
#include <QTextDocumentFragment>
|
2013-07-04 15:36:12 -04:00
|
|
|
#include <QCompleter>
|
|
|
|
#include <QAbstractItemView>
|
|
|
|
#include <QKeyEvent>
|
|
|
|
#include <QScrollBar>
|
2013-12-30 14:12:16 -05:00
|
|
|
#include <QMenu>
|
|
|
|
|
|
|
|
#include "MimeTextEdit.h"
|
|
|
|
#include "util/HandleRichText.h"
|
|
|
|
#include "gui/RetroShareLink.h"
|
|
|
|
|
|
|
|
#include <retroshare/rspeers.h>
|
2012-09-26 19:32:53 -04:00
|
|
|
|
|
|
|
MimeTextEdit::MimeTextEdit(QWidget *parent)
|
2014-12-24 21:39:40 -05:00
|
|
|
: RSTextEdit(parent), mCompleter(0)
|
2012-09-26 19:32:53 -04:00
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
mCompleterKeyModifiers = Qt::ControlModifier;
|
|
|
|
mCompleterKey = Qt::Key_Space;
|
|
|
|
mForceCompleterShowNextKeyEvent = false;
|
2015-12-12 20:01:06 -05:00
|
|
|
highliter = new RsSyntaxHighlighter(this);
|
2012-09-26 19:32:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool MimeTextEdit::canInsertFromMimeData(const QMimeData* source) const
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= 0x040700
|
|
|
|
// embedded images are not supported before QT 4.7.0
|
2013-01-21 19:14:10 -05:00
|
|
|
if (source != NULL) {
|
|
|
|
if (source->hasImage()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2012-09-26 19:32:53 -04:00
|
|
|
#endif
|
|
|
|
|
2014-12-24 21:39:40 -05:00
|
|
|
return RSTextEdit::canInsertFromMimeData(source);
|
2012-09-26 19:32:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::insertFromMimeData(const QMimeData* source)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= 0x040700
|
|
|
|
// embedded images are not supported before QT 4.7.0
|
2013-01-21 19:14:10 -05:00
|
|
|
if (source != NULL) {
|
|
|
|
if (source->hasImage()) {
|
|
|
|
// insert as embedded image
|
|
|
|
QImage image = qvariant_cast<QImage>(source->imageData());
|
|
|
|
if (image.isNull() == false) {
|
|
|
|
QString encodedImage;
|
|
|
|
if (RsHtml::makeEmbeddedImage(image, encodedImage, 640*480)) {
|
|
|
|
QTextDocumentFragment fragment = QTextDocumentFragment::fromHtml(encodedImage);
|
2013-12-30 14:12:16 -05:00
|
|
|
textCursor().insertFragment(fragment);
|
2013-01-21 19:14:10 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-09-26 19:32:53 -04:00
|
|
|
#endif
|
2017-04-19 08:15:34 -04:00
|
|
|
if (source == NULL) return;
|
2012-09-26 19:32:53 -04:00
|
|
|
|
2016-10-30 19:23:39 -04:00
|
|
|
//insert retroshare links
|
|
|
|
QList<RetroShareLink> links;
|
|
|
|
RSLinkClipboard::parseText(source->text(), links);
|
|
|
|
if(links.size() > 0)
|
|
|
|
{
|
|
|
|
for(int i = 0; i < links.size(); ++i)
|
|
|
|
insertHtml(links[i].toHtml() + "<br>");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-12-24 21:39:40 -05:00
|
|
|
return RSTextEdit::insertFromMimeData(source);
|
2012-09-26 19:32:53 -04:00
|
|
|
}
|
2013-07-04 15:36:12 -04:00
|
|
|
|
|
|
|
void MimeTextEdit::setCompleter(QCompleter *completer)
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
if (mCompleter)
|
|
|
|
QObject::disconnect(mCompleter, 0, this, 0);
|
2013-07-04 15:36:12 -04:00
|
|
|
|
2013-12-30 14:12:16 -05:00
|
|
|
mCompleter = completer;
|
2013-07-04 15:36:12 -04:00
|
|
|
|
2013-12-30 14:12:16 -05:00
|
|
|
if (!mCompleter)
|
|
|
|
return;
|
2013-07-04 15:36:12 -04:00
|
|
|
|
2013-12-30 14:12:16 -05:00
|
|
|
mCompleter->setWidget(this);
|
|
|
|
mCompleter->setCompletionMode(QCompleter::PopupCompletion);
|
|
|
|
mCompleter->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
QObject::connect(mCompleter, SIGNAL(activated(QString)), this, SLOT(insertCompletion(QString)));
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
QCompleter *MimeTextEdit::completer() const
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
return mCompleter;
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::insertCompletion(const QString& completion)
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
if (mCompleter->widget() != this)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QTextCursor tc = textCursor();
|
|
|
|
if (mCompleter->completionPrefix().length() > 0) {
|
|
|
|
tc.movePosition(QTextCursor::PreviousWord, QTextCursor::KeepAnchor);
|
|
|
|
}
|
|
|
|
tc.removeSelectedText();
|
|
|
|
tc.insertText(mCompleterStartString+completion);
|
|
|
|
mCompleterStartString.clear();
|
|
|
|
setTextCursor(tc);
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
QString MimeTextEdit::textUnderCursor() const
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
QTextCursor tc = textCursor();
|
|
|
|
tc.select(QTextCursor::WordUnderCursor);
|
|
|
|
return tc.selectedText();
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::focusInEvent(QFocusEvent *e)
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
if (mCompleter)
|
|
|
|
mCompleter->setWidget(this);
|
|
|
|
|
2014-12-24 21:39:40 -05:00
|
|
|
RSTextEdit::focusInEvent(e);
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::keyPressEvent(QKeyEvent *e)
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
if (mCompleter && mCompleter->popup()->isVisible()) {
|
|
|
|
// The following keys are forwarded by the completer to the widget
|
|
|
|
switch (e->key()) {
|
|
|
|
case Qt::Key_Enter:
|
|
|
|
case Qt::Key_Return:
|
|
|
|
case Qt::Key_Escape:
|
|
|
|
case Qt::Key_Tab:
|
|
|
|
case Qt::Key_Backtab:
|
|
|
|
mCompleter->popup()->hide();
|
|
|
|
mForceCompleterShowNextKeyEvent=false;
|
|
|
|
e->ignore();
|
|
|
|
return; // let the completer do default behavior
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isShortcut = ((e->modifiers() & mCompleterKeyModifiers) && e->key() == mCompleterKey);
|
|
|
|
if (isShortcut && !mForceCompleterShowNextKeyEvent) {
|
|
|
|
mCompleterStartString.clear();
|
|
|
|
}
|
|
|
|
isShortcut |= mForceCompleterShowNextKeyEvent;
|
|
|
|
if (!mCompleter || !isShortcut) // do not process the shortcut when we have a completer
|
2014-12-24 21:39:40 -05:00
|
|
|
RSTextEdit::keyPressEvent(e);
|
2013-12-30 14:12:16 -05:00
|
|
|
|
|
|
|
if (!mCompleter) return; //Nothing else to do if not mCompleter initialized
|
|
|
|
|
|
|
|
if (!isShortcut && (mCompleter && !mCompleter->popup()->isVisible())) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mForceCompleterShowNextKeyEvent) {
|
|
|
|
static QString eow(" ~!@#$%^&*()_+{}|:\"<>?,./;'[]\\-="); // end of word
|
|
|
|
if (!isShortcut && !e->text().isEmpty() && eow.contains(e->text())){
|
|
|
|
mCompleter->popup()->hide();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QString completionPrefix = textUnderCursor();
|
|
|
|
if (completionPrefix != mCompleter->completionPrefix()) {
|
|
|
|
mCompleter->setCompletionPrefix(completionPrefix);
|
|
|
|
mCompleter->popup()->setCurrentIndex(mCompleter->completionModel()->index(0, 0));
|
|
|
|
}
|
2013-07-04 15:36:12 -04:00
|
|
|
|
2013-12-30 14:12:16 -05:00
|
|
|
QRect cr = cursorRect();
|
|
|
|
cr.setWidth(mCompleter->popup()->sizeHintForColumn(0) + mCompleter->popup()->verticalScrollBar()->sizeHint().width());
|
|
|
|
mCompleter->complete(cr); // popup it up!
|
|
|
|
|
|
|
|
if (mCompleter->completionCount()==0 && isShortcut){
|
2014-12-24 21:39:40 -05:00
|
|
|
RSTextEdit::keyPressEvent(e);// Process the key if no match
|
2013-12-30 14:12:16 -05:00
|
|
|
}
|
|
|
|
mForceCompleterShowNextKeyEvent = false;
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::setCompleterKeyModifiers(Qt::KeyboardModifier modifiers)
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
mCompleterKeyModifiers = modifiers;
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
2013-12-30 14:12:16 -05:00
|
|
|
|
2013-07-04 15:36:12 -04:00
|
|
|
Qt::KeyboardModifier MimeTextEdit::getCompleterKeyModifiers() const
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
return mCompleterKeyModifiers;
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::setCompleterKey(Qt::Key key)
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
mCompleterKey = key;
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
2013-12-30 14:12:16 -05:00
|
|
|
|
2013-07-04 15:36:12 -04:00
|
|
|
Qt::Key MimeTextEdit::getCompleterKey() const
|
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
return mCompleterKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::forceCompleterShowNextKeyEvent(QString startString)
|
|
|
|
{
|
|
|
|
if (!mCompleter) return; //Nothing else to do if not mCompleter initialized
|
|
|
|
|
|
|
|
if(!mCompleter->popup()->isVisible()){
|
|
|
|
mForceCompleterShowNextKeyEvent = true;
|
|
|
|
mCompleterStartString = startString;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::addContextMenuAction(QAction *action)
|
|
|
|
{
|
|
|
|
mContextMenuActions.push_back(action);
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
2013-12-30 14:12:16 -05:00
|
|
|
|
|
|
|
void MimeTextEdit::contextMenuEvent(QContextMenuEvent *e)
|
2013-07-04 15:36:12 -04:00
|
|
|
{
|
2013-12-30 14:12:16 -05:00
|
|
|
emit calculateContextMenuActions();
|
|
|
|
|
|
|
|
QMenu *contextMenu = createStandardContextMenu(e->pos());
|
|
|
|
|
|
|
|
/* Add actions for pasting links */
|
2015-09-02 18:27:14 -04:00
|
|
|
contextMenu->addAction( tr("Paste as plain text"), this, SLOT(pastePlainText()));
|
2015-12-06 20:28:27 -05:00
|
|
|
QAction *spoilerAction = contextMenu->addAction(tr("Spoiler"), this, SLOT(spoiler()));
|
|
|
|
spoilerAction->setToolTip(tr("Select text to hide, then push this button"));
|
2013-12-30 14:12:16 -05:00
|
|
|
contextMenu->addSeparator();
|
|
|
|
QAction *pasteLinkAction = contextMenu->addAction(QIcon(":/images/pasterslink.png"), tr("Paste RetroShare Link"), this, SLOT(pasteLink()));
|
|
|
|
contextMenu->addAction(QIcon(":/images/pasterslink.png"), tr("Paste my certificate link"), this, SLOT(pasteOwnCertificateLink()));
|
2013-07-18 17:50:32 -04:00
|
|
|
|
2013-12-30 14:12:16 -05:00
|
|
|
if (RSLinkClipboard::empty()) {
|
|
|
|
pasteLinkAction->setDisabled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<QAction*>::iterator it;
|
|
|
|
for (it = mContextMenuActions.begin(); it != mContextMenuActions.end(); ++it) {
|
|
|
|
contextMenu->addAction(*it);
|
|
|
|
}
|
|
|
|
|
|
|
|
contextMenu->exec(QCursor::pos());
|
|
|
|
|
|
|
|
delete(contextMenu);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MimeTextEdit::pasteLink()
|
|
|
|
{
|
|
|
|
insertHtml(RSLinkClipboard::toHtml()) ;
|
2013-07-04 15:36:12 -04:00
|
|
|
}
|
2013-12-30 14:12:16 -05:00
|
|
|
|
|
|
|
void MimeTextEdit::pasteOwnCertificateLink()
|
|
|
|
{
|
|
|
|
RetroShareLink link;
|
2014-03-17 16:56:06 -04:00
|
|
|
RsPeerId ownId = rsPeers->getOwnId();
|
2013-12-30 14:12:16 -05:00
|
|
|
|
|
|
|
if (link.createCertificate(ownId)) {
|
|
|
|
insertHtml(link.toHtml() + " ");
|
|
|
|
}
|
2013-07-18 17:50:32 -04:00
|
|
|
}
|
2015-09-02 18:27:14 -04:00
|
|
|
|
|
|
|
void MimeTextEdit::pastePlainText()
|
|
|
|
{
|
2016-01-23 12:17:49 -05:00
|
|
|
insertPlainText(QApplication::clipboard()->text().remove(QChar(-4)));//Char used when image on text.
|
2015-09-02 18:27:14 -04:00
|
|
|
}
|
2015-12-06 20:28:27 -05:00
|
|
|
|
|
|
|
void MimeTextEdit::spoiler()
|
|
|
|
{
|
|
|
|
RsHtml::insertSpoilerText(this->textCursor());
|
|
|
|
}
|