mirror of
https://github.com/JakubMelka/PDF4QT.git
synced 2025-01-23 22:00:48 +01:00
289 lines
7.8 KiB
C++
289 lines
7.8 KiB
C++
// Copyright (C) 2019-2022 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 "pdfwidgetutils.h"
|
|
#include "pdfcolorconvertor.h"
|
|
|
|
#include <map>
|
|
#include <set>
|
|
|
|
#include <QMenu>
|
|
#include <QAction>
|
|
#include <QDialog>
|
|
#include <QLayout>
|
|
#include <QPainter>
|
|
#include <QPixmap>
|
|
#include <QGroupBox>
|
|
#include <QMessageBox>
|
|
#include <QApplication>
|
|
|
|
#include "pdfdbgheap.h"
|
|
|
|
#ifdef Q_OS_MAC
|
|
int qt_default_dpi_x() { return 72; }
|
|
int qt_default_dpi_y() { return 72; }
|
|
#else
|
|
int qt_default_dpi_x() { return 96; }
|
|
int qt_default_dpi_y() { return 96; }
|
|
#endif
|
|
|
|
namespace pdf
|
|
{
|
|
|
|
static constexpr bool isScalingNeeded()
|
|
{
|
|
return QT_VERSION_MAJOR < 6;
|
|
}
|
|
|
|
int PDFWidgetUtils::getPixelSize(const QPaintDevice* device, pdf::PDFReal sizeMM)
|
|
{
|
|
const int width = device->width();
|
|
const int height = device->height();
|
|
|
|
if (width > height)
|
|
{
|
|
return PDFReal(width) * sizeMM / PDFReal(device->widthMM());
|
|
}
|
|
else
|
|
{
|
|
return PDFReal(height) * sizeMM / PDFReal(device->heightMM());
|
|
}
|
|
}
|
|
|
|
int PDFWidgetUtils::scaleDPI_x(const QPaintDevice* device, int unscaledSize)
|
|
{
|
|
if constexpr (isScalingNeeded())
|
|
{
|
|
const double logicalDPI_x = device->logicalDpiX();
|
|
const double defaultDPI_x = qt_default_dpi_x();
|
|
return (logicalDPI_x / defaultDPI_x) * unscaledSize;
|
|
}
|
|
else
|
|
{
|
|
return unscaledSize;
|
|
}
|
|
}
|
|
|
|
int PDFWidgetUtils::scaleDPI_y(const QPaintDevice* device, int unscaledSize)
|
|
{
|
|
if constexpr (isScalingNeeded())
|
|
{
|
|
const double logicalDPI_y = device->logicalDpiY();
|
|
const double defaultDPI_y = qt_default_dpi_y();
|
|
return (logicalDPI_y / defaultDPI_y) * unscaledSize;
|
|
}
|
|
else
|
|
{
|
|
return unscaledSize;
|
|
}
|
|
}
|
|
|
|
PDFReal PDFWidgetUtils::scaleDPI_x(const QPaintDevice* device, PDFReal unscaledSize)
|
|
{
|
|
if constexpr (isScalingNeeded())
|
|
{
|
|
const double logicalDPI_x = device->logicalDpiX();
|
|
const double defaultDPI_x = qt_default_dpi_x();
|
|
return (logicalDPI_x / defaultDPI_x) * unscaledSize;
|
|
}
|
|
else
|
|
{
|
|
return unscaledSize;
|
|
}
|
|
}
|
|
|
|
void PDFWidgetUtils::scaleWidget(QWidget* widget, QSize unscaledSize)
|
|
{
|
|
if constexpr (!isScalingNeeded())
|
|
{
|
|
return;
|
|
}
|
|
|
|
const double logicalDPI_x = widget->logicalDpiX();
|
|
const double logicalDPI_y = widget->logicalDpiY();
|
|
const double defaultDPI_x = qt_default_dpi_x();
|
|
const double defaultDPI_y = qt_default_dpi_y();
|
|
|
|
const int width = (logicalDPI_x / defaultDPI_x) * unscaledSize.width();
|
|
const int height = (logicalDPI_y / defaultDPI_y) * unscaledSize.height();
|
|
|
|
widget->resize(width, height);
|
|
}
|
|
|
|
QSize PDFWidgetUtils::scaleDPI(const QPaintDevice* widget, QSize unscaledSize)
|
|
{
|
|
if constexpr (isScalingNeeded())
|
|
{
|
|
const double logicalDPI_x = widget->logicalDpiX();
|
|
const double logicalDPI_y = widget->logicalDpiY();
|
|
const double defaultDPI_x = qt_default_dpi_x();
|
|
const double defaultDPI_y = qt_default_dpi_y();
|
|
|
|
const int width = (logicalDPI_x / defaultDPI_x) * unscaledSize.width();
|
|
const int height = (logicalDPI_y / defaultDPI_y) * unscaledSize.height();
|
|
|
|
return QSize(width, height);
|
|
}
|
|
else
|
|
{
|
|
return unscaledSize;
|
|
}
|
|
}
|
|
|
|
void PDFWidgetUtils::style(QWidget* widget)
|
|
{
|
|
const int dialogMarginX = scaleDPI_x(widget, 12);
|
|
const int dialogMarginY = scaleDPI_y(widget, 12);
|
|
const int dialogSpacing = scaleDPI_y(widget, 12);
|
|
|
|
const int groupBoxMarginX = scaleDPI_x(widget, 20);
|
|
const int groupBoxMarginY = scaleDPI_y(widget, 20);
|
|
const int groupBoxSpacing = scaleDPI_y(widget, 12);
|
|
|
|
QList<QWidget*> childWidgets = widget->findChildren<QWidget*>();
|
|
childWidgets.append(widget);
|
|
|
|
for (QWidget* childWidget : childWidgets)
|
|
{
|
|
if (qobject_cast<QGroupBox*>(childWidget))
|
|
{
|
|
childWidget->layout()->setContentsMargins(groupBoxMarginX, groupBoxMarginY, groupBoxMarginX, groupBoxMarginY);
|
|
childWidget->layout()->setSpacing(groupBoxSpacing);
|
|
}
|
|
else if (qobject_cast<QDialog*>(childWidget))
|
|
{
|
|
childWidget->layout()->setContentsMargins(dialogMarginX, dialogMarginY, dialogMarginX, dialogMarginY);
|
|
childWidget->layout()->setSpacing(dialogSpacing);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool PDFWidgetUtils::isDarkTheme()
|
|
{
|
|
QPalette palette = QApplication::palette();
|
|
QColor backgroundColor = palette.color(QPalette::Window);
|
|
QColor textColor = palette.color(QPalette::WindowText);
|
|
|
|
return backgroundColor.lightness() < textColor.lightness();
|
|
}
|
|
|
|
void PDFWidgetUtils::convertActionForDarkTheme(QAction* action, QSize iconSize, qreal devicePixelRatioF)
|
|
{
|
|
if (!action)
|
|
{
|
|
return;
|
|
}
|
|
|
|
QIcon icon = action->icon();
|
|
if (!icon.isNull())
|
|
{
|
|
icon = pdf::PDFWidgetUtils::convertIconForDarkTheme(icon, iconSize, devicePixelRatioF);
|
|
action->setIcon(icon);
|
|
}
|
|
}
|
|
|
|
QIcon PDFWidgetUtils::convertIconForDarkTheme(QIcon icon, QSize iconSize, qreal devicePixelRatioF)
|
|
{
|
|
QPixmap pixmap(iconSize * devicePixelRatioF);
|
|
pixmap.setDevicePixelRatio(devicePixelRatioF);
|
|
pixmap.fill(Qt::transparent);
|
|
|
|
QPainter painter(&pixmap);
|
|
icon.paint(&painter, QRect(0, 0, iconSize.width(), iconSize.height()));
|
|
painter.end();
|
|
|
|
PDFColorConvertor convertor;
|
|
convertor.setMode(PDFColorConvertor::Mode::InvertedColors);
|
|
|
|
QImage image = pixmap.toImage();
|
|
image = convertor.convert(image);
|
|
pixmap = QPixmap::fromImage(image);
|
|
|
|
return QIcon(pixmap);
|
|
}
|
|
|
|
void PDFWidgetUtils::checkMenuAccessibility(QWidget* widget)
|
|
{
|
|
QList<QMenu*> menus = widget->findChildren<QMenu*>();
|
|
|
|
for (QMenu* menu : menus)
|
|
{
|
|
checkMenuAccessibility(menu);
|
|
}
|
|
}
|
|
|
|
void PDFWidgetUtils::checkMenuAccessibility(QMenu* menu)
|
|
{
|
|
QStringList actionsDuplicities;
|
|
QStringList actionsWithNoAmpersands;
|
|
std::map<QChar, std::set<QAction*>> actions;
|
|
|
|
for (QAction* action : menu->actions())
|
|
{
|
|
if (action->isSeparator())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (QMenu* submenu = action->menu())
|
|
{
|
|
checkMenuAccessibility(submenu);
|
|
}
|
|
|
|
QString text = action->text();
|
|
int i = text.indexOf(QChar('&'));
|
|
|
|
if (i == -1)
|
|
{
|
|
actionsWithNoAmpersands << text;
|
|
}
|
|
else
|
|
{
|
|
QString accesibilityChar = text.mid(i + 1, 1);
|
|
if (accesibilityChar.size() == 1)
|
|
{
|
|
actions[accesibilityChar.front().toUpper()].insert(action);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const auto& actionItem : actions)
|
|
{
|
|
if (actionItem.second.size() > 1)
|
|
{
|
|
QStringList actionTexts;
|
|
for (QAction* action : actionItem.second)
|
|
{
|
|
actionTexts << action->text();
|
|
}
|
|
|
|
actionsDuplicities << QString("'%1': Duplicities = '%2'").arg(QString(actionItem.first), actionTexts.join("', '"));
|
|
}
|
|
}
|
|
|
|
QStringList errors;
|
|
errors.append(std::move(actionsDuplicities));
|
|
errors.append(std::move(actionsWithNoAmpersands));
|
|
|
|
if (!errors.isEmpty())
|
|
{
|
|
QMessageBox::warning(nullptr, "Accesibility Check", errors.join("<br>"));
|
|
}
|
|
}
|
|
|
|
} // namespace pdf
|