mirror of
https://github.com/JakubMelka/PDF4QT.git
synced 2025-01-16 10:32:29 +01:00
241 lines
9.3 KiB
C++
241 lines
9.3 KiB
C++
// Copyright (C) 2021 Jakub Melka
|
|
//
|
|
// This file is part of PDF4QT.
|
|
//
|
|
// PDF4QT is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// with the written consent of the copyright owner, any later version.
|
|
//
|
|
// PDF4QT 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 Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with PDF4QT. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
#include "pageitemdelegate.h"
|
|
#include "pageitemmodel.h"
|
|
#include "pdfwidgetutils.h"
|
|
#include "pdfpainterutils.h"
|
|
#include "pdfrenderer.h"
|
|
#include "pdfcompiler.h"
|
|
#include "pdfconstants.h"
|
|
|
|
#include <QPainter>
|
|
#include <QPixmapCache>
|
|
|
|
namespace pdfdocpage
|
|
{
|
|
|
|
PageItemDelegate::PageItemDelegate(PageItemModel* model, QObject* parent) :
|
|
BaseClass(parent),
|
|
m_model(model),
|
|
m_rasterizer(nullptr)
|
|
{
|
|
m_rasterizer = new pdf::PDFRasterizer(this);
|
|
m_rasterizer->reset(pdf::RendererEngine::Blend2D_SingleThread);
|
|
}
|
|
|
|
PageItemDelegate::~PageItemDelegate()
|
|
{
|
|
|
|
}
|
|
|
|
void PageItemDelegate::paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const
|
|
{
|
|
const PageGroupItem* item = m_model->getItem(index);
|
|
|
|
if (!item)
|
|
{
|
|
return;
|
|
}
|
|
|
|
QRect rect = option.rect;
|
|
|
|
m_dpiScaleRatio = option.widget->devicePixelRatioF();
|
|
QSize scaledSize = pdf::PDFWidgetUtils::scaleDPI(option.widget, m_pageImageSize);
|
|
int verticalSpacing = pdf::PDFWidgetUtils::scaleDPI_y(option.widget, getVerticalSpacing());
|
|
int horizontalSpacing = pdf::PDFWidgetUtils::scaleDPI_x(option.widget, getHorizontalSpacing());
|
|
|
|
QRect pageBoundingRect = QRect(QPoint(rect.left() + (rect.width() - scaledSize.width()) / 2, rect.top() + verticalSpacing), scaledSize);
|
|
|
|
// Draw page preview
|
|
if (!item->groups.empty())
|
|
{
|
|
const PageGroupItem::GroupItem& groupItem = item->groups.front();
|
|
QSizeF rotatedPageSize = pdf::PDFPage::getRotatedBox(QRectF(QPointF(0, 0), groupItem.rotatedPageDimensionsMM), groupItem.pageAdditionalRotation).size();
|
|
QSize pageImageSize = rotatedPageSize.scaled(pageBoundingRect.size(), Qt::KeepAspectRatio).toSize();
|
|
QRect pageImageRect(pageBoundingRect.topLeft() + QPoint((pageBoundingRect.width() - pageImageSize.width()) / 2, (pageBoundingRect.height() - pageImageSize.height()) / 2), pageImageSize);
|
|
|
|
painter->setBrush(Qt::white);
|
|
painter->drawRect(pageImageRect);
|
|
|
|
QPixmap pageImagePixmap = getPageImagePixmap(item, pageImageRect);
|
|
if (!pageImagePixmap.isNull())
|
|
{
|
|
painter->drawPixmap(pageImageRect, pageImagePixmap);
|
|
}
|
|
|
|
painter->setPen(QPen(Qt::black));
|
|
painter->setBrush(Qt::NoBrush);
|
|
painter->drawRect(pageImageRect);
|
|
}
|
|
|
|
int textOffset = pageBoundingRect.bottom() + verticalSpacing;
|
|
QRect textRect = option.rect;
|
|
textRect.setTop(textOffset);
|
|
textRect.setHeight(option.fontMetrics.lineSpacing());
|
|
painter->drawText(textRect, Qt::AlignCenter | Qt::TextSingleLine, item->groupName);
|
|
textRect.translate(0, textRect.height());
|
|
painter->drawText(textRect, Qt::AlignCenter | Qt::TextSingleLine, item->pagesCaption);
|
|
|
|
if (option.state.testFlag(QStyle::State_Selected))
|
|
{
|
|
QColor selectedColor = option.palette.color(QPalette::Active, QPalette::Highlight);
|
|
selectedColor.setAlphaF(0.3f);
|
|
painter->fillRect(rect, selectedColor);
|
|
}
|
|
|
|
QPoint tagPoint = rect.topRight() + QPoint(-horizontalSpacing, verticalSpacing);
|
|
for (const QString& tag : item->tags)
|
|
{
|
|
QStringList splitted = tag.split('@', Qt::KeepEmptyParts);
|
|
if (splitted.size() != 2 || splitted.back().isEmpty())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
QColor color;
|
|
color.fromString(splitted.front());
|
|
QRect bubbleRect = pdf::PDFPainterHelper::drawBubble(painter, tagPoint, color, splitted.back(), Qt::AlignLeft | Qt::AlignBottom);
|
|
tagPoint.ry() += bubbleRect.height() + verticalSpacing;
|
|
}
|
|
}
|
|
|
|
QSize PageItemDelegate::sizeHint(const QStyleOptionViewItem& option, const QModelIndex& index) const
|
|
{
|
|
Q_UNUSED(index);
|
|
|
|
QSize scaledSize = pdf::PDFWidgetUtils::scaleDPI(option.widget, m_pageImageSize);
|
|
int height = scaledSize.height() + option.fontMetrics.lineSpacing() * 2 + 2 * pdf::PDFWidgetUtils::scaleDPI_y(option.widget, getVerticalSpacing());
|
|
int width = qMax(pdf::PDFWidgetUtils::scaleDPI_x(option.widget, 40), scaledSize.width() + 2 * pdf::PDFWidgetUtils::scaleDPI_x(option.widget, getHorizontalSpacing()));
|
|
return QSize(width, height);
|
|
}
|
|
|
|
QSize PageItemDelegate::getPageImageSize() const
|
|
{
|
|
return m_pageImageSize;
|
|
}
|
|
|
|
void PageItemDelegate::setPageImageSize(QSize pageImageSize)
|
|
{
|
|
if (m_pageImageSize != pageImageSize)
|
|
{
|
|
m_pageImageSize = pageImageSize;
|
|
Q_EMIT sizeHintChanged(QModelIndex());
|
|
}
|
|
}
|
|
|
|
QPixmap PageItemDelegate::getPageImagePixmap(const PageGroupItem* item, QRect rect) const
|
|
{
|
|
QPixmap pixmap;
|
|
|
|
Q_ASSERT(item);
|
|
if (item->groups.empty())
|
|
{
|
|
return pixmap;
|
|
}
|
|
|
|
const PageGroupItem::GroupItem& groupItem = item->groups.front();
|
|
if (groupItem.pageType == PT_Empty)
|
|
{
|
|
return pixmap;
|
|
}
|
|
|
|
// Jakub Melka: generate key and see, if pixmap is not cached
|
|
QString key = QString("%1#%2#%3#%4#%5@%6x%7").arg(groupItem.documentIndex).arg(groupItem.imageIndex).arg(int(groupItem.pageAdditionalRotation)).arg(groupItem.pageIndex).arg(groupItem.pageType).arg(rect.width()).arg(rect.height());
|
|
|
|
if (!QPixmapCache::find(key, &pixmap))
|
|
{
|
|
// We must draw the pixmap
|
|
pixmap = QPixmap(rect.width(), rect.height());
|
|
pixmap.fill(Qt::transparent);
|
|
|
|
switch (groupItem.pageType)
|
|
{
|
|
case pdfdocpage::PT_DocumentPage:
|
|
{
|
|
const auto& documents = m_model->getDocuments();
|
|
auto it = documents.find(groupItem.documentIndex);
|
|
if (it != documents.cend())
|
|
{
|
|
const pdf::PDFDocument& document = it->second.document;
|
|
const pdf::PDFInteger pageIndex = groupItem.pageIndex - 1;
|
|
if (pageIndex >= 0 && pageIndex < pdf::PDFInteger(document.getCatalog()->getPageCount()))
|
|
{
|
|
const pdf::PDFPage* page = document.getCatalog()->getPage(pageIndex);
|
|
Q_ASSERT(page);
|
|
|
|
pdf::PDFPrecompiledPage compiledPage;
|
|
pdf::PDFFontCache fontCache(pdf::DEFAULT_FONT_CACHE_LIMIT, pdf::DEFAULT_REALIZED_FONT_CACHE_LIMIT);
|
|
pdf::PDFCMSManager cmsManager(nullptr);
|
|
pdf::PDFOptionalContentActivity optionalContentActivity(&document, pdf::OCUsage::View, nullptr);
|
|
|
|
fontCache.setDocument(pdf::PDFModifiedDocument(const_cast<pdf::PDFDocument*>(&document), &optionalContentActivity));
|
|
cmsManager.setDocument(&document);
|
|
|
|
pdf::PDFCMSPointer cms = cmsManager.getCurrentCMS();
|
|
pdf::PDFRenderer renderer(&document, &fontCache, cms.data(), &optionalContentActivity, pdf::PDFRenderer::getDefaultFeatures(), pdf::PDFMeshQualitySettings());
|
|
renderer.compile(&compiledPage, pageIndex);
|
|
|
|
QSize imageSize = rect.size() * m_dpiScaleRatio;
|
|
QImage pageImage = m_rasterizer->render(pageIndex, page, &compiledPage, imageSize, pdf::PDFRenderer::getDefaultFeatures(), nullptr, groupItem.pageAdditionalRotation);
|
|
pixmap = QPixmap::fromImage(qMove(pageImage));
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case pdfdocpage::PT_Image:
|
|
{
|
|
const auto& images = m_model->getImages();
|
|
auto it = images.find(groupItem.imageIndex);
|
|
if (it != images.cend())
|
|
{
|
|
const QImage& image = it->second.image;
|
|
if (!image.isNull())
|
|
{
|
|
QRect drawRect(QPoint(0, 0), rect.size());
|
|
QRect mediaBox(QPoint(0, 0), image.size());
|
|
QRectF rotatedMediaBox = pdf::PDFPage::getRotatedBox(mediaBox, groupItem.pageAdditionalRotation);
|
|
QTransform matrix = pdf::PDFRenderer::createMediaBoxToDevicePointMatrix(rotatedMediaBox, drawRect, groupItem.pageAdditionalRotation);
|
|
|
|
QPainter painter(&pixmap);
|
|
painter.setWorldTransform(QTransform(matrix));
|
|
painter.translate(0, image.height());
|
|
painter.scale(1.0, -1.0);
|
|
painter.drawImage(0, 0, image);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case pdfdocpage::PT_Empty:
|
|
Q_ASSERT(false);
|
|
break;
|
|
|
|
default:
|
|
Q_ASSERT(false);
|
|
break;
|
|
}
|
|
|
|
QPixmapCache::insert(key, pixmap);
|
|
}
|
|
|
|
return pixmap;
|
|
}
|
|
|
|
} // namespace pdfdocpage
|