12#include "QskSkinManager.h"
13#include "QskInternalMacros.h"
19#include <private/qquickitem_p.h>
20#include <private/qquickitemchangelistener_p.h>
21#include <private/qquickwindow_p.h>
22#include <private/qsgrenderer_p.h>
25#include <qpa/qwindowsysteminterface.h>
26#include <QGuiApplication>
30#ifdef QSK_DEBUG_RENDER_TIMING
32#include <qelapsedtimer.h>
33#include <qloggingcategory.h>
34Q_LOGGING_CATEGORY( logTiming,
"qsk.window.timing", QtCriticalMsg )
38extern QLocale qskInheritedLocale(
const QObject* );
39extern void qskInheritLocale( QObject*,
const QLocale& );
41static void qskResolveLocale(
QskWindow* );
42static bool qskEnforcedSkin =
false;
44static inline void qskSendEventTo( QObject*
object, QEvent::Type type )
47 QCoreApplication::sendEvent(
object, &event );
50static QQuickItem* qskDefaultFocusItem( QQuickWindow* window )
52 const auto children = qskPaintOrderChildItems( window->contentItem() );
53 for (
auto it = children.crbegin(); it != children.crend(); ++it )
57 if ( child->isFocusScope() && child->isVisible()
58 && child->isEnabled() && child->activeFocusOnTab() )
64 return window->contentItem()->nextItemInFocusChain(
true );
69 class ChildListener final :
public QQuickItemChangeListener
72 void setEnabled( QQuickItem* contentItem,
bool on )
74 m_contentItem = contentItem;
76 const QQuickItemPrivate::ChangeTypes types = QQuickItemPrivate::Children;
78 QQuickItemPrivate* p = QQuickItemPrivate::get( contentItem );
80 p->addItemChangeListener(
this, types );
82 p->removeItemChangeListener(
this, types );
85 void itemChildAdded( QQuickItem*, QQuickItem* )
override
88 if ( window->isExposed() )
91 QCoreApplication::postEvent( window,
new QEvent( QEvent::LayoutRequest ) );
96 QQuickItem* m_contentItem =
nullptr;
100static inline int qskToIntegerConstraint( qreal valueF )
106 if ( valueF >= std::numeric_limits< int >::max() )
107 value = std::numeric_limits< int >::max();
109 value = ( int ) qCeil( valueF );
115static inline void qskSetVisualizationMode(
116 QQuickWindow* window,
const QByteArray& mode )
118 auto d = QQuickWindowPrivate::get( window );
119#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
120 d->visualizationMode = mode;
122 d->customRenderMode = mode;
126static inline const QByteArray& qskVisualizationMode(
const QQuickWindow* window )
128 auto d = QQuickWindowPrivate::get(
const_cast< QQuickWindow*
>( window ) );
129#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
130 return d->visualizationMode;
132 return d->customRenderMode;
136class QskWindowPrivate :
public QQuickWindowPrivate
142 : preferredSize( -1, -1 )
143 , eventAcceptance(
QskWindow::EventProcessed )
144 , explicitLocale( false )
145 , deleteOnClose( false )
146 , autoLayoutChildren( true )
147 , showedOnce( false )
151#ifdef QSK_DEBUG_RENDER_TIMING
152 QElapsedTimer renderInterval;
155 QPointer< QskSkin > skin;
157 ChildListener contentItemListener;
163 QskWindow::EventAcceptance eventAcceptance;
165 bool explicitLocale : 1;
166 bool deleteOnClose : 1;
167 bool autoLayoutChildren : 1;
171QskWindow::QskWindow( QWindow* parent )
172 : Inherited( *( new QskWindowPrivate() ), parent )
174 QSurfaceFormat fmt = format();
178 fmt.setOption( QSurfaceFormat::DebugContext );
186 contentItem()->setProperty(
"locale", locale() );
191 qskResolveLocale(
this );
194 d_func()->contentItemListener.setEnabled( contentItem(),
true );
196 if ( !qskEnforcedSkin )
197 connect(
this, &QQuickWindow::afterAnimating,
this, &QskWindow::enforceSkin );
200QskWindow::QskWindow( QQuickRenderControl* renderControl, QWindow* parent )
205 d->renderControl = renderControl;
206 d->init(
this, renderControl );
209QskWindow::~QskWindow()
213void QskWindow::setScreen(
const QString& name )
215 if ( !name.isEmpty() )
217 const auto screens = QGuiApplication::screens();
218 for (
auto screen : screens )
220 if ( screen->name() == name )
228 setScreen( QGuiApplication::primaryScreen() );
231void QskWindow::resizeF(
const QSizeF& size )
233 const int w =
static_cast< int >( qCeil( size.width() ) );
234 const int h =
static_cast< int >( qCeil( size.height() ) );
239bool QskWindow::deleteOnClose()
const
242 return d->deleteOnClose;
245void QskWindow::setDeleteOnClose(
bool on )
249 if ( on != d->deleteOnClose )
251 d->deleteOnClose = on;
252 Q_EMIT deleteOnCloseChanged();
256void QskWindow::setAutoLayoutChildren(
bool on )
260 if ( on != d->autoLayoutChildren )
262 d->autoLayoutChildren = on;
264 qskSendEventTo(
this, QEvent::LayoutRequest );
266 Q_EMIT autoLayoutChildrenChanged();
270bool QskWindow::autoLayoutChildren()
const
273 return d->autoLayoutChildren;
276void QskWindow::addItem( QQuickItem* item )
278 if ( item ==
nullptr )
281 item->setParent( contentItem() );
282 item->setParentItem( contentItem() );
285void QskWindow::polishItems()
291bool QskWindow::event( QEvent* event )
303 d->eventAcceptance = EventProcessed;
305 switch ( event->type() )
309 if ( !d->showedOnce )
311 d->showedOnce =
true;
326 if ( d->geometry.size().isEmpty() )
328 QSize sz = sizeConstraint();
331 sz = sz.expandedTo( minimumSize() );
332 sz = sz.boundedTo( maximumSize() );
341 case QEvent::LayoutRequest:
347 case QEvent::LocaleChange:
349 Q_EMIT localeChanged( locale() );
354 if ( event->isAccepted() )
356 if ( d->deleteOnClose )
361 case QEvent::UpdateRequest:
363#ifdef QSK_DEBUG_RENDER_TIMING
364 if ( logTiming().isDebugEnabled() )
366 if ( !d->renderInterval.isValid() )
367 d->renderInterval.start();
369 qCDebug( logTiming() ) <<
"update timer - elapsed"
370 << d->renderInterval.restart() << objectName();
380 return Inherited::event( event );
383void QskWindow::keyPressEvent( QKeyEvent* event )
385 if ( qskFocusChainIncrement( event ) != 0 )
387 auto focusItem = activeFocusItem();
388 if ( focusItem ==
nullptr || focusItem == contentItem() )
403 ensureFocus( Qt::TabFocusReason );
410 Inherited::keyPressEvent( event );
413void QskWindow::keyReleaseEvent( QKeyEvent* event )
415 Inherited::keyReleaseEvent( event );
418void QskWindow::exposeEvent( QExposeEvent* event )
420#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
421 if ( qskRenderingHardwareInterface(
this ) )
430 qFatal(
"the experimental Qt5 RHI implementation is not supported:\n"
431 "\tuse Qt6 or run the default OpenGL backend." );
434 ensureFocus( Qt::OtherFocusReason );
437 Inherited::exposeEvent( event );
440void QskWindow::resizeEvent( QResizeEvent* event )
442 auto rootItem = contentItem();
444 const auto oldRect = qskItemGeometry( rootItem );
445 Inherited::resizeEvent( event );
447 const auto newRect = qskItemGeometry( rootItem );
448 if ( newRect != oldRect )
451 QCoreApplication::sendEvent( rootItem, &ev );
458QLocale QskWindow::locale()
const
464void QskWindow::setLocale(
const QLocale& locale )
468 d->explicitLocale =
true;
470 if ( d->locale != locale )
473 qskSendEventTo(
this, QEvent::LocaleChange );
474 qskInheritLocale(
this, locale );
478void QskWindow::resetLocale()
482 d->explicitLocale =
false;
483 qskResolveLocale(
this );
486QSK_HIDDEN_EXTERNAL_BEGIN
488bool qskInheritLocale(
QskWindow* window,
const QLocale& locale )
490 auto d =
static_cast< QskWindowPrivate*
>( QQuickWindowPrivate::get( window ) );
492 if ( d->explicitLocale || d->locale == locale )
496 qskSendEventTo( window, QEvent::LocaleChange );
501QSK_HIDDEN_EXTERNAL_END
503static void qskResolveLocale(
QskWindow* window )
505 auto d =
static_cast< QskWindowPrivate*
>( QQuickWindowPrivate::get( window ) );
507 const auto locale = qskInheritedLocale( window );
509 if ( d->locale != locale )
512 qskSendEventTo( window, QEvent::LocaleChange );
514 qskInheritLocale( window, locale );
518void QskWindow::setPreferredSize(
const QSize& size )
521 d->preferredSize = size;
524QSize QskWindow::preferredSize()
const
527 return d->preferredSize;
530QSize QskWindow::sizeConstraint()
const
534 QSizeF constraint = d->preferredSize;
536 if ( !constraint.isValid() )
538 const bool doWidth = constraint.width() < 0;
539 const bool doHeight = constraint.height() < 0;
541 const auto children = contentItem()->childItems();
542 for (
auto child : children )
544 if ( qskIsVisibleToLayout( child ) )
546 const auto size = qskSizeConstraint( child, Qt::PreferredSize );
549 constraint.setWidth( qMax( constraint.width(), size.width() ) );
552 constraint.setHeight( qMax( constraint.height(), size.height() ) );
559 return QSize( qskToIntegerConstraint( constraint.width() ),
560 qskToIntegerConstraint( constraint.height() ) );
563void QskWindow::setFixedSize(
const QSize& size )
566 setMinimumSize( size );
567 setMaximumSize( size );
570void QskWindow::layoutItems()
574 if ( !d->autoLayoutChildren )
577 const QRectF rect = qskItemGeometry( contentItem() );
579 const auto children = contentItem()->childItems();
580 for (
auto child : children )
582 if ( qskIsAdjustableByLayout( child ) )
584 const auto r = qskConstrainedItemRect( child, rect );
585 qskSetItemGeometry( child, r );
590void QskWindow::ensureFocus( Qt::FocusReason reason )
592 auto focusItem = contentItem()->scopedFocusItem();
594 if ( focusItem ==
nullptr )
596 focusItem = qskDefaultFocusItem(
this );
598 focusItem->setFocus(
true, reason );
602void QskWindow::setCustomRenderMode(
const char* mode )
604 class RenderJob final :
public QRunnable
607 RenderJob( QQuickWindow* window,
const QByteArray& mode )
615 qskSetVisualizationMode( m_window, m_mode );
617 auto d = QQuickWindowPrivate::get( m_window );
620 delete d->renderer->rootNode();
622 d->renderer =
nullptr;
624 QMetaObject::invokeMethod( m_window,
"update" );
629 QQuickWindow* m_window;
630 const QByteArray m_mode;
633 const QByteArray newMode( mode );
635 const auto oldMode = qskVisualizationMode(
this );
637 if ( newMode != oldMode )
648 if ( newMode.isEmpty() != oldMode.isEmpty() )
649 scheduleRenderJob(
new RenderJob(
this, newMode ), AfterSwapStage );
651 qskSetVisualizationMode(
this, newMode );
657const char* QskWindow::customRenderMode()
const
659 return qskVisualizationMode(
this );
662void QskWindow::enforceSkin()
664 if ( !qskEnforcedSkin )
670 ( void ) qskSkinManager->skin();
671 qskEnforcedSkin =
true;
674 disconnect(
this, &QQuickWindow::afterAnimating,
this, &QskWindow::enforceSkin );
677void QskWindow::setEventAcceptance( EventAcceptance acceptance )
679 d_func()->eventAcceptance = acceptance;
682QskWindow::EventAcceptance QskWindow::eventAcceptance()
const
684 return d_func()->eventAcceptance;
687void QskWindow::setSkin(
const QString& skinName )
690 auto skin = QskSkinManager::instance()->createSkin( skinName );
694void QskWindow::setSkin(
QskSkin* skin )
698 if ( d->skin == skin )
703 if ( d->skin->parent() ==
this )
707 if ( skin && skin->parent() ==
nullptr )
708 skin->setParent(
this );
713QskSkin* QskWindow::skin()
const
715 return d_func()->skin;
718QskSkin* qskEffectiveSkin(
const QQuickWindow* window )
720 if (
auto w = qobject_cast< const QskWindow* >( window ) )
722 if (
auto skin = w->skin() )
726 return qskSkinManager->skin();
729#include "moc_QskWindow.cpp"