PDF4QT/Pdf4QtLibWidgets/sources/pdfwidgetutils.cpp

371 lines
11 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 <optional>
#include <QMenu>
#include <QAction>
#include <QDialog>
#include <QLayout>
#include <QPainter>
#include <QPixmap>
#include <QGroupBox>
#include <QMessageBox>
#include <QApplication>
#include <QStyleHints>
#include <QStyleFactory>
#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);
}
}
}
void PDFWidgetUtils::setDarkTheme(bool isLightTheme, bool isDarkTheme)
{
if (isLightTheme)
{
QApplication::styleHints()->setColorScheme(Qt::ColorScheme::Light);
}
if (isDarkTheme)
{
QApplication::styleHints()->setColorScheme(Qt::ColorScheme::Dark);
}
if (PDFWidgetUtils::isDarkTheme())
{
QPalette darkPalette = QApplication::palette();
#ifdef Q_OS_WIN
QApplication::setStyle(QStyleFactory::create("Fusion"));
// Basic colors
darkPalette.setColor(QPalette::WindowText, QColor(220, 220, 220));
darkPalette.setColor(QPalette::Button, QColor(53, 53, 53));
darkPalette.setColor(QPalette::Light, QColor(70, 70, 70));
darkPalette.setColor(QPalette::Midlight, QColor(60, 60, 60));
darkPalette.setColor(QPalette::Dark, QColor(35, 35, 35));
darkPalette.setColor(QPalette::Mid, QColor(40, 40, 40));
// Texts
darkPalette.setColor(QPalette::Text, QColor(220, 220, 220));
darkPalette.setColor(QPalette::BrightText, Qt::red);
darkPalette.setColor(QPalette::ButtonText, QColor(220, 220, 220));
// Background
darkPalette.setColor(QPalette::Base, QColor(42, 42, 42));
darkPalette.setColor(QPalette::Window, QColor(53, 53, 53));
darkPalette.setColor(QPalette::Shadow, QColor(20, 20, 20));
// Highlight
darkPalette.setColor(QPalette::Highlight, QColor(42, 130, 218)); // Barva výběru
darkPalette.setColor(QPalette::HighlightedText, Qt::black); // Text ve výběru
// Links
darkPalette.setColor(QPalette::Link, QColor(42, 130, 218)); // Odkazy
darkPalette.setColor(QPalette::LinkVisited, QColor(100, 100, 150)); // Navštívené odkazy
// Alternative background
darkPalette.setColor(QPalette::AlternateBase, QColor(66, 66, 66)); // Např. střídavé řádky v tabulce
// Special roles
darkPalette.setColor(QPalette::NoRole, QColor(0, 0, 0, 0));
// Help
darkPalette.setColor(QPalette::ToolTipBase, QColor(53, 53, 53));
darkPalette.setColor(QPalette::ToolTipText, QColor(220, 220, 220));
for (int i = 0; i < QPalette::NColorRoles; ++i)
{
QColor disabledColor = darkPalette.color(QPalette::Disabled, static_cast<QPalette::ColorRole>(i));
disabledColor = disabledColor.darker(200);
darkPalette.setColor(QPalette::Disabled, static_cast<QPalette::ColorRole>(i), disabledColor);
QColor currentColor = darkPalette.color(QPalette::Current, static_cast<QPalette::ColorRole>(i));
currentColor = currentColor.lighter(150);
darkPalette.setColor(QPalette::Current, static_cast<QPalette::ColorRole>(i), currentColor);
}
#endif
// Placeholder text (Qt 5.12+)
darkPalette.setColor(QPalette::PlaceholderText, QColor(150, 150, 150));
// Accents (Qt 6.5+)
darkPalette.setColor(QPalette::Accent, QColor(42, 130, 218));
QApplication::setPalette(darkPalette);
}
}
bool PDFWidgetUtils::isDarkTheme()
{
Qt::ColorScheme colorScheme = QApplication::styleHints()->colorScheme();
return colorScheme == Qt::ColorScheme::Dark;
}
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 (text.isEmpty())
{
continue;
}
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