#include #include #include #include #include #include "RSTextBrowser.h" #include "RSImageBlockWidget.h" #include //To get RsAccounts RSTextBrowser::RSTextBrowser(QWidget *parent) : QTextBrowser(parent) { setOpenExternalLinks(true); setOpenLinks(false); mShowImages = true; mImageBlockWidget = NULL; mLinkClickActive = true; highlighter = new RsSyntaxHighlighter(this); connect(this, SIGNAL(anchorClicked(QUrl)), this, SLOT(linkClicked(QUrl))); } void RSTextBrowser::linkClicked(const QUrl &url) { if (!mLinkClickActive) { return; } // some links are opened directly in the QTextBrowser with open external links set to true, // so we handle links by our own #ifdef TO_DO // If we want extra file links to be anonymous, we need to insert the actual source here. if(url.host() == HOST_EXTRAFILE) { std::cerr << "Extra file link detected. Adding parent id " << _target_sslid << " to sourcelist" << std::endl; RetroShareLink link ; link.fromUrl(url) ; link.createExtraFile( link.name(),link.size(),link.hash(), _target_ssl_id) ; QDesktopServices::openUrl(link.toUrl()); } else #endif QDesktopServices::openUrl(url); } void RSTextBrowser::setPlaceholderText(const QString &text) { mPlaceholderText = text; viewport()->repaint(); } void RSTextBrowser::paintEvent(QPaintEvent *event) { QTextBrowser::paintEvent(event); if (mPlaceholderText.isEmpty() == false && document()->isEmpty()) { QWidget *vieportWidget = viewport(); QPainter painter(vieportWidget); QPen pen = painter.pen(); QColor color = pen.color(); color.setAlpha(128); pen.setColor(color); painter.setPen(pen); painter.drawText(QRect(QPoint(), vieportWidget->size()), Qt::AlignHCenter | Qt::AlignVCenter | Qt::TextWordWrap, mPlaceholderText); } #ifdef RSTEXTBROWSER_CHECKIMAGE_DEBUG QPainter painter(viewport()); QPen pen = painter.pen(); pen.setWidth(2); pen.setColor(QColor(qRgba(255,0,0,128))); painter.setPen(pen); painter.drawRect(mCursorRectStart); pen.setColor(QColor(qRgba(0,255,0,128))); painter.setPen(pen); painter.drawRect(mCursorRectLeft); pen.setColor(QColor(qRgba(0,0,255,128))); painter.setPen(pen); painter.drawRect(mCursorRectRight); pen.setColor(QColor(qRgba(0,0,0,128))); painter.setPen(pen); painter.drawRect(mCursorRectEnd); #endif } QVariant RSTextBrowser::loadResource(int type, const QUrl &name) { // case 1: always trust the image if it comes from an internal resource. if(name.scheme().compare("qrc",Qt::CaseInsensitive)==0 && type == QTextDocument::ImageResource) return QTextBrowser::loadResource(type, name); // case 2: always trust the image if it comes from local Config or Data directories. if(type == QTextDocument::ImageResource) { QFileInfo fi = QFileInfo(name.path()); if(fi.exists() && fi.isFile()) { QString cpath = fi.canonicalFilePath(); if (cpath.startsWith(QDir(QString::fromUtf8(RsAccounts::ConfigDirectory().c_str())).canonicalPath(),Qt::CaseInsensitive) || cpath.startsWith(QDir(QString::fromUtf8(RsAccounts::DataDirectory().c_str())).canonicalPath(),Qt::CaseInsensitive)) return QTextBrowser::loadResource(type, name); }} // case 3: only display if the user allows it. Data resources can be bad (svg bombs) but we filter them out globally at the network layer. // It would be good to add here a home-made resource loader that only loads images and not svg crap, just in case. if(name.scheme().compare("data",Qt::CaseInsensitive)==0 && mShowImages) return QTextBrowser::loadResource(type, name); // case 4: otherwise, do not display std::cerr << "TEXTBROWSER: refusing load ressource request: type=" << type << " scheme=" << name.scheme().toStdString() << ", url=" << name.toString().toStdString() << std::endl; if (mImageBlockWidget) mImageBlockWidget->show(); return getBlockedImage(); } QPixmap RSTextBrowser::getBlockedImage() { return QPixmap(":/images/imageblocked_24.png"); } void RSTextBrowser::setImageBlockWidget(RSImageBlockWidget *widget) { if (mImageBlockWidget) { // disconnect disconnect(mImageBlockWidget, SIGNAL(destroyed()), this, SLOT(destroyImageBlockWidget())); disconnect(mImageBlockWidget, SIGNAL(showImages()), this, SLOT(showImages())); } mImageBlockWidget = widget; if (mImageBlockWidget) { // connect connect(mImageBlockWidget, SIGNAL(destroyed()), this, SLOT(destroyImageBlockWidget())); connect(mImageBlockWidget, SIGNAL(showImages()), this, SLOT(showImages())); } resetImagesStatus(false); } void RSTextBrowser::destroyImageBlockWidget() { mImageBlockWidget = NULL; } void RSTextBrowser::showImages() { if (mImageBlockWidget && sender() == mImageBlockWidget) { mImageBlockWidget->hide(); } if (mShowImages) { return; } mShowImages = true; QString html = toHtml(); clear(); setHtml(html); } void RSTextBrowser::resetImagesStatus(bool load) { if (mImageBlockWidget) { mImageBlockWidget->hide(); } mShowImages = load; } void RSTextBrowser::activateLinkClick(bool active) { mLinkClickActive = active; } /** * @brief RSTextBrowser::checkImage * @param pos where to check if image is shown in viewport coordinate * @param imageStr return html source of cursor * @return True if an image is under cursor */ bool RSTextBrowser::checkImage(QPoint pos, QString &imageStr) { //Get text cursor under pos. But if pos is under text browser end line this return last cursor. QTextCursor cursor = cursorForPosition(pos); //First get rect of cursor (could be at left or right of image) QRect cursorRectStart = cursorRect(cursor); //Second get text cursor.movePosition(QTextCursor::Left, QTextCursor::MoveAnchor, 1);//To get character just before QRect cursorRectLeft = cursorRect(cursor); cursor.movePosition(QTextCursor::Right, QTextCursor::KeepAnchor, 2); QRect cursorRectRight = cursorRect(cursor); imageStr = cursor.selection().toHtml(); #ifdef RSTEXTBROWSER_CHECKIMAGE_DEBUG mCursorRectStart = cursorRectStart; mCursorRectLeft = cursorRectLeft; mCursorRectRight = cursorRectRight; std::cerr << "cursorRect LTRB :" << cursorRectStart.left() << ";" << cursorRectStart.top() << ";" << cursorRectStart.right() << ";" << cursorRectStart.bottom() << std::endl; std::cerr << "cursorRectLeft :" << cursorRectLeft.left() << ";" << cursorRectLeft.top() << ";" << cursorRectLeft.right() << ";" << cursorRectLeft.bottom() << std::endl; std::cerr << "cursorRectRight :" << cursorRectRight.left() << ";" << cursorRectRight.top() << ";" << cursorRectRight.right() << ";" << cursorRectRight.bottom() << std::endl; std::cerr << "pos XY :" << pos.x() << ";" << pos.y() << std::endl; #endif QRect cursorRectEnd = cursorRectStart; //Finally set left with right of precedent character. if (cursorRectEnd.top() < cursorRectLeft.bottom()) { cursorRectEnd.setLeft(cursorRectLeft.right()); } else { //Image on new line cursorRectEnd.setLeft(0); } //And set Right with left of next character. if (cursorRectEnd.bottom() > cursorRectRight.top()) { cursorRectEnd.setRight(cursorRectRight.left()); } else { //New line after Image. } #ifdef RSTEXTBROWSER_CHECKIMAGE_DEBUG mCursorRectEnd = cursorRectEnd; std::cerr << "final cursorRect:" << cursorRectEnd.left() << ";" << cursorRectEnd.top() << ";" << cursorRectEnd.right() << ";" << cursorRectEnd.bottom() << std::endl; viewport()->update(); #endif //If pos is on text rect if (cursorRectEnd.contains(pos)) { return imageStr.indexOf("base64,") != -1; } return false; } /** * @brief RSTextBrowser::anchorForPosition Replace anchorAt that doesn't works as expected. * @param pos Where to get anchor from text * @return anchor If text at pos is inside anchor, else empty string. */ QString RSTextBrowser::anchorForPosition(const QPoint &pos) const { // Many calls when time label shows up QTextCursor cursor = cursorForPosition(pos); cursor.select(QTextCursor::WordUnderCursor); QString word = cursor.selectedText(); QString anchor = ""; if (word.isEmpty()) return anchor; // For finding positions QTextCursor cursor_line = cursorForPosition(pos); cursor_line.select(QTextCursor::LineUnderCursor); QString line = cursor_line.selectedText(); // End of nickname (more or less, of course, because some can has colon in // name) int end_of_name = line.indexOf(": ") + 1; // Start of nickname (after time) int space_index = line.indexOf(' ') + 1; int word_index = line.indexOf(word) + 1; int once = 1; cursor_line.movePosition(QTextCursor::StartOfLine, QTextCursor::MoveAnchor, once); bool after_name = cursor_line.position() + end_of_name < cursor.position(); bool cursor_after_time = space_index < word_index; bool cursor_within_name = word_index <= end_of_name; if (!after_name && cursor_after_time && cursor_within_name) { cursor.movePosition(QTextCursor::StartOfLine, QTextCursor::MoveAnchor, once); cursor.movePosition(QTextCursor::NextCharacter, QTextCursor::MoveAnchor, space_index); cursor.movePosition(QTextCursor::NextCharacter, QTextCursor::KeepAnchor, end_of_name - space_index); if (cursor.selectedText().isEmpty()) { return anchor; } } else { return anchor; } QRegExp rx("