// 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 .
#include "pageitemdelegate.h"
#include "pageitemmodel.h"
#include "pdfwidgetutils.h"
#include "pdfpainterutils.h"
#include "pdfrenderer.h"
#include "pdfcompiler.h"
#include "pdfconstants.h"
#include
#include
namespace pdfdocpage
{
PageItemDelegate::PageItemDelegate(PageItemModel* model, QObject* parent) :
BaseClass(parent),
m_model(model),
m_rasterizer(nullptr)
{
m_rasterizer = new pdf::PDFRasterizer(this);
QSurfaceFormat format;
format.setSamples(16);
m_rasterizer->reset(false, format);
}
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.setNamedColor(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(&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