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