mirror of
https://invent.kde.org/network/falkon.git
synced 2024-11-11 17:42:10 +01:00
280 lines
9.0 KiB
C++
280 lines
9.0 KiB
C++
/* ============================================================
|
|
* QupZilla - WebKit based browser
|
|
* Copyright (C) 2010-2014 David Rosca <nowrep@gmail.com>
|
|
*
|
|
* 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 "pagescreen.h"
|
|
#include "ui_pagescreen.h"
|
|
#include "tabbedwebview.h"
|
|
#include "webpage.h"
|
|
#include "qztools.h"
|
|
#include "browserwindow.h"
|
|
#include "settings.h"
|
|
|
|
#include <QFileDialog>
|
|
#include <QMessageBox>
|
|
#include <QWebFrame>
|
|
#include <QLabel>
|
|
#include <QTimer>
|
|
#include <QMovie>
|
|
#include <QPushButton>
|
|
#include <QCloseEvent>
|
|
#include <QPrinter>
|
|
|
|
#if QT_VERSION >= 0x050000
|
|
#include <QtConcurrent/QtConcurrentRun>
|
|
#else
|
|
#include <QtConcurrentRun>
|
|
#endif
|
|
|
|
PageScreen::PageScreen(WebView* view, QWidget* parent)
|
|
: QDialog(parent)
|
|
, ui(new Ui::PageScreen)
|
|
, m_view(view)
|
|
, m_imageScaling(0)
|
|
{
|
|
setAttribute(Qt::WA_DeleteOnClose);
|
|
ui->setupUi(this);
|
|
|
|
m_formats[0] = QLatin1String("PNG");
|
|
m_formats[1] = QLatin1String("BMP");
|
|
m_formats[2] = QLatin1String("JPG");
|
|
m_formats[3] = QLatin1String("PPM");
|
|
m_formats[4] = QLatin1String("TIFF");
|
|
m_formats[5] = QLatin1String("PDF");
|
|
|
|
QHashIterator<int, QString> i(m_formats);
|
|
while (i.hasNext()) {
|
|
i.next();
|
|
ui->formats->addItem(tr("Save as %1").arg(i.value()));
|
|
}
|
|
|
|
// Set png as a default format
|
|
m_pageTitle = m_view->title();
|
|
|
|
Settings settings;
|
|
const QString name = QzTools::filterCharsFromFilename(m_pageTitle).replace(QLatin1Char(' '), QLatin1Char('_'));
|
|
const QString path = settings.value("FileDialogPaths/PageScreen-Location", QDir::homePath()).toString();
|
|
ui->location->setText(QString("%1/%2.png").arg(path, name));
|
|
|
|
QMovie* mov = new QMovie(":html/loading.gif");
|
|
ui->label->setMovie(mov);
|
|
mov->start();
|
|
|
|
connect(ui->changeLocation, SIGNAL(clicked()), this, SLOT(changeLocation()));
|
|
connect(ui->formats, SIGNAL(currentIndexChanged(int)), this, SLOT(formatChanged()));
|
|
connect(ui->buttonBox->button(QDialogButtonBox::Save), SIGNAL(clicked()), this, SLOT(dialogAccepted()));
|
|
connect(ui->buttonBox->button(QDialogButtonBox::Cancel), SIGNAL(clicked()), this, SLOT(close()));
|
|
|
|
QTimer::singleShot(200, this, SLOT(createThumbnail()));
|
|
}
|
|
|
|
void PageScreen::formatChanged()
|
|
{
|
|
QString text = ui->location->text();
|
|
int pos = text.lastIndexOf(QLatin1Char('.'));
|
|
|
|
if (pos > -1) {
|
|
text = text.left(pos + 1) + m_formats[ui->formats->currentIndex()].toLower();
|
|
}
|
|
else {
|
|
text.append(QLatin1Char('.') + m_formats[ui->formats->currentIndex()].toLower());
|
|
}
|
|
|
|
ui->location->setText(text);
|
|
}
|
|
|
|
void PageScreen::changeLocation()
|
|
{
|
|
const QString name = QzTools::filterCharsFromFilename(m_pageTitle).replace(QLatin1Char(' '), QLatin1Char('_'));
|
|
const QString suggestedPath = QString("%1/%2.%3").arg(QDir::homePath(), name, m_formats[ui->formats->currentIndex()].toLower());
|
|
|
|
const QString path = QzTools::getOpenFileName("PageScreen-Location", this, tr("Choose location..."), suggestedPath);
|
|
|
|
if (!path.isEmpty()) {
|
|
ui->location->setText(path);
|
|
}
|
|
}
|
|
|
|
void PageScreen::dialogAccepted()
|
|
{
|
|
if (!ui->location->text().isEmpty()) {
|
|
if (QFile::exists(ui->location->text())) {
|
|
const QString text = tr("File '%1' already exists. Do you want to overwrite it?").arg(ui->location->text());
|
|
QMessageBox::StandardButton button = QMessageBox::warning(this, tr("File already exists"), text,
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::No);
|
|
|
|
if (button != QMessageBox::Yes) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
|
|
|
const QString format = m_formats[ui->formats->currentIndex()];
|
|
if (format == QLatin1String("PDF")) {
|
|
saveAsDocument(format);
|
|
}
|
|
else {
|
|
saveAsImage(format);
|
|
}
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
|
|
close();
|
|
}
|
|
}
|
|
|
|
void PageScreen::saveAsImage(const QString &format)
|
|
{
|
|
const QString suffix = QLatin1Char('.') + format.toLower();
|
|
|
|
QString pathWithoutSuffix = ui->location->text();
|
|
if (pathWithoutSuffix.endsWith(suffix, Qt::CaseInsensitive)) {
|
|
pathWithoutSuffix = pathWithoutSuffix.mid(0, pathWithoutSuffix.length() - suffix.length());
|
|
}
|
|
|
|
if (m_pageImages.count() == 1) {
|
|
m_pageImages.first().save(pathWithoutSuffix + suffix, format.toUtf8());
|
|
}
|
|
else {
|
|
int part = 1;
|
|
foreach (const QImage &image, m_pageImages) {
|
|
const QString fileName = pathWithoutSuffix + ".part" + QString::number(part);
|
|
image.save(fileName + suffix, format.toUtf8());
|
|
part++;
|
|
}
|
|
}
|
|
}
|
|
|
|
void PageScreen::saveAsDocument(const QString &format)
|
|
{
|
|
const QString suffix = QLatin1Char('.') + format.toLower();
|
|
|
|
QString pathWithoutSuffix = ui->location->text();
|
|
if (pathWithoutSuffix.endsWith(suffix, Qt::CaseInsensitive)) {
|
|
pathWithoutSuffix = pathWithoutSuffix.mid(0, pathWithoutSuffix.length() - suffix.length());
|
|
}
|
|
|
|
QPrinter printer;
|
|
printer.setCreator(BrowserWindow::tr("QupZilla %1 (%2)").arg(Qz::VERSION, Qz::WWWADDRESS));
|
|
printer.setOutputFileName(pathWithoutSuffix + suffix);
|
|
printer.setOutputFormat(QPrinter::PdfFormat);
|
|
printer.setPaperSize(m_pageImages.first().size(), QPrinter::DevicePixel);
|
|
printer.setPageMargins(0, 0, 0, 0, QPrinter::DevicePixel);
|
|
printer.setFullPage(true);
|
|
|
|
QPainter painter;
|
|
painter.begin(&printer);
|
|
|
|
for (int i = 0; i < m_pageImages.size(); ++i) {
|
|
const QImage image = m_pageImages.at(i);
|
|
painter.drawImage(0, 0, image);
|
|
|
|
if (i != m_pageImages.size() - 1) {
|
|
printer.newPage();
|
|
}
|
|
}
|
|
|
|
painter.end();
|
|
}
|
|
|
|
void PageScreen::createThumbnail()
|
|
{
|
|
QWebPage* page = m_view->page();
|
|
|
|
const int heightLimit = 20000;
|
|
const QPoint originalScrollPosition = page->mainFrame()->scrollPosition();
|
|
const QSize originalSize = page->viewportSize();
|
|
const QSize frameSize = page->mainFrame()->contentsSize();
|
|
const int verticalScrollbarSize = page->mainFrame()->scrollBarGeometry(Qt::Vertical).width();
|
|
const int horizontalScrollbarSize = page->mainFrame()->scrollBarGeometry(Qt::Horizontal).height();
|
|
|
|
int yPosition = 0;
|
|
bool canScroll = frameSize.height() > heightLimit;
|
|
|
|
// We will split rendering page into smaller parts to avoid infinite loops
|
|
// or crashes.
|
|
|
|
do {
|
|
int remainingHeight = frameSize.height() - yPosition;
|
|
if (remainingHeight <= 0) {
|
|
break;
|
|
}
|
|
|
|
QSize size(frameSize.width(),
|
|
remainingHeight > heightLimit ? heightLimit : remainingHeight);
|
|
page->setViewportSize(size);
|
|
page->mainFrame()->scroll(0, qMax(0, yPosition - horizontalScrollbarSize));
|
|
|
|
QImage image(page->viewportSize().width() - verticalScrollbarSize,
|
|
page->viewportSize().height() - horizontalScrollbarSize,
|
|
QImage::Format_ARGB32_Premultiplied);
|
|
QPainter painter(&image);
|
|
page->mainFrame()->render(&painter);
|
|
painter.end();
|
|
|
|
m_pageImages.append(image);
|
|
|
|
canScroll = remainingHeight > heightLimit;
|
|
yPosition += size.height();
|
|
}
|
|
while (canScroll);
|
|
|
|
page->setViewportSize(originalSize);
|
|
page->mainFrame()->setScrollBarValue(Qt::Vertical, originalScrollPosition.y());
|
|
page->mainFrame()->setScrollBarValue(Qt::Horizontal, originalScrollPosition.x());
|
|
|
|
m_imageScaling = new QFutureWatcher<QImage>(this);
|
|
m_imageScaling->setFuture(QtConcurrent::run(this, &PageScreen::scaleImage));
|
|
connect(m_imageScaling, SIGNAL(finished()), SLOT(showImage()));
|
|
}
|
|
|
|
QImage PageScreen::scaleImage()
|
|
{
|
|
QVector<QImage> scaledImages;
|
|
int sumHeight = 0;
|
|
|
|
foreach (const QImage &image, m_pageImages) {
|
|
QImage scaled = image.scaledToWidth(450, Qt::SmoothTransformation);
|
|
|
|
scaledImages.append(scaled);
|
|
sumHeight += scaled.height();
|
|
}
|
|
|
|
QImage finalImage(QSize(450, sumHeight), QImage::Format_ARGB32_Premultiplied);
|
|
QPainter painter(&finalImage);
|
|
|
|
int offset = 0;
|
|
foreach (const QImage &image, scaledImages) {
|
|
painter.drawImage(0, offset, image);
|
|
offset += image.height();
|
|
}
|
|
|
|
return finalImage;
|
|
}
|
|
|
|
void PageScreen::showImage()
|
|
{
|
|
delete ui->label->movie();
|
|
|
|
ui->label->setPixmap(QPixmap::fromImage(m_imageScaling->result()));
|
|
}
|
|
|
|
PageScreen::~PageScreen()
|
|
{
|
|
delete ui;
|
|
}
|