nheko/src/timeline/widgets/ImageItem.cc

221 lines
6.3 KiB
C++
Raw Normal View History

2017-04-28 13:56:45 +02:00
/*
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
*
* 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 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <QBrush>
#include <QDebug>
#include <QDesktopServices>
2017-09-10 11:58:00 +02:00
#include <QFileInfo>
2017-04-28 13:56:45 +02:00
#include <QPainter>
#include <QPixmap>
#include "dialogs/ImageOverlay.h"
2017-11-30 12:53:28 +01:00
#include "timeline/widgets/ImageItem.h"
2017-04-28 13:56:45 +02:00
2017-08-20 12:47:22 +02:00
ImageItem::ImageItem(QSharedPointer<MatrixClient> client,
const mtx::events::RoomEvent<mtx::events::msg::Image> &event,
2017-09-10 11:58:00 +02:00
QWidget *parent)
2017-08-20 12:47:22 +02:00
: QWidget(parent)
2017-11-05 22:04:55 +01:00
, event_{event}
, client_{client}
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
setMouseTracking(true);
setCursor(Qt::PointingHandCursor);
setAttribute(Qt::WA_Hover, true);
2017-04-28 13:56:45 +02:00
url_ = QString::fromStdString(event.content.url);
text_ = QString::fromStdString(event.content.body);
2017-09-10 11:58:00 +02:00
QList<QString> url_parts = url_.toString().split("mxc://");
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
if (url_parts.size() != 2) {
qDebug() << "Invalid format for image" << url_.toString();
return;
}
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
QString media_params = url_parts[1];
url_ = QString("%1/_matrix/media/r0/download/%2")
.arg(client_.data()->getHomeServer().toString(), media_params);
2017-04-28 13:56:45 +02:00
client_.data()->downloadImage(QString::fromStdString(event.event_id), url_);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
connect(client_.data(),
SIGNAL(imageDownloaded(const QString &, const QPixmap &)),
this,
SLOT(imageDownloaded(const QString &, const QPixmap &)));
}
ImageItem::ImageItem(QSharedPointer<MatrixClient> client,
const QString &url,
const QString &filename,
QWidget *parent)
: QWidget(parent)
2017-11-05 22:04:55 +01:00
, url_{url}
, text_{QFileInfo(filename).fileName()}
, client_{client}
2017-09-10 11:58:00 +02:00
{
setMouseTracking(true);
setCursor(Qt::PointingHandCursor);
setAttribute(Qt::WA_Hover, true);
QList<QString> url_parts = url_.toString().split("mxc://");
if (url_parts.size() != 2) {
qDebug() << "Invalid format for image" << url_.toString();
return;
}
QString media_params = url_parts[1];
url_ = QString("%1/_matrix/media/r0/download/%2")
.arg(client_.data()->getHomeServer().toString(), media_params);
setImage(QPixmap(filename));
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
void
ImageItem::imageDownloaded(const QString &event_id, const QPixmap &img)
2017-04-28 13:56:45 +02:00
{
if (event_id != QString::fromStdString(event_.event_id))
2017-09-10 11:58:00 +02:00
return;
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
setImage(img);
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
void
ImageItem::openUrl()
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
if (url_.toString().isEmpty())
return;
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
if (!QDesktopServices::openUrl(url_))
qWarning() << "Could not open url" << url_.toString();
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
void
ImageItem::scaleImage()
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
if (image_.isNull())
return;
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
auto width_ratio = (double)max_width_ / (double)image_.width();
auto height_ratio = (double)max_height_ / (double)image_.height();
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
auto min_aspect_ratio = std::min(width_ratio, height_ratio);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
if (min_aspect_ratio > 1) {
width_ = image_.width();
height_ = image_.height();
} else {
width_ = image_.width() * min_aspect_ratio;
height_ = image_.height() * min_aspect_ratio;
}
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
setFixedSize(width_, height_);
scaled_image_ =
image_.scaled(width_, height_, Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
QSize
ImageItem::sizeHint() const
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
if (image_.isNull())
return QSize(max_width_, bottom_height_);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
return QSize(width_, height_);
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
void
ImageItem::setImage(const QPixmap &image)
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
image_ = image;
scaleImage();
update();
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
void
ImageItem::mousePressEvent(QMouseEvent *event)
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
if (event->button() != Qt::LeftButton)
return;
if (image_.isNull()) {
openUrl();
return;
}
auto point = event->pos();
// Click on the text box.
if (QRect(0, height_ - bottom_height_, width_, bottom_height_).contains(point)) {
openUrl();
} else {
auto image_dialog = new dialogs::ImageOverlay(image_, this);
2017-09-10 11:58:00 +02:00
image_dialog->show();
}
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
void
ImageItem::resizeEvent(QResizeEvent *event)
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
Q_UNUSED(event);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
scaleImage();
2017-04-28 13:56:45 +02:00
}
2017-08-20 12:47:22 +02:00
void
ImageItem::paintEvent(QPaintEvent *event)
2017-04-28 13:56:45 +02:00
{
2017-09-10 11:58:00 +02:00
Q_UNUSED(event);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
QPainter painter(this);
painter.setRenderHint(QPainter::Antialiasing);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
QFont font("Open Sans");
font.setPixelSize(12);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
QFontMetrics metrics(font);
int fontHeight = metrics.height();
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
if (image_.isNull()) {
int height = fontHeight + 10;
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
QString elidedText = metrics.elidedText(text_, Qt::ElideRight, max_width_ - 10);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
setFixedSize(metrics.width(elidedText), fontHeight + 10);
2017-09-10 11:58:00 +02:00
painter.setFont(font);
painter.setPen(QPen(QColor(66, 133, 244)));
painter.drawText(QPoint(0, height / 2 + 2), elidedText);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
return;
}
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
painter.fillRect(QRect(0, 0, width_, height_), scaled_image_);
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
if (underMouse()) {
// Bottom text section
painter.fillRect(QRect(0, height_ - bottom_height_, width_, bottom_height_),
QBrush(QColor(33, 33, 33, 128)));
2017-04-28 13:56:45 +02:00
2017-09-10 11:58:00 +02:00
QString elidedText = metrics.elidedText(text_, Qt::ElideRight, width_ - 10);
2017-04-28 13:56:45 +02:00
2017-09-28 13:42:16 +02:00
font.setWeight(80);
2017-09-10 11:58:00 +02:00
painter.setFont(font);
painter.setPen(QPen(QColor("white")));
painter.drawText(QPoint(5, height_ - fontHeight / 2), elidedText);
}
2017-04-28 13:56:45 +02:00
}