Files
PDF4QT/Pdf4QtDiff/differencesdockwidget.cpp
2025-04-27 12:39:06 +02:00

313 lines
10 KiB
C++

// MIT License
//
// Copyright (c) 2018-2025 Jakub Melka and Contributors
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
#include "differencesdockwidget.h"
#include "ui_differencesdockwidget.h"
#include "pdfdiff.h"
#include "pdfwidgetutils.h"
#include <QPixmap>
#include <QPainter>
#include <QTreeWidgetItem>
namespace pdfdiff
{
DifferenceItemDelegate::DifferenceItemDelegate(QObject* parent) :
BaseClass(parent)
{
}
void DifferenceItemDelegate::paint(QPainter* painter,
const QStyleOptionViewItem& option,
const QModelIndex& index) const
{
BaseClass::paint(painter, option, index);
}
QSize DifferenceItemDelegate::sizeHint(const QStyleOptionViewItem& option,
const QModelIndex& index) const
{
QStyleOptionViewItem adjustedOption = option;
if (!option.rect.isValid())
{
// Jakub Melka: Why this? We need to use text wrapping. Unfortunately,
// standard delegate needs correct text rectangle (at least rectangle width),
// for word wrap calculation. So we must manually calculate rectangle width.
// Of course, we cant use visualRect of the tree widget, because of cyclical
// dependence.
const QTreeWidget* treeWidget = qobject_cast<const QTreeWidget*>(option.widget);
int xOffset = treeWidget->columnViewportPosition(index.column());
int height = option.fontMetrics.lineSpacing();
int yOffset = 0;
int width = treeWidget->columnWidth(index.column());
int level = treeWidget->rootIsDecorated() ? 1 : 0;
QModelIndex currentIndex = index.parent();
while (currentIndex.isValid())
{
++level;
currentIndex = currentIndex.parent();
}
xOffset += level * treeWidget->indentation();
adjustedOption.rect = QRect(xOffset, yOffset, width - xOffset, height);
}
return BaseClass::sizeHint(adjustedOption, index);
}
DifferencesDockWidget::DifferencesDockWidget(QWidget* parent,
pdf::PDFDiffResult* unfilteredDiffResult,
pdf::PDFDiffResult* diffResult,
pdf::PDFDiffResultNavigator* diffNavigator,
Settings* settings) :
QDockWidget(parent),
ui(new Ui::DifferencesDockWidget),
m_unfilteredDiffResult(unfilteredDiffResult),
m_diffResult(diffResult),
m_diffNavigator(diffNavigator),
m_settings(settings),
m_disableChangeSelectedResultIndex(false)
{
ui->setupUi(this);
ui->differencesTreeWidget->setItemDelegate(new DifferenceItemDelegate(this));
ui->differencesTreeWidget->setIconSize(pdf::PDFWidgetUtils::scaleDPI(ui->differencesTreeWidget, QSize(16, 16)));
connect(diffNavigator, &pdf::PDFDiffResultNavigator::selectionChanged, this, &DifferencesDockWidget::onSelectionChanged);
connect(ui->differencesTreeWidget, &QTreeWidget::currentItemChanged, this, &DifferencesDockWidget::onCurrentItemChanged);
toggleViewAction()->setText(tr("Differen&ces"));
setMinimumWidth(pdf::PDFWidgetUtils::scaleDPI_x(this, 120));
}
DifferencesDockWidget::~DifferencesDockWidget()
{
delete ui;
}
QColor DifferencesDockWidget::getColorForIndex(size_t index) const
{
QColor color;
const size_t resultIndex = index;
if (m_diffResult->isReplaceDifference(resultIndex))
{
color = m_settings->colorReplaced;
}
else if (m_diffResult->isRemoveDifference(resultIndex))
{
color = m_settings->colorRemoved;
}
else if (m_diffResult->isAddDifference(resultIndex))
{
color = m_settings->colorAdded;
}
else if (m_diffResult->isPageMoveDifference(resultIndex))
{
color = m_settings->colorPageMove;
}
return color;
}
QModelIndex DifferencesDockWidget::findResultIndex(size_t index) const
{
QAbstractItemModel* model = ui->differencesTreeWidget->model();
const int count = ui->differencesTreeWidget->topLevelItemCount();
for (int i = 0; i < count; ++i)
{
QModelIndex parentIndex = model->index(i, 0);
if (parentIndex.isValid())
{
const int childCount = model->rowCount(parentIndex);
for (int j = 0; j < childCount; ++j)
{
QModelIndex childIndex = model->index(j, 0, parentIndex);
QVariant childUserData = childIndex.data(Qt::UserRole);
if (childUserData.isValid())
{
if (childUserData.toULongLong() == index)
{
return childIndex;
}
}
}
}
}
return QModelIndex();
}
void DifferencesDockWidget::update()
{
ui->differencesTreeWidget->clear();
QList<QTreeWidgetItem*> topItems;
QLocale locale;
if (m_diffResult && !m_diffResult->isSame())
{
std::map<QRgb, QIcon> icons;
auto getIcon = [this, &icons](QColor color)
{
auto it = icons.find(color.rgb());
if (it == icons.end())
{
QPixmap pixmap(ui->differencesTreeWidget->iconSize());
pixmap.fill(Qt::transparent);
QPainter painter(&pixmap);
painter.setPen(Qt::NoPen);
painter.setRenderHint(QPainter::Antialiasing);
painter.setBrush(QBrush(color));
painter.drawEllipse(0, 0, pixmap.width(), pixmap.height());
painter.end();
QIcon icon(pixmap);
it = icons.insert(std::make_pair(color.rgb(), std::move(icon))).first;
}
return it->second;
};
const size_t differenceCount = m_diffResult->getDifferencesCount();
const size_t unfilteredDifferenceCount = m_unfilteredDiffResult->getDifferencesCount();
const size_t hiddenDifferences = unfilteredDifferenceCount - differenceCount;
if (hiddenDifferences > 0)
{
ui->infoTextLabel->setText(tr("%1 Differences (+%2 hidden)").arg(differenceCount).arg(hiddenDifferences));
}
else
{
ui->infoTextLabel->setText(tr("%1 Differences").arg(differenceCount));
}
pdf::PDFInteger lastLeftPageIndex = -1;
pdf::PDFInteger lastRightPageIndex = -1;
for (size_t i = 0; i < differenceCount; ++i)
{
pdf::PDFInteger pageIndexLeft = m_diffResult->getLeftPage(i);
pdf::PDFInteger pageIndexRight = m_diffResult->getRightPage(i);
if (lastLeftPageIndex != pageIndexLeft ||
lastRightPageIndex != pageIndexRight ||
topItems.empty())
{
// Create new top level item
QStringList captionParts;
captionParts << QString("#%1:").arg(topItems.size() + 1);
if (pageIndexLeft == pageIndexRight)
{
captionParts << tr("Page %1").arg(locale.toString(pageIndexLeft + 1));
}
else
{
if (pageIndexLeft != -1)
{
captionParts << tr("Left %1").arg(locale.toString(pageIndexLeft + 1));
}
if (pageIndexRight != -1)
{
captionParts << tr("Right %1").arg(locale.toString(pageIndexRight + 1));
}
}
QTreeWidgetItem* item = new QTreeWidgetItem(QStringList() << captionParts.join(" "));
topItems << item;
lastLeftPageIndex = pageIndexLeft;
lastRightPageIndex = pageIndexRight;
}
Q_ASSERT(!topItems.isEmpty());
QTreeWidgetItem* parent = topItems.back();
QTreeWidgetItem* item = new QTreeWidgetItem(parent, QStringList() << m_diffResult->getMessage(i));
item->setData(0, Qt::UserRole, QVariant(static_cast<qint64>(i)));
QColor color = getColorForIndex(i);
if (color.isValid())
{
item->setData(0, Qt::DecorationRole, getIcon(color));
}
}
}
else
{
ui->infoTextLabel->setText(tr("No Differences Found!"));
}
ui->differencesTreeWidget->addTopLevelItems(topItems);
ui->differencesTreeWidget->expandAll();
}
void DifferencesDockWidget::onSelectionChanged(size_t currentIndex)
{
if (m_disableChangeSelectedResultIndex)
{
return;
}
pdf::PDFTemporaryValueChange guard(&m_disableChangeSelectedResultIndex, true);
QModelIndex index = findResultIndex(currentIndex);
if (index.isValid())
{
ui->differencesTreeWidget->scrollTo(index);
ui->differencesTreeWidget->setCurrentIndex(index);
}
}
void DifferencesDockWidget::onCurrentItemChanged(QTreeWidgetItem* current, QTreeWidgetItem* previous)
{
Q_UNUSED(previous);
if (m_disableChangeSelectedResultIndex || !current)
{
return;
}
pdf::PDFTemporaryValueChange guard(&m_disableChangeSelectedResultIndex, true);
QVariant childUserData = current->data(0, Qt::UserRole);
if (childUserData.isValid())
{
m_diffNavigator->select(childUserData.toULongLong());
}
}
} // namespace pdfdiff