HiDPI support for search indicators (#12162)

Requires Qt5
This commit is contained in:
Juergen Spitzmueller 2021-02-28 12:35:29 +01:00
parent 170f06089d
commit 5573825202
2 changed files with 125 additions and 24 deletions

View File

@ -33,6 +33,9 @@
#include <QPainter> #include <QPainter>
#include <QStyle> #include <QStyle>
#include <QPaintEvent> #include <QPaintEvent>
#if QT_VERSION >= 0x050000
#include <QWindow>
#endif
enum { margin = 6 }; enum { margin = 6 };
@ -194,8 +197,13 @@ void FancyLineEdit::updateMargins()
Side realLeft = (leftToRight ? Left : Right); Side realLeft = (leftToRight ? Left : Right);
Side realRight = (leftToRight ? Right : Left); Side realRight = (leftToRight ? Right : Left);
int leftMargin = m_d->m_iconbutton[realLeft]->pixmap().width() + 8; qreal dpr = 1.0;
int rightMargin = m_d->m_iconbutton[realRight]->pixmap().width() + 8; #if QT_VERSION >= 0x050000
// Consider device/pixel ratio (HiDPI)
dpr = devicePixelRatio();
#endif
int leftMargin = (m_d->m_iconbutton[realLeft]->pixmap().width() / dpr ) + 8;
int rightMargin = (m_d->m_iconbutton[realRight]->pixmap().width() / dpr) + 8;
// Note KDE does not reserve space for the highlight color // Note KDE does not reserve space for the highlight color
if (style()->inherits("OxygenStyle")) { if (style()->inherits("OxygenStyle")) {
leftMargin = qMax(24, leftMargin); leftMargin = qMax(24, leftMargin);
@ -334,14 +342,21 @@ IconButton::IconButton(QWidget *parent)
void IconButton::paintEvent(QPaintEvent *) void IconButton::paintEvent(QPaintEvent *)
{ {
QPainter painter(this); qreal dpr = 1.0;
QRect pixmapRect = QRect(0, 0, m_pixmap.width(), m_pixmap.height()); #if QT_VERSION >= 0x050000
// Consider device/pixel ratio (HiDPI)
QWindow * window = this->window()->windowHandle();
dpr = window->devicePixelRatio();
#endif
QRect pixmapRect(QPoint(), m_pixmap.size() / dpr);
pixmapRect.moveCenter(rect().center()); pixmapRect.moveCenter(rect().center());
QPixmap pm = m_pixmap;
QPainter painter(this);
if (m_autoHide) if (m_autoHide)
painter.setOpacity(m_iconOpacity); painter.setOpacity(m_iconOpacity);
painter.drawPixmap(pixmapRect, m_pixmap); painter.drawPixmap(pixmapRect, pm);
} }

View File

@ -27,8 +27,11 @@
#include "GuiKeySymbol.h" #include "GuiKeySymbol.h"
#include "GuiView.h" #include "GuiView.h"
#include "qt_helpers.h"
#include "support/filetools.h"
#include "support/debug.h" #include "support/debug.h"
#include "support/gettext.h" #include "support/gettext.h"
#include "support/FileName.h"
#include "frontends/alert.h" #include "frontends/alert.h"
#include "frontends/Clipboard.h" #include "frontends/Clipboard.h"
@ -38,8 +41,12 @@
#include <QSettings> #include <QSettings>
#include <QShowEvent> #include <QShowEvent>
#include "QSizePolicy" #include "QSizePolicy"
#if QT_VERSION >= 0x050000
#include <QSvgRenderer>
#endif
using namespace std; using namespace std;
using namespace lyx::support;
using lyx::KeySymbol; using lyx::KeySymbol;
@ -221,13 +228,24 @@ void GuiSearchWidget::handleIndicators()
if (wrapCB->isChecked()) if (wrapCB->isChecked())
++pms; ++pms;
bool const dark_mode = guiApp && guiApp->isInDarkMode();
qreal dpr = 1.0;
#if QT_VERSION >= 0x050000
// Consider device/pixel ratio (HiDPI)
if (guiApp && guiApp->currentView())
dpr = guiApp->currentView()->devicePixelRatio();
#endif
QString imagedir = "images/";
QPixmap bpixmap = getPixmap("images/", "search-options", "svgz,png"); QPixmap bpixmap = getPixmap("images/", "search-options", "svgz,png");
QPixmap pm = bpixmap;
if (pms > 0) { if (pms > 0) {
int const gap = 3; int const gap = 3;
QPixmap tpixmap(pms * (bpixmap.width() + gap), bpixmap.height()); QPixmap scaled_pm = QPixmap(bpixmap.size() * dpr);
tpixmap.fill(Qt::transparent); pm = QPixmap(pms * scaled_pm.width() + ((pms - 1) * gap),
QPainter painter(&tpixmap); scaled_pm.height());
pm.fill(Qt::transparent);
QPainter painter(&pm);
int x = 0; int x = 0;
tip = qt_("Active options:"); tip = qt_("Active options:");
@ -235,50 +253,118 @@ void GuiSearchWidget::handleIndicators()
if (caseCB->isChecked()) { if (caseCB->isChecked()) {
tip += "<li>" + qt_("Case sensitive search"); tip += "<li>" + qt_("Case sensitive search");
QPixmap spixmap = getPixmap("images/", "search-case-sensitive", "svgz,png"); QPixmap spixmap = getPixmap("images/", "search-case-sensitive", "svgz,png");
#if QT_VERSION < 0x050000
painter.drawPixmap(x, 0, spixmap); painter.drawPixmap(x, 0, spixmap);
x += spixmap.width() + gap; #else
// With Qt5, we render SVG directly for HiDPI scalability
FileName fname = imageLibFileSearch(imagedir, "search-case-sensitive", "svgz,png");
QString fpath = toqstr(fname.absFileName());
if (!fpath.isEmpty()) {
QSvgRenderer svgRenderer(fpath);
if (svgRenderer.isValid())
svgRenderer.render(&painter, QRectF(0, 0, spixmap.width() * dpr,
spixmap.height() * dpr));
}
#endif
x += (spixmap.width() * dpr) + gap;
} }
if (wordsCB->isChecked()) { if (wordsCB->isChecked()) {
tip += "<li>" + qt_("Whole words only"); tip += "<li>" + qt_("Whole words only");
QPixmap spixmap = getPixmap("images/", "search-whole-words", "svgz,png"); QPixmap spixmap = getPixmap("images/", "search-whole-words", "svgz,png");
#if QT_VERSION < 0x050000
painter.drawPixmap(x, 0, spixmap); painter.drawPixmap(x, 0, spixmap);
x += spixmap.width() + gap; #else
FileName fname = imageLibFileSearch(imagedir, "search-whole-words", "svgz,png");
QString fpath = toqstr(fname.absFileName());
if (!fpath.isEmpty()) {
QSvgRenderer svgRenderer(fpath);
if (svgRenderer.isValid())
svgRenderer.render(&painter, QRectF(x, 0, spixmap.width() * dpr,
spixmap.height() * dpr));
}
#endif
x += (spixmap.width() * dpr) + gap;
} }
if (selectionCB->isChecked()) { if (selectionCB->isChecked()) {
tip += "<li>" + qt_("Search only in selection"); tip += "<li>" + qt_("Search only in selection");
QPixmap spixmap = getPixmap("images/", "search-selection", "svgz,png"); QPixmap spixmap = getPixmap("images/", "search-selection", "svgz,png");
#if QT_VERSION < 0x050000
painter.drawPixmap(x, 0, spixmap); painter.drawPixmap(x, 0, spixmap);
x += spixmap.width() + gap; #else
FileName fname = imageLibFileSearch(imagedir, "search-selection", "svgz,png");
QString fpath = toqstr(fname.absFileName());
if (!fpath.isEmpty()) {
QSvgRenderer svgRenderer(fpath);
if (svgRenderer.isValid())
svgRenderer.render(&painter, QRectF(x, 0, spixmap.width() * dpr,
spixmap.height() * dpr));
}
#endif
x += (spixmap.width() * dpr) + gap;
} }
if (instantSearchCB->isChecked()) { if (instantSearchCB->isChecked()) {
tip += "<li>" + qt_("Search as you type"); tip += "<li>" + qt_("Search as you type");
QPixmap spixmap = getPixmap("images/", "search-instant", "svgz,png"); QPixmap spixmap = getPixmap("images/", "search-instant", "svgz,png");
#if QT_VERSION < 0x050000
painter.drawPixmap(x, 0, spixmap); painter.drawPixmap(x, 0, spixmap);
x += spixmap.width() + gap; #else
FileName fname = imageLibFileSearch(imagedir, "search-instant", "svgz,png");
QString fpath = toqstr(fname.absFileName());
if (!fpath.isEmpty()) {
QSvgRenderer svgRenderer(fpath);
if (svgRenderer.isValid())
svgRenderer.render(&painter, QRectF(x, 0, spixmap.width() * dpr,
spixmap.height() * dpr));
}
#endif
x += (spixmap.width() * dpr) + gap;
} }
if (wrapCB->isChecked()) { if (wrapCB->isChecked()) {
tip += "<li>" + qt_("Wrap search"); tip += "<li>" + qt_("Wrap search");
QPixmap spixmap = getPixmap("images/", "search-wrap", "svgz,png"); QPixmap spixmap = getPixmap("images/", "search-wrap", "svgz,png");
#if QT_VERSION < 0x050000
painter.drawPixmap(x, 0, spixmap); painter.drawPixmap(x, 0, spixmap);
x += spixmap.width() + gap; #else
FileName fname = imageLibFileSearch(imagedir, "search-wrap", "svgz,png");
QString fpath = toqstr(fname.absFileName());
if (!fpath.isEmpty()) {
QSvgRenderer svgRenderer(fpath);
if (svgRenderer.isValid())
svgRenderer.render(&painter, QRectF(x, 0, spixmap.width() * dpr,
spixmap.height() * dpr));
}
#endif
x += (spixmap.width() * dpr) + gap;
} }
tip += "</ul>"; tip += "</ul>";
#if QT_VERSION >= 0x050000
pm.setDevicePixelRatio(dpr);
#endif
painter.end(); painter.end();
if (guiApp && guiApp->isInDarkMode()) {
QImage img = tpixmap.toImage();
img.invertPixels();
tpixmap.convertFromImage(img);
}
findLE_->setButtonPixmap(FancyLineEdit::Right, tpixmap);
} else { } else {
tip = qt_("Click here to change search options"); tip = qt_("Click here to change search options");
if (guiApp && guiApp->isInDarkMode()) { #if QT_VERSION >= 0x050000
QImage img = bpixmap.toImage(); // With Qt5, we render SVG directly for HiDPI scalability
FileName fname = imageLibFileSearch(imagedir, "search-options", "svgz,png");
QString fpath = toqstr(fname.absFileName());
if (!fpath.isEmpty()) {
QSvgRenderer svgRenderer(fpath);
if (svgRenderer.isValid()) {
pm = QPixmap(bpixmap.size() * dpr);
pm.fill(Qt::transparent);
QPainter painter(&pm);
svgRenderer.render(&painter);
pm.setDevicePixelRatio(dpr);
}
}
#endif
}
if (dark_mode) {
QImage img = pm.toImage();
img.invertPixels(); img.invertPixels();
bpixmap.convertFromImage(img); pm.convertFromImage(img);
}
findLE_->setButtonPixmap(FancyLineEdit::Right, bpixmap);
} }
findLE_->setButtonPixmap(FancyLineEdit::Right, pm);
} }
findLE_->setButtonToolTip(FancyLineEdit::Right, tip); findLE_->setButtonToolTip(FancyLineEdit::Right, tip);
} }