Files
pilotclient/src/blackgui/guiutility.cpp
Klaus Basan 8e2a1b1b6f Ref T225, utility function to force stylesheet update
* Needed when setting UI element to readonly, and stylesheet is different for readonly
* see https://stackoverflow.com/q/48141205/356726
2018-01-30 20:29:29 +01:00

514 lines
17 KiB
C++

/* Copyright (C) 2014
* swift project Community / Contributors
*
* This file is part of swift project. It is subject to the license terms in the LICENSE file found in the top-level
* directory of this distribution and at http://www.swift-project.org/license.html. No part of swift project,
* including this file, may be copied, modified, propagated, or distributed except according to the terms
* contained in the LICENSE file.
*/
#include "blackgui/enableforframelesswindow.h"
#include "blackgui/guiutility.h"
#include "blackgui/overlaymessagesframe.h"
#include "blackmisc/verify.h"
#include <QApplication>
#include <QCheckBox>
#include <QComboBox>
#include <QJsonDocument>
#include <QJsonObject>
#include <QJsonValue>
#include <QLayout>
#include <QLayoutItem>
#include <QList>
#include <QMainWindow>
#include <QMetaType>
#include <QMimeData>
#include <QObject>
#include <QRegularExpression>
#include <QTabWidget>
#include <QThreadStorage>
#include <QWidget>
#include <QTimer>
#include <Qt>
#include <QtGlobal>
#include <QGraphicsOpacityEffect>
#include <QPropertyAnimation>
#include <QDesktopWidget>
using namespace BlackMisc;
namespace BlackGui
{
QWidget *CGuiUtility::s_mainApplicationWidget = nullptr;
CEnableForFramelessWindow *CGuiUtility::mainFramelessEnabledWindow()
{
const QWidgetList tlw = topLevelApplicationWidgetsWithName();
for (QWidget *w : tlw)
{
// best choice is to check on frameless window
CEnableForFramelessWindow *mw = dynamic_cast<CEnableForFramelessWindow *>(w);
if (mw && mw->isMainApplicationWindow()) { return mw; }
}
return nullptr;
}
namespace Private
{
QWidget *mainApplicationWidgetSearch()
{
CEnableForFramelessWindow *mw = CGuiUtility::mainFramelessEnabledWindow();
if (mw && mw->getWidget())
{
return mw->getWidget();
}
// second choice, try via QMainWindow
const QWidgetList tlw = CGuiUtility::topLevelApplicationWidgetsWithName();
for (QWidget *w : tlw)
{
QMainWindow *qmw = qobject_cast<QMainWindow *>(w);
if (!qmw) { continue; }
if (!qmw->parentWidget()) { return qmw; }
}
return nullptr;
}
} // ns
void CGuiUtility::registerMainApplicationWidget(QWidget *mainWidget)
{
CGuiUtility::s_mainApplicationWidget = mainWidget;
}
QWidget *CGuiUtility::mainApplicationWidget()
{
if (!CGuiUtility::s_mainApplicationWidget)
{
CGuiUtility::s_mainApplicationWidget = Private::mainApplicationWidgetSearch();
}
return CGuiUtility::s_mainApplicationWidget;
}
qreal CGuiUtility::mainApplicationWidgetPixelRatio()
{
const QWidget *mw = CGuiUtility::mainApplicationWidget();
if (mw) { return mw->devicePixelRatio(); }
return 1.0;
}
QSize CGuiUtility::desktopSize()
{
const QWidget *mw = CGuiUtility::mainApplicationWidget();
if (mw) { return QApplication::desktop()->screenGeometry(mw).size(); }
// main screen
return QApplication::desktop()->screenGeometry().size();
}
bool CGuiUtility::isMainWindowFrameless()
{
const CEnableForFramelessWindow *mw = CGuiUtility::mainFramelessEnabledWindow();
return (mw && mw->isFrameless());
}
bool CGuiUtility::lenientTitleComparison(const QString &title, const QString &comparison)
{
if (title == comparison) { return true; }
QString t(title.trimmed().toLower().simplified());
QString c(comparison.trimmed().toLower().simplified());
Q_ASSERT_X(!t.isEmpty(), Q_FUNC_INFO, "missing value");
Q_ASSERT_X(!c.isEmpty(), Q_FUNC_INFO, "missing value");
if (t == c) { return true; }
// further unify
static QThreadStorage<QRegularExpression> tsRegex;
if (! tsRegex.hasLocalData()) { tsRegex.setLocalData(QRegularExpression("[^a-z0-9\\s]")); }
const QRegularExpression &regexp = tsRegex.localData();
t = t.remove(regexp);
c = c.remove(regexp);
return t == c;
}
bool CGuiUtility::setComboBoxValueByStartingString(QComboBox *box, const QString &candidate, const QString &unspecified)
{
if (!box) { return false; }
if (!candidate.isEmpty())
{
for (int i = 0; i < box->count(); i++)
{
const QString t(box->itemText(i));
if (t.startsWith(candidate, Qt::CaseInsensitive))
{
box->setCurrentIndex(i);
return true;
}
}
}
// not found
if (unspecified.isEmpty()) { return false; }
for (int i = 0; i < box->count(); i++)
{
const QString t(box->itemText(i));
if (t.startsWith(unspecified, Qt::CaseInsensitive))
{
box->setCurrentIndex(i);
return true;
}
}
return false;
}
bool CGuiUtility::hasSwiftVariantMimeType(const QMimeData *mime)
{
return mime && mime->hasFormat(swiftJsonDragAndDropMimeType());
}
CVariant CGuiUtility::fromSwiftDragAndDropData(const QMimeData *mime)
{
if (hasSwiftVariantMimeType(mime))
{
return fromSwiftDragAndDropData(mime->data(swiftJsonDragAndDropMimeType()));
}
return CVariant();
}
CVariant CGuiUtility::fromSwiftDragAndDropData(const QByteArray &utf8Data)
{
if (utf8Data.isEmpty()) { return CVariant(); }
const QJsonDocument jsonDoc(QJsonDocument::fromJson(utf8Data));
const QJsonObject jsonObj(jsonDoc.object());
const QString typeName(jsonObj.value("type").toString());
const int typeId = QMetaType::type(qPrintable(typeName));
// check if a potential valid value object
if (typeName.isEmpty() || typeId == QMetaType::UnknownType) { return CVariant(); }
CVariant valueVariant;
const CStatusMessage status = valueVariant.convertFromJsonNoThrow(jsonObj, {}, {});
if (status.isFailure()) { return CVariant(); }
return valueVariant;
}
int CGuiUtility::metaTypeIdFromSwiftDragAndDropData(const QMimeData *mime)
{
constexpr int Unknown = static_cast<int>(QMetaType::UnknownType);
if (!hasSwiftVariantMimeType(mime)) { return Unknown; }
static const QJsonObject jsonObj(QJsonDocument::fromJson(mime->data(swiftJsonDragAndDropMimeType())).object());
Q_ASSERT_X(!jsonObj.isEmpty(), Q_FUNC_INFO, "Empty JSON object");
const QString typeName(jsonObj.value("type").toString());
if (typeName.isEmpty()) { return Unknown; }
const int typeId = QMetaType::type(qPrintable(typeName));
return typeId;
}
COverlayMessagesFrame *CGuiUtility::nextOverlayMessageFrame(QWidget *widget, int maxLevels)
{
if (!widget || maxLevels < 1) { return nullptr; }
COverlayMessagesFrame *o = qobject_cast<COverlayMessagesFrame *> (widget);
if (o) { return o; }
int cl = 0;
QWidget *cw = widget->parentWidget();
while (cl < maxLevels && cw)
{
o = qobject_cast<COverlayMessagesFrame *> (cw);
if (o) { return o; }
cl++;
cw = cw->parentWidget();
}
return nullptr;
}
const QString &CGuiUtility::swiftJsonDragAndDropMimeType()
{
static const QString m("text/json/swift");
return m;
}
void CGuiUtility::checkBoxReadOnly(QCheckBox *checkBox, bool readOnly)
{
static const QCheckBox defaultBox;
BLACK_VERIFY_X(checkBox, Q_FUNC_INFO, "no checkbox");
if (!checkBox) { return; }
if (readOnly)
{
checkBox->setAttribute(Qt::WA_TransparentForMouseEvents);
checkBox->setFocusPolicy(Qt::NoFocus);
}
else
{
checkBox->setAttribute(Qt::WA_TransparentForMouseEvents, defaultBox.testAttribute(Qt::WA_TransparentForMouseEvents));
checkBox->setFocusPolicy(defaultBox.focusPolicy());
}
}
void CGuiUtility::checkBoxesReadOnly(QWidget *parent, bool readOnly)
{
if (!parent) { return; }
QList<QCheckBox *> allCheckBoxes = parent->findChildren<QCheckBox *>();
for (QCheckBox *cb : allCheckBoxes)
{
CGuiUtility::checkBoxReadOnly(cb, readOnly);
}
}
QWidgetList CGuiUtility::topLevelApplicationWidgetsWithName()
{
QWidgetList tlw = QApplication::topLevelWidgets();
QWidgetList rl;
for (QWidget *w : tlw)
{
if (w->objectName().isEmpty()) { continue; }
rl.append(w);
}
return rl;
}
QPoint CGuiUtility::mainWidgetPosition()
{
CEnableForFramelessWindow *mw = CGuiUtility::mainFramelessEnabledWindow();
return (mw) ? mw->getWidget()->pos() : QPoint();
}
QString CGuiUtility::replaceTabCountValue(const QString &oldName, int count)
{
const QString v = QString(" (").append(QString::number(count)).append(")");
if (oldName.isEmpty()) { return v; }
int index = oldName.lastIndexOf('(');
if (index == 0) { return v; }
if (index < 0) { return QString(oldName).trimmed().append(v); }
return QString(oldName.left(index)).trimmed().append(v);
}
void CGuiUtility::deleteLayout(QLayout *layout, bool deleteWidgets)
{
// http://stackoverflow.com/a/7569928/356726
if (!layout) { return; }
QLayoutItem *item {nullptr};
while ((item = layout->takeAt(0)))
{
QLayout *sublayout {nullptr};
QWidget *widget {nullptr};
if ((sublayout = item->layout()))
{
deleteLayout(sublayout, deleteWidgets);
}
else if ((widget = item->widget()))
{
widget->hide();
if (deleteWidgets)
{
delete widget;
}
}
else {delete item;}
}
// then finally
delete layout;
}
bool CGuiUtility::staysOnTop(QWidget *widget)
{
if (!widget) { return false; }
const Qt::WindowFlags flags = widget->windowFlags();
return Qt::WindowStaysOnTopHint & flags;
}
QTabWidget *CGuiUtility::parentTabWidget(QWidget *widget, int maxLevels)
{
int level = 0;
do
{
widget = widget->parentWidget();
if (!widget) { return nullptr; }
QTabWidget *tw = qobject_cast<QTabWidget *>(widget);
if (tw) { return tw; }
level++;
}
while (level < maxLevels);
return nullptr;
}
bool CGuiUtility::toggleStayOnTop(QWidget *widget)
{
if (!widget) { return false; }
Qt::WindowFlags flags = widget->windowFlags();
if (Qt::WindowStaysOnTopHint & flags)
{
flags &= ~Qt::WindowStaysOnTopHint;
flags |= Qt::WindowStaysOnBottomHint;
}
else
{
flags &= ~Qt::WindowStaysOnBottomHint;
flags |= Qt::WindowStaysOnTopHint;
}
widget->setWindowFlags(flags);
widget->show();
return Qt::WindowStaysOnTopHint & flags;
}
QString CGuiUtility::marginsToString(const QMargins &margins)
{
const QString s("%1:%2:%3:%4");
return s.arg(margins.left()).arg(margins.top()).arg(margins.right()).arg(margins.bottom());
}
QMargins CGuiUtility::stringToMargins(const QString &str)
{
const QStringList parts = str.split(":");
Q_ASSERT_X(parts.size() == 4, Q_FUNC_INFO, "malformed");
bool ok = false;
const int l = parts.at(0).toInt(&ok);
Q_ASSERT_X(ok, Q_FUNC_INFO, "malformed number");
const int t = parts.at(1).toInt(&ok);
Q_ASSERT_X(ok, Q_FUNC_INFO, "malformed number");
const int r = parts.at(2).toInt(&ok);
Q_ASSERT_X(ok, Q_FUNC_INFO, "malformed number");
const int b = parts.at(3).toInt(&ok);
Q_ASSERT_X(ok, Q_FUNC_INFO, "malformed number");
Q_UNUSED(ok);
return QMargins(l, t, r, b);
}
QList<int> CGuiUtility::indexToUniqueRows(const QModelIndexList &indexes)
{
QList<int> rows;
for (const QModelIndex &i : indexes)
{
const int r = i.row();
if (rows.contains(r)) { continue; }
rows.append(r);
}
return rows;
}
bool CGuiUtility::isTopLevelWidget(QWidget *widget)
{
return QApplication::topLevelWidgets().contains(widget);
}
QGraphicsOpacityEffect *CGuiUtility::fadeInWidget(int durationMs, QWidget *widget, double startValue, double endValue)
{
// http://stackoverflow.com/questions/19087822/how-to-make-qt-widgets-fade-in-or-fade-out#
Q_ASSERT(widget);
QGraphicsOpacityEffect *effect = new QGraphicsOpacityEffect(widget);
widget->setGraphicsEffect(effect);
QPropertyAnimation *a = new QPropertyAnimation(effect, "opacity");
a->setDuration(durationMs);
a->setStartValue(startValue);
a->setEndValue(endValue);
a->setEasingCurve(QEasingCurve::InBack);
a->start(QPropertyAnimation::DeleteWhenStopped);
return effect;
}
QGraphicsOpacityEffect *CGuiUtility::fadeOutWidget(int durationMs, QWidget *widget, double startValue, double endValue)
{
Q_ASSERT(widget);
QGraphicsOpacityEffect *effect = new QGraphicsOpacityEffect(widget);
widget->setGraphicsEffect(effect);
QPropertyAnimation *a = new QPropertyAnimation(effect, "opacity");
a->setDuration(durationMs);
a->setStartValue(startValue);
a->setEndValue(endValue);
a->setEasingCurve(QEasingCurve::OutBack);
a->start(QPropertyAnimation::DeleteWhenStopped);
return effect;
}
QFontMetrics CGuiUtility::currentFontMetrics()
{
const QWidget *w = CGuiUtility::mainApplicationWidget();
if (w) { return w->fontMetrics(); }
return QApplication::fontMetrics();
}
QFontMetricsF CGuiUtility::currentFontMetricsF()
{
return QFontMetricsF(CGuiUtility::currentFontMetrics());
}
QFont CGuiUtility::currentFont()
{
const QWidget *w = CGuiUtility::mainApplicationWidget();
if (w) { return w->font(); }
return QApplication::font();
}
QSizeF CGuiUtility::fontMetrics80Chars(bool withRatio)
{
static const QString s("01234567890123456789012345678901234567890123456789012345678901234567890123456789");
const QFontMetricsF fm = CGuiUtility::currentFontMetricsF();
const qreal scale = withRatio ? CGuiUtility::mainApplicationWidgetPixelRatio() : 1.0;
const qreal w = fm.width(s) * scale;
const qreal h = fm.height() * scale;
return QSizeF(w, h);
}
QSizeF CGuiUtility::fontMetricsLazyDog43Chars(bool withRatio)
{
// 43 characters 0123456789012345678901234567890123456789012
static const QString s("The quick brown fox jumps over the lazy dog");
const QFontMetricsF fm = CGuiUtility::currentFontMetrics();
const qreal scale = withRatio ? CGuiUtility::mainApplicationWidgetPixelRatio() : 1.0;
const qreal w = fm.width(s) * scale;
const qreal h = fm.height() * scale;
return QSizeF(w, h);
}
QSizeF CGuiUtility::fontMetricsEstimateSize(int xCharacters, int yCharacters, bool withRatio)
{
// 1920/1080: 560/16 256/16 => 530/960
// 3840/2160: 400/10 178/10 => 375/600
const QSizeF s1 = CGuiUtility::fontMetrics80Chars(withRatio);
const QSizeF s2 = CGuiUtility::fontMetricsLazyDog43Chars(withRatio);
const QSizeF s = s1 + s2;
const qreal w = s.width() * xCharacters / 123;
const qreal h = s.height() * yCharacters / 2;
return QSizeF(w, h);
}
QString CGuiUtility::metricsInfo()
{
static const QString s("%1 %2 %3 | 80 chars: w%4 h%5 | 43 chars: w%6 h%7");
const QSizeF s80 = CGuiUtility::fontMetrics80Chars();
const QSizeF s43 = CGuiUtility::fontMetricsLazyDog43Chars();
QString ratio("-");
QString desktop("-");
const QWidget *mainWidget = CGuiUtility::mainApplicationWidget();
if (mainWidget)
{
QSize sd = QApplication::desktop()->screenGeometry().size();
desktop = QString("Desktop w%1 w%2").arg(sd.width()).arg(sd.height());
ratio = QString("ratio: %1").arg(mainWidget->devicePixelRatioF());
}
return s.
arg(desktop).
arg(CGuiUtility::isUsingHighDpiScreenSupport() ? "hi DPI" : "-").
arg(ratio).
arg(s80.width()).arg(s80.height()).arg(s43.width()).arg(s43.height());
}
bool CGuiUtility::isUsingHighDpiScreenSupport()
{
const QByteArray v = qgetenv("QT_AUTO_SCREEN_SCALE_FACTOR");
const QString vs(v);
const bool highDpi = stringToBool(vs);
return highDpi;
}
void CGuiUtility::forceStyleSheetUpdate(QWidget *widget)
{
if (!widget) { return; }
widget->setStyleSheet(widget->styleSheet());
}
} // ns