Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qtpropertybrowserutils.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
5
6#include <QtWidgets/qapplication.h>
7#include <QtWidgets/qboxlayout.h>
8#include <QtWidgets/qcheckbox.h>
9#include <QtWidgets/qlineedit.h>
10#include <QtWidgets/qmenu.h>
11
12#include <QtGui/qevent.h>
13#include <QtGui/qpainter.h>
14#include <QtGui/qpainterstateguard.h>
15
16#include <QtCore/qlocale.h>
17#include <QtCore/qoperatingsystemversion.h>
18
19#include <utility>
20
21QT_BEGIN_NAMESPACE
22
23using namespace Qt::StringLiterals;
24
25// Make sure icons are removed as soon as QApplication is destroyed, otherwise,
26// handles are leaked on X11.
31
32using ShapeNames = QHash<Qt::CursorShape, const char *>;
33
34static const ShapeNames &shapeNames()
35{
36 static const ShapeNames result =
37 {{Qt::ArrowCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Arrow")},
38 {Qt::UpArrowCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Up Arrow")},
39 {Qt::CrossCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Cross")},
40 {Qt::WaitCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Wait")},
41 {Qt::IBeamCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "IBeam")},
42 {Qt::SizeVerCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Size Vertical")},
43 {Qt::SizeHorCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Size Horizontal")},
44 {Qt::SizeFDiagCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Size Backslash")},
45 {Qt::SizeBDiagCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Size Slash")},
46 {Qt::SizeAllCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Size All")},
47 {Qt::BlankCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Blank")},
48 {Qt::SplitVCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Split Vertical")},
49 {Qt::SplitHCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Split Horizontal")},
50 {Qt::PointingHandCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Pointing Hand")},
51 {Qt::ForbiddenCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Forbidden")},
52 {Qt::OpenHandCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Open Hand")},
53 {Qt::ClosedHandCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Closed Hand")},
54 {Qt::WhatsThisCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "What's This")},
55 {Qt::BusyCursor, QT_TRANSLATE_NOOP("QtCursorDatabase", "Busy")}};
56 return result;
57}
58
59using CursorResource = std::pair<Qt::CursorShape, QLatin1StringView>;
60
61static constexpr CursorResource cursorResources[] = {
62 {Qt::ArrowCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-arrow.png"_L1},
63 {Qt::UpArrowCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-uparrow.png"_L1},
64 {Qt::CrossCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-cross.png"_L1},
65 {Qt::WaitCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-wait.png"_L1},
66 {Qt::IBeamCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-ibeam.png"_L1},
67 {Qt::SizeVerCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-sizev.png"_L1},
68 {Qt::SizeHorCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-sizeh.png"_L1},
69 {Qt::SizeFDiagCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-sizef.png"_L1},
70 {Qt::SizeBDiagCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-sizeb.png"_L1},
71 {Qt::SizeAllCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-sizeall.png"_L1},
72 {Qt::SplitVCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-vsplit.png"_L1},
73 {Qt::SplitHCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-hsplit.png"_L1},
74 {Qt::PointingHandCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-hand.png"_L1},
75 {Qt::ForbiddenCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-forbidden.png"_L1},
76 {Qt::OpenHandCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-openhand.png"_L1},
77 {Qt::ClosedHandCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-closedhand.png"_L1},
78 {Qt::WhatsThisCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-whatsthis.png"_L1},
79 {Qt::BusyCursor, ":/qt-project.org/qtpropertybrowser/images/cursor-busy.png"_L1}
80};
81
82// A Pixmap icon engine for resource pixmaps that do not have a DPR set.
83// It sets the DPR on the pixmaps and either scales it down or centers it.
85{
86public:
88
90
92 {
93 return new QtPixmapIconEngine(m_pixmap);
94 }
95
97 {
100
101 const auto dpr = painter->device()->devicePixelRatioF();
103 QSize dipSize =
105 const int dipWidth = dipSize.width();
106 const int dipHeight = dipSize.height();
107 if (dipWidth > rect.width() || dipHeight > rect.height()) {
109 painter->drawPixmap(rect, m_pixmap); // Too big: Let Qt scale down
110 } else {
111 QPoint point = rect.topLeft(); // center if needed
112 if (dipWidth < rect.width())
113 point.rx() += (rect.width() - dipWidth) / 2;
114 if (dipHeight < rect.height())
115 point.ry() += (rect.height() - dipHeight) / 2;
117 }
118 }
119
120private:
122};
123
124static inline QIcon getResourceCursorIcon(const QString &fn)
125{
126 return QIcon{new QtPixmapIconEngine(QPixmap(fn))};
127}
128
129// Get an icon from the resources of the Windows QPA plugin. No scaling required.
130[[maybe_unused]] static QIcon getWindowsCursorIcon(QLatin1StringView name)
131{
132 static constexpr auto root = ":/qt-project.org/windows/cursors/images/"_L1;
133 static constexpr QLatin1StringView suffixes[] = {"_32.png"_L1, "_48.png"_L1, "_64.png"_L1};
134 QIcon result;
135 for (auto suffix : suffixes) {
136 const QString path = root + name + suffix;
137 QPixmap pixmap(path);
138 if (pixmap.isNull())
139 qWarning("%s: Failed to load \"%s\"", __FUNCTION__, qPrintable(path));
140 else
141 result.addPixmap(pixmap);
142 }
143 return result;
144}
145
146// Get a (small) icon from the resources of the Cocoa QPA plugin.
147[[maybe_unused]] static QIcon getCocoaCursorIcon(QLatin1StringView name)
148{
149 static constexpr auto root = ":/qt-project.org/mac/cursors/images/"_L1;
150 const QString path = root + name;
151 QPixmap pixmap(path);
152 if (pixmap.isNull()) {
153 qWarning("%s: Failed to load \"%s\"", __FUNCTION__, qPrintable(path));
154 return {};
155 }
156 return QIcon{new QtPixmapIconEngine(pixmap)};
157}
158
160{
161 qAddPostRoutine(clearCursorDatabase);
162
163 if constexpr (QOperatingSystemVersion::currentType() == QOperatingSystemVersion::Windows) {
164 appendCursor(Qt::ClosedHandCursor, getWindowsCursorIcon("closedhandcursor"_L1));
165 appendCursor(Qt::OpenHandCursor, getWindowsCursorIcon("openhandcursor"_L1));
166 appendCursor(Qt::SplitHCursor, getWindowsCursorIcon("splithcursor"_L1));
167 appendCursor(Qt::SplitVCursor, getWindowsCursorIcon("splitvcursor"_L1));
168 }
169
170 if constexpr (QOperatingSystemVersion::currentType() == QOperatingSystemVersion::MacOS) {
171 appendCursor(Qt::BusyCursor, getCocoaCursorIcon("spincursor.png"_L1));
172 appendCursor(Qt::WaitCursor, getCocoaCursorIcon("waitcursor.png"_L1));
173 }
174
175 appendCursor(Qt::BlankCursor, QIcon{});
176
177 for (const auto &cursorResource : cursorResources) {
178 if (!hasCursor(cursorResource.first))
179 appendCursor(cursorResource.first, getResourceCursorIcon(cursorResource.second));
180 }
181}
182
184{
185 m_cursorNames.clear();
186 m_cursorIcons.clear();
187 m_valueToCursorShape.clear();
188 m_cursorShapeToValue.clear();
189}
190
191void QtCursorDatabase::appendCursor(Qt::CursorShape shape, const QIcon &icon)
192{
193 if (hasCursor(shape) || (shape != Qt::BlankCursor && icon.isNull()))
194 return;
195 const qsizetype value = m_cursorNames.size();
196 const char *name = shapeNames().value(shape, nullptr);
197 m_cursorNames.append(name ? QCoreApplication::translate("QtCursorDatabase", name) : QString{});
198 m_cursorIcons.insert(value, icon);
199 m_valueToCursorShape.insert(value, shape);
200 m_cursorShapeToValue.insert(shape, value);
201}
202
204{
205 return m_cursorNames;
206}
207
209{
210 return m_cursorIcons;
211}
212
213QString QtCursorDatabase::cursorToShapeName(const QCursor &cursor) const
214{
215 int val = cursorToValue(cursor);
216 if (val >= 0)
217 return m_cursorNames.at(val);
218 return {};
219}
220
221QIcon QtCursorDatabase::cursorToShapeIcon(const QCursor &cursor) const
222{
223 int val = cursorToValue(cursor);
224 return m_cursorIcons.value(val);
225}
226
227int QtCursorDatabase::cursorToValue(const QCursor &cursor) const
228{
229#ifndef QT_NO_CURSOR
230 return m_cursorShapeToValue.value(cursor.shape(), -1);
231#endif
232 return -1;
233}
234
235#ifndef QT_NO_CURSOR
237{
238 auto it = m_valueToCursorShape.constFind(value);
239 if (it != m_valueToCursorShape.cend())
240 return QCursor(it.value());
241 return {};
242}
243#endif
244
245Q_GLOBAL_STATIC(QtCursorDatabase, cursorDatabase)
246
248{
249 return cursorDatabase();
250}
251
253{
254public:
256
258
260 {
262 }
263
265 {
270 if (color.alpha() != 255) { // indicate alpha by an inset
272 color.setAlpha(255);
274 const QPoint offset{rect.width() / 4, rect.height() / 4};
275 const QRect inset{rect.topLeft() + offset, rect.size() / 2};
277 }
278 }
279
280private:
282};
283
284QPixmap QtPropertyBrowserUtils::brushValuePixmap(const QBrush &b, const QSize &size,
285 qreal devicePixelRatio)
286{
288 return engine.drawPixmap(size, devicePixelRatio);
289}
290
291QIcon QtPropertyBrowserUtils::brushValueIcon(const QBrush &b)
292{
293 return QIcon(new QtPropertyBrushValueIconEngine(b));
294}
295
296QString QtPropertyBrowserUtils::colorValueText(QColor c)
297{
298 return QCoreApplication::translate("QtPropertyBrowserUtils", "[%1, %2, %3] (%4)")
299 .arg(c.red()).arg(c.green()).arg(c.blue()).arg(c.alpha());
300}
301
329
330QPixmap QtPropertyBrowserUtils::fontValuePixmap(const QFont &font, const QSize &size,
331 qreal devicePixelRatio)
332{
334 return engine.drawPixmap(size, devicePixelRatio);
335}
336
337QIcon QtPropertyBrowserUtils::fontValueIcon(const QFont &f)
338{
339 return QIcon(new QtPropertyFontValueIconEngine(f));
340}
341
342QString QtPropertyBrowserUtils::fontValueText(const QFont &f)
343{
344 return QCoreApplication::translate("QtPropertyBrowserUtils", "[%1, %2]")
345 .arg(f.family()).arg(f.pointSize());
346}
347
348QString QtPropertyBrowserUtils::dateFormat()
349{
350 QLocale loc;
351 QString format = loc.dateFormat(QLocale::ShortFormat);
352 // Change dd.MM.yy, MM/dd/yy to 4 digit years
353 if (format.count(QLatin1Char('y')) == 2)
354 format.insert(format.indexOf(QLatin1Char('y')), "yy"_L1);
355 return format;
356}
357
358QString QtPropertyBrowserUtils::timeFormat()
359{
360 QLocale loc;
361 // ShortFormat is missing seconds on UNIX.
362 return loc.timeFormat(QLocale::LongFormat);
363}
364
365QString QtPropertyBrowserUtils::dateTimeFormat()
366{
367 QString format = dateFormat();
368 format += QLatin1Char(' ');
369 format += timeFormat();
370 return format;
371}
372
373QtBoolEdit::QtBoolEdit(QWidget *parent) :
374 QWidget(parent),
375 m_checkBox(new QCheckBox(this)),
376 m_textVisible(true)
377{
378 auto *lt = new QHBoxLayout;
379 if (QApplication::layoutDirection() == Qt::LeftToRight)
380 lt->setContentsMargins(4, 0, 0, 0);
381 else
382 lt->setContentsMargins(0, 0, 4, 0);
383 lt->addWidget(m_checkBox);
384 setLayout(lt);
385 connect(m_checkBox, &QAbstractButton::toggled, this, &QtBoolEdit::toggled);
386 setFocusProxy(m_checkBox);
387 m_checkBox->setText(tr("True"));
388}
389
390void QtBoolEdit::setTextVisible(bool textVisible)
391{
392 if (m_textVisible == textVisible)
393 return;
394
395 m_textVisible = textVisible;
396 if (m_textVisible)
397 m_checkBox->setText(isChecked() ? tr("True") : tr("False"));
398 else
399 m_checkBox->setText(QString());
400}
401
403{
404 return m_checkBox->checkState();
405}
406
407void QtBoolEdit::setCheckState(Qt::CheckState state)
408{
409 m_checkBox->setCheckState(state);
410}
411
413{
414 return m_checkBox->isChecked();
415}
416
418{
419 m_checkBox->setChecked(c);
420 if (!m_textVisible)
421 return;
422 m_checkBox->setText(isChecked() ? tr("True") : tr("False"));
423}
424
426{
427 return m_checkBox->blockSignals(block);
428}
429
430void QtBoolEdit::mousePressEvent(QMouseEvent *event)
431{
432 if (event->buttons() == Qt::LeftButton) {
433 m_checkBox->click();
434 event->accept();
435 } else {
436 QWidget::mousePressEvent(event);
437 }
438}
439
440QPixmap QtPropertyIconEngine::createEmptyPixmap(const QSize &size, qreal devicePixelRatio)
441{
442 QPixmap result(size * devicePixelRatio);
443 result.fill(Qt::transparent);
444 result.setDevicePixelRatio(devicePixelRatio);
445 return result;
446}
447
449
450// ### FIXME Qt 7: Remove? (reimplemented in Qt 6 since the default
451// cannot be trusted to handle fractional dpr).
452QPixmap QtPropertyIconEngine::scaledPixmap(const QSize &size, QIcon::Mode mode, QIcon::State state,
453 qreal scale)
454{
455 QPixmap result = createEmptyPixmap(size, scale);
456 QPainter painter(&result);
457 paint(&painter, QRect(0, 0, size.width(), size.height()), mode, state);
458 painter.end();
459 return result;
460}
461
462QPixmap QtPropertyIconEngine::drawPixmap(const QSize &size, qreal devicePixelRatio)
463{
464 return scaledPixmap(size, QIcon::Mode::Normal, QIcon::State::On, devicePixelRatio);
465}
466
467QT_END_NAMESPACE
Qt::CheckState checkState() const
void setTextVisible(bool textVisible)
bool blockCheckBoxSignals(bool block)
void setCheckState(Qt::CheckState state)
QCursor valueToCursor(int value) const
QStringList cursorShapeNames() const
int cursorToValue(const QCursor &cursor) const
QString cursorToShapeName(const QCursor &cursor) const
QIcon cursorToShapeIcon(const QCursor &cursor) const
static QtCursorDatabase * instance()
QMap< int, QIcon > cursorShapeIcons() const
QPixmap drawPixmap(const QSize &size, qreal devicePixelRatio)
static QIcon getWindowsCursorIcon(QLatin1StringView name)
static void clearCursorDatabase()
static QIcon getCocoaCursorIcon(QLatin1StringView name)
static const ShapeNames & shapeNames()
static QIcon getResourceCursorIcon(const QString &fn)
static constexpr CursorResource cursorResources[]