QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskWindow.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskWindow.h"
7#include "QskControl.h"
8#include "QskEvent.h"
9#include "QskQuick.h"
10#include "QskSetup.h"
11#include "QskSkin.h"
12#include "QskSkinManager.h"
13#include "QskInternalMacros.h"
14
15#include <qmath.h>
16#include <qpointer.h>
17
18QSK_QT_PRIVATE_BEGIN
19#include <private/qquickitem_p.h>
20#include <private/qquickitemchangelistener_p.h>
21#include <private/qquickwindow_p.h>
22#include <private/qsgrenderer_p.h>
23QSK_QT_PRIVATE_END
24
25#include <qpa/qwindowsysteminterface.h>
26#include <QGuiApplication>
27
28// #define QSK_DEBUG_RENDER_TIMING
29
30#ifdef QSK_DEBUG_RENDER_TIMING
31
32#include <qelapsedtimer.h>
33#include <qloggingcategory.h>
34Q_LOGGING_CATEGORY( logTiming, "qsk.window.timing", QtCriticalMsg )
35
36#endif
37
38extern QLocale qskInheritedLocale( const QObject* );
39extern void qskInheritLocale( QObject*, const QLocale& );
40
41static void qskResolveLocale( QskWindow* );
42static bool qskEnforcedSkin = false;
43
44static inline void qskSendEventTo( QObject* object, QEvent::Type type )
45{
46 QEvent event( type );
47 QCoreApplication::sendEvent( object, &event );
48}
49
50static QQuickItem* qskDefaultFocusItem( QQuickWindow* window )
51{
52 const auto children = qskPaintOrderChildItems( window->contentItem() );
53 for ( auto it = children.crbegin(); it != children.crend(); ++it )
54 {
55 auto child = *it;
56
57 if ( child->isFocusScope() && child->isVisible()
58 && child->isEnabled() && child->activeFocusOnTab() )
59 {
60 return child;
61 }
62 }
63
64 return window->contentItem()->nextItemInFocusChain( true );
65}
66
67namespace
68{
69 class ChildListener final : public QQuickItemChangeListener
70 {
71 public:
72 void setEnabled( QQuickItem* contentItem, bool on )
73 {
74 m_contentItem = contentItem;
75
76 const QQuickItemPrivate::ChangeTypes types = QQuickItemPrivate::Children;
77
78 QQuickItemPrivate* p = QQuickItemPrivate::get( contentItem );
79 if ( on )
80 p->addItemChangeListener( this, types );
81 else
82 p->removeItemChangeListener( this, types );
83 }
84
85 void itemChildAdded( QQuickItem*, QQuickItem* ) override
86 {
87 QskWindow* window = static_cast< QskWindow* >( m_contentItem->window() );
88 if ( window->isExposed() )
89 {
90 // the child is not fully constructed
91 QCoreApplication::postEvent( window, new QEvent( QEvent::LayoutRequest ) );
92 }
93 }
94
95 private:
96 QQuickItem* m_contentItem = nullptr;
97 };
98}
99
100static inline int qskToIntegerConstraint( qreal valueF )
101{
102 int value = -1;
103
104 if ( valueF >= 0.0 )
105 {
106 if ( valueF >= std::numeric_limits< int >::max() )
107 value = std::numeric_limits< int >::max();
108 else
109 value = ( int ) qCeil( valueF );
110 }
111
112 return value;
113}
114
115static inline void qskSetVisualizationMode(
116 QQuickWindow* window, const QByteArray& mode )
117{
118 auto d = QQuickWindowPrivate::get( window );
119#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
120 d->visualizationMode = mode;
121#else
122 d->customRenderMode = mode;
123#endif
124}
125
126static inline const QByteArray& qskVisualizationMode( const QQuickWindow* window )
127{
128 auto d = QQuickWindowPrivate::get( const_cast< QQuickWindow* >( window ) );
129#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
130 return d->visualizationMode;
131#else
132 return d->customRenderMode;
133#endif
134}
135
136class QskWindowPrivate : public QQuickWindowPrivate
137{
138 Q_DECLARE_PUBLIC( QskWindow )
139
140 public:
141 QskWindowPrivate()
142 : preferredSize( -1, -1 )
143 , eventAcceptance( QskWindow::EventProcessed )
144 , explicitLocale( false )
145 , deleteOnClose( false )
146 , autoLayoutChildren( true )
147 , showedOnce( false )
148 {
149 }
150
151#ifdef QSK_DEBUG_RENDER_TIMING
152 QElapsedTimer renderInterval;
153#endif
154
155 QPointer< QskSkin > skin;
156
157 ChildListener contentItemListener;
158 QLocale locale;
159
160 // minimum/maximum constraints are offered by QWindow
161 QSize preferredSize;
162
163 QskWindow::EventAcceptance eventAcceptance;
164
165 bool explicitLocale : 1;
166 bool deleteOnClose : 1;
167 bool autoLayoutChildren : 1;
168 bool showedOnce : 1;
169};
170
171QskWindow::QskWindow( QWindow* parent )
172 : Inherited( *( new QskWindowPrivate() ), parent )
173{
174 QSurfaceFormat fmt = format();
175 fmt.setSamples( 4 );
176#if 0
177 // for QOpenGLDebugLogger
178 fmt.setOption( QSurfaceFormat::DebugContext );
179#endif
180 setFormat( fmt );
181
182 /*
183 So that inheriting/resolving of the locale works
184 over all windows and items.
185 */
186 contentItem()->setProperty( "locale", locale() );
187
188 if ( parent )
189 {
190 // also when the parent changes TODO ...
191 qskResolveLocale( this );
192 }
193
194 d_func()->contentItemListener.setEnabled( contentItem(), true );
195
196 if ( !qskEnforcedSkin )
197 connect( this, &QQuickWindow::afterAnimating, this, &QskWindow::enforceSkin );
198}
199
200QskWindow::QskWindow( QQuickRenderControl* renderControl, QWindow* parent )
201 : QskWindow( parent )
202{
203 Q_D( QskWindow );
204
205 d->renderControl = renderControl;
206 d->init( this, renderControl );
207}
208
209QskWindow::~QskWindow()
210{
211}
212
213void QskWindow::setScreen( const QString& name )
214{
215 if ( !name.isEmpty() )
216 {
217 const auto screens = QGuiApplication::screens();
218 for ( auto screen : screens )
219 {
220 if ( screen->name() == name )
221 {
222 setScreen( screen );
223 return;
224 }
225 }
226 }
227
228 setScreen( QGuiApplication::primaryScreen() );
229}
230
231void QskWindow::resizeF( const QSizeF& size )
232{
233 const int w = static_cast< int >( qCeil( size.width() ) );
234 const int h = static_cast< int >( qCeil( size.height() ) );
235
236 resize( w, h );
237}
238
239bool QskWindow::deleteOnClose() const
240{
241 Q_D( const QskWindow );
242 return d->deleteOnClose;
243}
244
245void QskWindow::setDeleteOnClose( bool on )
246{
247 Q_D( QskWindow );
248
249 if ( on != d->deleteOnClose )
250 {
251 d->deleteOnClose = on;
252 Q_EMIT deleteOnCloseChanged();
253 }
254}
255
256void QskWindow::setAutoLayoutChildren( bool on )
257{
258 Q_D( QskWindow );
259
260 if ( on != d->autoLayoutChildren )
261 {
262 d->autoLayoutChildren = on;
263 if ( on )
264 qskSendEventTo( this, QEvent::LayoutRequest );
265
266 Q_EMIT autoLayoutChildrenChanged();
267 }
268}
269
270bool QskWindow::autoLayoutChildren() const
271{
272 Q_D( const QskWindow );
273 return d->autoLayoutChildren;
274}
275
276void QskWindow::addItem( QQuickItem* item )
277{
278 if ( item == nullptr )
279 return;
280
281 item->setParent( contentItem() );
282 item->setParentItem( contentItem() );
283}
284
285void QskWindow::polishItems()
286{
287 Q_D( QskWindow );
288 d->polishItems();
289}
290
291bool QskWindow::event( QEvent* event )
292{
293 /*
294 Qt/Quick is lacking a flag like Qt::WA_NoMousePropagation, so
295 the only way to stop propagating of input events is to accept
296 the event. Then the window can't decide anymore if someone was
297 interested and when to do some fallback handling.
298 To improve the situation we add an extra flag in QskWindow,
299 while the right class to solve this shortcoming would be QQuickWindow.
300 */
301
302 Q_D( QskWindow );
303 d->eventAcceptance = EventProcessed;
304
305 switch ( event->type() )
306 {
307 case QEvent::Show:
308 {
309 if ( !d->showedOnce )
310 {
311 d->showedOnce = true;
312
313 /*
314 When a window has a platform window its size is taken
315 from the platform window - otherwise from d->geometry.
316 A top level window that has not been shown does not have
317 a platform window yet and therefore any size set before calling
318 QWindow::show() is stored there.
319
320 Starting with Qt 6.5 an initial default size is set to the platform
321 window ( at least QXcbWindow ) before the Show event is sent
322 and we can't rely on QWindow::size() anymore. But even if d->geometry
323 is not used anymore we can initialize depending on it.
324 */
325
326 if ( d->geometry.size().isEmpty() )
327 {
328 QSize sz = sizeConstraint();
329 if ( !sz.isEmpty() )
330 {
331 sz = sz.expandedTo( minimumSize() );
332 sz = sz.boundedTo( maximumSize() );
333
334 resize( sz );
335 }
336 }
337 }
338
339 break;
340 }
341 case QEvent::LayoutRequest:
342 {
343 if ( isExposed() )
344 layoutItems();
345 break;
346 }
347 case QEvent::LocaleChange:
348 {
349 Q_EMIT localeChanged( locale() );
350 break;
351 }
352 case QEvent::Close:
353 {
354 if ( event->isAccepted() )
355 {
356 if ( d->deleteOnClose )
357 deleteLater();
358 }
359 break;
360 }
361 case QEvent::UpdateRequest:
362 {
363#ifdef QSK_DEBUG_RENDER_TIMING
364 if ( logTiming().isDebugEnabled() )
365 {
366 if ( !d->renderInterval.isValid() )
367 d->renderInterval.start();
368
369 qCDebug( logTiming() ) << "update timer - elapsed"
370 << d->renderInterval.restart() << objectName();
371 }
372#endif
373 break;
374 }
375
376 default:
377 break;
378 }
379
380 return Inherited::event( event );
381}
382
383void QskWindow::keyPressEvent( QKeyEvent* event )
384{
385 if ( qskFocusChainIncrement( event ) != 0 )
386 {
387 auto focusItem = activeFocusItem();
388 if ( focusItem == nullptr || focusItem == contentItem() )
389 {
390 /*
391 The Qt/Quick implementation for navigating along the
392 focus tab chain gives unsufficient results, when the
393 starting point is the root item. In this specific
394 situation we also have to include all items being
395 tab fences into consideration.
396
397 In certain situations Qt/Quick gets even stuck in a non
398 terminating loop: see Qt-Bug 65943
399
400 So we better block the focus navigation and find the
401 next focus item on our own.
402 */
403 ensureFocus( Qt::TabFocusReason );
404 event->accept();
405
406 return;
407 }
408 }
409
410 Inherited::keyPressEvent( event );
411}
412
413void QskWindow::keyReleaseEvent( QKeyEvent* event )
414{
415 Inherited::keyReleaseEvent( event );
416}
417
418void QskWindow::exposeEvent( QExposeEvent* event )
419{
420#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
421 if ( qskRenderingHardwareInterface( this ) )
422 {
423 /*
424 Actually our code supports Qt5 RHI ( f9c08c34fb2cc64546bbe6ce9d359f416e934961 ),
425 but Qt5 does not come with the qsb tool out of the box. Then we run into
426 problems with compiling the shader code from the makefiles.
427 But why should anyone use the experimental Qt5 RHI implementations
428 instead of using Qt6 ...
429 */
430 qFatal( "the experimental Qt5 RHI implementation is not supported:\n"
431 "\tuse Qt6 or run the default OpenGL backend." );
432 }
433#endif
434 ensureFocus( Qt::OtherFocusReason );
435 layoutItems();
436
437 Inherited::exposeEvent( event );
438}
439
440void QskWindow::resizeEvent( QResizeEvent* event )
441{
442 auto rootItem = contentItem();
443
444 const auto oldRect = qskItemGeometry( rootItem );
445 Inherited::resizeEvent( event );
446
447 const auto newRect = qskItemGeometry( rootItem );
448 if ( newRect != oldRect )
449 {
450 QskGeometryChangeEvent ev( newRect, oldRect );
451 QCoreApplication::sendEvent( rootItem, &ev );
452 }
453
454 if ( isExposed() )
455 layoutItems();
456}
457
458QLocale QskWindow::locale() const
459{
460 Q_D( const QskWindow );
461 return d->locale;
462}
463
464void QskWindow::setLocale( const QLocale& locale )
465{
466 Q_D( QskWindow );
467
468 d->explicitLocale = true;
469
470 if ( d->locale != locale )
471 {
472 d->locale = locale;
473 qskSendEventTo( this, QEvent::LocaleChange );
474 qskInheritLocale( this, locale );
475 }
476}
477
478void QskWindow::resetLocale()
479{
480 Q_D( QskWindow );
481
482 d->explicitLocale = false;
483 qskResolveLocale( this );
484}
485
486QSK_HIDDEN_EXTERNAL_BEGIN
487
488bool qskInheritLocale( QskWindow* window, const QLocale& locale )
489{
490 auto d = static_cast< QskWindowPrivate* >( QQuickWindowPrivate::get( window ) );
491
492 if ( d->explicitLocale || d->locale == locale )
493 return false;
494
495 d->locale = locale;
496 qskSendEventTo( window, QEvent::LocaleChange );
497
498 return true;
499}
500
501QSK_HIDDEN_EXTERNAL_END
502
503static void qskResolveLocale( QskWindow* window )
504{
505 auto d = static_cast< QskWindowPrivate* >( QQuickWindowPrivate::get( window ) );
506
507 const auto locale = qskInheritedLocale( window );
508
509 if ( d->locale != locale )
510 {
511 d->locale = locale;
512 qskSendEventTo( window, QEvent::LocaleChange );
513
514 qskInheritLocale( window, locale );
515 }
516}
517
518void QskWindow::setPreferredSize( const QSize& size )
519{
520 Q_D( QskWindow );
521 d->preferredSize = size;
522}
523
524QSize QskWindow::preferredSize() const
525{
526 Q_D( const QskWindow );
527 return d->preferredSize;
528}
529
530QSize QskWindow::sizeConstraint() const
531{
532 Q_D( const QskWindow );
533
534 QSizeF constraint = d->preferredSize;
535
536 if ( !constraint.isValid() )
537 {
538 const bool doWidth = constraint.width() < 0;
539 const bool doHeight = constraint.height() < 0;
540
541 const auto children = contentItem()->childItems();
542 for ( auto child : children )
543 {
544 if ( qskIsVisibleToLayout( child ) )
545 {
546 const auto size = qskSizeConstraint( child, Qt::PreferredSize );
547
548 if ( doWidth )
549 constraint.setWidth( qMax( constraint.width(), size.width() ) );
550
551 if ( doHeight )
552 constraint.setHeight( qMax( constraint.height(), size.height() ) );
553 }
554 }
555 }
556
557 // QWindow geometries are in integers
558
559 return QSize( qskToIntegerConstraint( constraint.width() ),
560 qskToIntegerConstraint( constraint.height() ) );
561}
562
563void QskWindow::setFixedSize( const QSize& size )
564{
565 // ????
566 setMinimumSize( size );
567 setMaximumSize( size );
568}
569
570void QskWindow::layoutItems()
571{
572 Q_D( QskWindow );
573
574 if ( !d->autoLayoutChildren )
575 return;
576
577 const QRectF rect = qskItemGeometry( contentItem() );
578
579 const auto children = contentItem()->childItems();
580 for ( auto child : children )
581 {
582 if ( qskIsAdjustableByLayout( child ) )
583 {
584 const auto r = qskConstrainedItemRect( child, rect );
585 qskSetItemGeometry( child, r );
586 }
587 }
588}
589
590void QskWindow::ensureFocus( Qt::FocusReason reason )
591{
592 auto focusItem = contentItem()->scopedFocusItem();
593
594 if ( focusItem == nullptr )
595 {
596 focusItem = qskDefaultFocusItem( this );
597 if ( focusItem )
598 focusItem->setFocus( true, reason );
599 }
600}
601
602void QskWindow::setCustomRenderMode( const char* mode )
603{
604 class RenderJob final : public QRunnable
605 {
606 public:
607 RenderJob( QQuickWindow* window, const QByteArray& mode )
608 : m_window( window )
609 , m_mode( mode )
610 {
611 }
612
613 void run() override
614 {
615 qskSetVisualizationMode( m_window, m_mode );
616
617 auto d = QQuickWindowPrivate::get( m_window );
618 if ( d->renderer )
619 {
620 delete d->renderer->rootNode();
621 delete d->renderer;
622 d->renderer = nullptr;
623
624 QMetaObject::invokeMethod( m_window, "update" );
625 }
626 }
627
628 private:
629 QQuickWindow* m_window;
630 const QByteArray m_mode;
631 };
632
633 const QByteArray newMode( mode );
634
635 const auto oldMode = qskVisualizationMode( this );
636
637 if ( newMode != oldMode )
638 {
639 /*
640 batch renderer uses an optimized memory allocation strategy,
641 that is disabled, when a customRenderMode is enabled.
642 This seems to be the reason for crashes, when changing the mode
643 at runtime. The code above tries to get rid of all memory
644 from the previous allocation strategy by deleting the renderer
645 after swapping.
646 */
647
648 if ( newMode.isEmpty() != oldMode.isEmpty() )
649 scheduleRenderJob( new RenderJob( this, newMode ), AfterSwapStage );
650 else
651 qskSetVisualizationMode( this, newMode );
652
653 update();
654 }
655}
656
657const char* QskWindow::customRenderMode() const
658{
659 return qskVisualizationMode( this );
660}
661
662void QskWindow::enforceSkin()
663{
664 if ( !qskEnforcedSkin )
665 {
666 // usually the skin is set in the application startup code, but if not
667 // let's create a default skin on the GUI thread - whatever it is
668 // good for.
669
670 ( void ) qskSkinManager->skin();
671 qskEnforcedSkin = true;
672 }
673
674 disconnect( this, &QQuickWindow::afterAnimating, this, &QskWindow::enforceSkin );
675}
676
677void QskWindow::setEventAcceptance( EventAcceptance acceptance )
678{
679 d_func()->eventAcceptance = acceptance;
680}
681
682QskWindow::EventAcceptance QskWindow::eventAcceptance() const
683{
684 return d_func()->eventAcceptance;
685}
686
687void QskWindow::setSkin( const QString& skinName )
688{
689 // we should compare the skinName with the previous one
690 auto skin = QskSkinManager::instance()->createSkin( skinName );
691 setSkin( skin );
692}
693
694void QskWindow::setSkin( QskSkin* skin )
695{
696 Q_D( QskWindow );
697
698 if ( d->skin == skin )
699 return;
700
701 if ( d->skin )
702 {
703 if ( d->skin->parent() == this )
704 delete d->skin;
705 }
706
707 if ( skin && skin->parent() == nullptr )
708 skin->setParent( this );
709
710 d->skin = skin;
711}
712
713QskSkin* QskWindow::skin() const
714{
715 return d_func()->skin;
716}
717
718QskSkin* qskEffectiveSkin( const QQuickWindow* window )
719{
720 if ( auto w = qobject_cast< const QskWindow* >( window ) )
721 {
722 if ( auto skin = w->skin() )
723 return skin;
724 }
725
726 return qskSkinManager->skin();
727}
728
729#include "moc_QskWindow.cpp"