QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskPopup.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskPopup.h"
7#include "QskAspect.h"
8#include "QskInputGrabber.h"
9#include "QskQuick.h"
10#include "QskWindow.h"
11#include "QskEvent.h"
12#include "QskPlatform.h"
13#include "QskHintAnimator.h"
14#include "QskInternalMacros.h"
15
16#include <qpa/qplatformintegration.h>
17
18QSK_QT_PRIVATE_BEGIN
19#include <private/qquickwindow_p.h>
20#include <private/qquickitem_p.h>
21QSK_QT_PRIVATE_END
22
23QSK_SUBCONTROL( QskPopup, Overlay )
24QSK_SYSTEM_STATE( QskPopup, Closed, QskAspect::FirstSystemState << 1 )
25
26static void qskSetFocus( QQuickItem* item, bool on )
27{
28 if ( item->window() == nullptr )
29 return;
30
31 /*
32 For unknown reasons Qt::PopupFocusReason is blocked inside of
33 QQuickItem::setFocus. So let's bypass it calling
34 QQuickWindowPrivate::setFocusInScope/clearFocusInScope directly,
35 */
36
37 if ( const auto scope = qskNearestFocusScope( item ) )
38 {
39#if QT_VERSION >= QT_VERSION_CHECK( 6, 1, 0 )
40 auto dw = QQuickItemPrivate::get( item )->deliveryAgentPrivate();
41#else
42 auto dw = QQuickWindowPrivate::get( item->window() );
43#endif
44 if ( on )
45 {
46 dw->setFocusInScope( scope, item, Qt::PopupFocusReason );
47 }
48 else
49 {
50 // to avoid an assertion we have to check if we have the subFocus
51 if ( QQuickItemPrivate::get( scope )->subFocusItem == item )
52 dw->clearFocusInScope( scope, item, Qt::PopupFocusReason );
53 }
54 }
55}
56
57static inline void qskSendPopupEvent(
58 QQuickWindow* window, QskPopup* popup, bool on )
59{
60 if ( window )
61 {
62 const auto type = on ? QskEvent::PopupAdded : QskEvent::PopupRemoved;
63
64 QskPopupEvent event( type, popup );
65 QCoreApplication::sendEvent( window, &event );
66 }
67}
68
69static bool qskReplayMousePress()
70{
71 if ( const auto pf = qskPlatformIntegration() )
72 {
73 const auto styleHint = QPlatformIntegration::ReplayMousePressOutsidePopup;
74 return pf->styleHint( styleHint ).toBool();
75 }
76
77 return false;
78}
79
80static inline QskAspect qskEffectiveFadingAspect( const QskPopup* popup )
81{
82 auto aspect = popup->fadingAspect();
83 aspect.setSubcontrol( popup->effectiveSubcontrol( aspect.subControl() ) );
84
85 return aspect;
86}
87
88static void qskStartFading( QskPopup* popup, bool on )
89{
90 const auto aspect = qskEffectiveFadingAspect( popup );
91
92 auto hint = popup->animationHint( aspect );
93
94 if ( hint.isValid() )
95 {
96 hint.updateFlags = QskAnimationHint::UpdatePolish | QskAnimationHint::UpdateNode;
97
98 const qreal from = on ? 0.0 : 1.0;
99 const qreal to = on ? 1.0 : 0.0;
100
101 popup->startTransition( aspect, hint, from, to );
102 }
103}
104
105namespace
106{
107 class InputGrabber final : public QskInputGrabber
108 {
109 using Inherited = QskInputGrabber;
110
111 public:
112 InputGrabber( QskPopup* parent )
113 : Inherited( parent )
114 {
115 }
116
117 protected:
118 bool event( QEvent* event ) override
119 {
120 bool ok = Inherited::event( event );
121
122 const int eventType = event->type();
123
124 if ( eventType == QEvent::MouseButtonPress )
125 {
126 if ( auto popup = static_cast< QskPopup* >( parentItem() ) )
127 {
128 if ( event->isAccepted() &&
129 ( popup->popupFlags() & QskPopup::CloseOnPressOutside ) )
130 {
131 popup->close();
132
133 if ( qskReplayMousePress() )
134 {
135 // TODO ...
136 }
137 }
138 }
139 }
140 else if ( eventType == QskEvent::GeometryChange )
141 {
142 if ( auto popup = static_cast< QskPopup* >( parentItem() ) )
143 {
144 if ( popup->hasOverlay() )
145 popup->update();
146 }
147 }
148
149 return ok;
150 }
151 };
152}
153
154class QskPopup::PrivateData
155{
156 public:
157 PrivateData()
158 : flags( 0 )
159 , isModal( false )
160 , autoGrabFocus( true )
161 , handoverFocus( true )
162 {
163 }
164
165 InputGrabber* inputGrabber = nullptr;
166
167 uint priority = 0;
168
169 int flags : 4;
170 bool isModal : 1;
171
172 const bool autoGrabFocus : 1;
173 const bool handoverFocus : 1;
174};
175
176QskPopup::QskPopup( QQuickItem* parent )
177 : Inherited( parent )
178 , m_data( new PrivateData() )
179{
180 // initially the popup is closed and invisible
181 Inherited::setVisible( false );
182 setSkinStateFlag( QskPopup::Closed );
183
184 /*
185 We need to stop event propagation.
186
187 Unfortunatly derived classes can't use setAcceptedMouseButtons anymore.
188 Need to think about a solution TODO ...
189 */
190
191 setAcceptedMouseButtons( Qt::AllButtons );
192 setWheelEnabled( true );
193
194 // we don't want to be resized by layout code
195 setPlacementPolicy( QskPlacementPolicy::Ignore );
196
197 setFlag( ItemIsFocusScope, true );
198 setTabFence( true );
199 setFocusPolicy( Qt::StrongFocus );
200
201 /*
202 sending a notification, that can be used to register popups
203 for some sort of popup/window management
204 */
205 qskSendPopupEvent( window(), this, true );
206}
207
208QskPopup::~QskPopup()
209{
210 qskSendPopupEvent( window(), this, false );
211}
212
213void QskPopup::open()
214{
215 setOpen( true );
216}
217
218void QskPopup::close()
219{
220 setOpen( false );
221}
222
223void QskPopup::toggle()
224{
225 setOpen( !isOpen() );
226}
227
228void QskPopup::setOpen( bool on )
229{
230 if ( on == isOpen() )
231 return;
232
233 if ( on )
234 QskControl::setVisible( on );
235
236 setSkinStateFlag( QskPopup::Closed, !on );
237
238 Q_EMIT openChanged( on );
239
240 if ( on )
241 Q_EMIT opened();
242 else
243 Q_EMIT closed();
244
245 if ( !on )
246 grabFocus( false );
247
248 qskStartFading( this, on );
249
250 if ( isFading() )
251 {
252 Q_EMIT fadingChanged( true );
253 }
254 else
255 {
256 if ( !on )
257 {
258 Inherited::setVisible( false );
259
260 if ( testPopupFlag( QskPopup::DeleteOnClose ) )
261 deleteLater();
262 }
263 }
264}
265
266bool QskPopup::isOpen() const
267{
268 return !hasSkinState( QskPopup::Closed );
269}
270
271QskAspect QskPopup::fadingAspect() const
272{
273 return QskAspect();
274}
275
276bool QskPopup::isFading() const
277{
278 const auto aspect = qskEffectiveFadingAspect( this );
279 return runningHintAnimator( aspect ) != nullptr;
280}
281
282qreal QskPopup::fadingFactor() const
283{
284 const auto aspect = qskEffectiveFadingAspect( this );
285 if ( auto animator = runningHintAnimator( aspect ) )
286 return animator->currentValue().value< qreal >();
287
288 return isOpen() ? 1.0 : 0.0;
289}
290
291QRectF QskPopup::overlayRect() const
292{
293 if ( hasOverlay() && m_data->inputGrabber )
294 return m_data->inputGrabber->grabberRect();
295
296 return QRectF();
297}
298
299void QskPopup::updateInputGrabber()
300{
301 if ( parentItem() && isVisible() &&
302 ( isModal() || testPopupFlag( CloseOnPressOutside ) ) )
303 {
304 if ( m_data->inputGrabber == nullptr )
305 {
306 const auto children = childItems();
307 m_data->inputGrabber = new InputGrabber( this );
308
309 if ( !children.isEmpty() )
310 {
311 /*
312 Even if the input grabber has no content it has an effect
313 on QQuickItem::childAt. Also tools like Squish struggle with
314 sorting out items without content.
315 So let's better move the grabber to the beginning.
316 */
317 m_data->inputGrabber->stackBefore( children.first() );
318 }
319 }
320 }
321 else
322 {
323 if ( m_data->inputGrabber )
324 {
325 /*
326 In QQuickWindowPrivate::deliverPressOrReleaseEvent ( 5.12 )
327 might crash, when we delete the grabber as a result of a
328 mouse event somewehere below the popup.
329 */
330 m_data->inputGrabber->setParentItem( nullptr );
331 m_data->inputGrabber->setParent( nullptr );
332 m_data->inputGrabber->deleteLater();
333
334 m_data->inputGrabber = nullptr;
335 }
336 }
337}
338
340{
341 if ( isVisible() && !isInitiallyPainted() )
342 {
343 /*
344 Usually we suppress transitions, when a control has never been
345 painted before as there is no valid starting point. Popups are
346 different as we want to have smooth fade/slide appearances.
347 */
348 if ( ( aspect.value() == 0 ) )
349 return true;
350
351 if ( aspect.subControl() == qskEffectiveFadingAspect( this ).subControl() )
352 return true;
353
354 if ( aspect.subControl() == effectiveSubcontrol( QskPopup::Overlay ) )
355 return true;
356 }
357
358 return Inherited::isTransitionAccepted( aspect );
359}
360
361void QskPopup::setPriority( uint priority )
362{
363 if ( m_data->priority != priority )
364 {
365 m_data->priority = priority;
366 Q_EMIT priorityChanged( priority );
367 }
368}
369
370uint QskPopup::priority() const
371{
372 return m_data->priority;
373}
374
375void QskPopup::setModal( bool on )
376{
377 if ( on == m_data->isModal )
378 return;
379
380 m_data->isModal = on;
381 updateInputGrabber();
382
383 Q_EMIT modalChanged( on );
384}
385
386bool QskPopup::isModal() const
387{
388 return m_data->isModal;
389}
390
391void QskPopup::setPopupFlags( PopupFlags flags )
392{
393 const auto newFlags = static_cast< int >( flags );
394
395 if ( newFlags != m_data->flags )
396 {
397 m_data->flags = newFlags;
398 updateInputGrabber();
399 }
400}
401
402QskPopup::PopupFlags QskPopup::popupFlags() const
403{
404 return static_cast< PopupFlags >( m_data->flags );
405}
406
407void QskPopup::setPopupFlag( PopupFlag flag, bool on )
408{
409 auto flags = m_data->flags;
410
411 if ( on )
412 flags |= flag;
413 else
414 flags &= ~flag;
415
416 setPopupFlags( PopupFlags( flags ) );
417}
418
419bool QskPopup::testPopupFlag( PopupFlag flag ) const
420{
421 return m_data->flags & flag;
422}
423
424void QskPopup::setOverlay( bool on )
425{
426 if ( setFlagHint( Overlay | QskAspect::Style, on ) )
427 Q_EMIT overlayChanged( on );
428}
429
430void QskPopup::resetOverlay()
431{
432 if ( resetSkinHint( Overlay | QskAspect::Style ) )
433 Q_EMIT overlayChanged( hasOverlay() );
434}
435
436bool QskPopup::hasOverlay() const
437{
438 return flagHint< bool >( QskPopup::Overlay | QskAspect::Style, true );
439}
440
441void QskPopup::grabFocus( bool on )
442{
443 if ( on == hasFocus() )
444 return;
445
446 if ( on )
447 {
448 qskSetFocus( this, true );
449 }
450 else
451 {
452 QQuickItem* successor = nullptr;
453
454 if ( m_data->handoverFocus )
455 {
456 /*
457 Qt/Quick does not handover the focus to another item,
458 when the active focus gets lost. For the situation of
459 a popup being closed we try to do it.
460 */
461 successor = focusSuccessor();
462 }
463
464 if ( successor )
465 qskSetFocus( successor, true );
466
467 if ( hasFocus() )
468 qskSetFocus( this, false );
469 }
470}
471
472bool QskPopup::event( QEvent* event )
473{
474 bool ok = Inherited::event( event );
475
476 switch ( static_cast< int >( event->type() ) )
477 {
478 case QEvent::KeyPress:
479 case QEvent::KeyRelease:
480
481 case QEvent::Wheel:
482
483 case QEvent::MouseButtonPress:
484 case QEvent::MouseMove:
485 case QEvent::MouseButtonRelease:
486
487 case QEvent::HoverEnter:
488 case QEvent::HoverLeave:
489 {
490 // swallow the event
491 event->accept();
492 if ( auto w = qobject_cast< QskWindow* >( window() ) )
493 w->setEventAcceptance( QskWindow::EventPropagationStopped );
494
495 break;
496 }
497 case QskEvent::Animator:
498 {
499 const auto animatorEvent = static_cast< QskAnimatorEvent* >( event );
500
501 if ( ( animatorEvent->state() == QskAnimatorEvent::Terminated )
502 && ( animatorEvent->aspect() == qskEffectiveFadingAspect( this ) ) )
503 {
504 if ( !isOpen() )
505 {
506 Inherited::setVisible( false );
507
508 if ( testPopupFlag( QskPopup::DeleteOnClose ) )
509 deleteLater();
510 }
511
512 Q_EMIT fadingChanged( false );
513 }
514
515 break;
516 }
517 default:
518 {
519 /*
520 Don't accept touch events otherwise we don't receive the
521 synthesized mouse events and need to handle both type of
522 events from now on.
523 But by accepting the mouse event propagation of the touch
524 events also stops.
525 */
526
527 break;
528 }
529 }
530
531 return ok;
532}
533
534void QskPopup::keyPressEvent( QKeyEvent* event )
535{
536 if ( qskIsStandardKeyInput( event, QKeySequence::Cancel ) )
537 {
538 close();
539 return;
540 }
541
542 return Inherited::keyPressEvent( event );
543}
544
545void QskPopup::focusInEvent( QFocusEvent* event )
546{
547 Inherited::focusInEvent( event );
548
549 if ( isFocusScope() && isTabFence() && ( scopedFocusItem() == nullptr ) )
550 {
551 if ( event->reason() == Qt::PopupFocusReason )
552 {
553 /*
554 When receiving the focus we need to have a focused
555 item, so that the tab focus chain has a starting point.
556
557 But we only do it when the reason is Qt::PopupFocusReason
558 as we also receive focus events during the process of reparenting
559 children and setting the focus there can leave the item tree
560 in an invalid state.
561 */
562
563 if ( auto focusItem = nextItemInFocusChain( true ) )
564 {
565 if ( !qskIsItemInDestructor( focusItem )
566 && qskIsAncestorOf( this, focusItem ) )
567 {
568 focusItem->setFocus( true );
569 }
570 }
571 }
572 }
573}
574
575void QskPopup::focusOutEvent( QFocusEvent* event )
576{
577 Inherited::focusOutEvent( event );
578}
579
580QQuickItem* QskPopup::focusSuccessor() const
581{
582 if ( const auto scope = qskNearestFocusScope( this ) )
583 {
584 const auto children = qskPaintOrderChildItems( scope );
585 for ( auto it = children.crbegin(); it != children.crend(); ++it )
586 {
587 auto child = *it;
588
589 if ( ( child != this ) && child->isFocusScope() &&
590 child->activeFocusOnTab() && child->isVisible() )
591 {
592 return child;
593 }
594 }
595 }
596
597 return nullptr;
598}
599
601{
602 if ( m_data->autoGrabFocus )
603 {
604 // What to do, when we are hidden below another popup ??
605 grabFocus( true );
606 }
607
609}
610
611void QskPopup::itemChange( QQuickItem::ItemChange change,
612 const QQuickItem::ItemChangeData& value )
613{
614 Inherited::itemChange( change, value );
615
616 if ( change == QQuickItem::ItemVisibleHasChanged )
617 {
618 updateInputGrabber();
619
620 if ( value.boolValue )
621 {
622 polish(); // so that aboutToShow is called. TODO ...
623 }
624 else
625 {
626 grabFocus( false );
627 }
628 }
629 else if ( change == QQuickItem::ItemParentHasChanged )
630 {
631 delete m_data->inputGrabber;
632 m_data->inputGrabber = nullptr;
633
634 updateInputGrabber();
635 }
636}
637
638void QskPopup::geometryChange(
639 const QRectF& newGeometry, const QRectF& oldGeometry )
640{
641 Inherited::geometryChange( newGeometry, oldGeometry );
642}
643
645{
646 qskSendPopupEvent( event->oldWindow(), this, false );
647 qskSendPopupEvent( event->window(), this, true );
648
650}
651
652int QskPopup::execPopup()
653{
654 class EventLoop : public QEventLoop
655 {
656 public:
657 EventLoop( QskPopup* popup )
658 : QEventLoop( popup )
659 {
660 /*
661 We want popup being the parent, so that the loop can be found
662 by popup->findChild< QEventLoop* >()
663 */
664
665 connect( popup, &QObject::destroyed, this, &EventLoop::reject );
666 connect( popup, &QskPopup::fadingChanged, this, &EventLoop::maybeQuit );
667 connect( popup, &QskPopup::openChanged, this, &EventLoop::maybeQuit );
668 }
669
670 private:
671 void reject()
672 {
673 setParent( nullptr );
674 QEventLoop::exit( 1 );
675 }
676
677 void maybeQuit()
678 {
679 if ( auto popup = qobject_cast< const QskPopup* >( parent() ) )
680 {
681 if ( popup->isOpen() || popup->isFading() )
682 return;
683 }
684
685 QEventLoop::exit( 0 );
686 }
687 };
688
689 if ( isOpen() || isFading() )
690 {
691 qWarning() << "QskPopup::exec: popup is already opened";
692 return -1;
693 }
694
695 if ( priority() > 0 )
696 {
697 qWarning( "QskPopup::exec for a popup with non default priority." );
698 }
699
700 open();
701
702 if ( window() == nullptr )
703 {
704 qWarning( "trying to exec a popup without window." );
705 return -1;
706 }
707
708 if ( auto mouseGrabber = window()->mouseGrabberItem() )
709 {
710 // when being called from QQuickWindow::mouseReleaseEvent
711 // the mouse grabber has not yet been released.
712
713 if( !qskIsAncestorOf( this, mouseGrabber ) )
714 qskUngrabMouse( mouseGrabber );
715 }
716
717 return EventLoop( this ).exec( QEventLoop::DialogExec );
718}
719
720#include "moc_QskPopup.cpp"
Lookup key for a QskSkinHintTable.
Definition QskAspect.h:15
@ FirstSystemState
Definition QskAspect.h:115
constexpr quint64 value() const noexcept
Definition QskAspect.h:402
constexpr Subcontrol subControl() const noexcept
Definition QskAspect.h:417
void itemChange(ItemChange, const ItemChangeData &) override
void geometryChange(const QRectF &, const QRectF &) override
virtual void aboutToShow()
Definition QskItem.cpp:1099
bool isTabFence() const
Definition QskItem.cpp:412
bool isInitiallyPainted() const
Definition QskItem.cpp:584
virtual void windowChangeEvent(QskWindowChangeEvent *)
Definition QskItem.cpp:862
void aboutToShow() override
Definition QskPopup.cpp:600
bool isTransitionAccepted(QskAspect) const override
Additional check if an transition should be started.
Definition QskPopup.cpp:339
void windowChangeEvent(QskWindowChangeEvent *) override
Definition QskPopup.cpp:644
bool resetSkinHint(QskAspect)
Remove a hint from the local hint table.
virtual bool isTransitionAccepted(QskAspect) const
Additional check if an transition should be started.
QskAnimationHint animationHint(QskAspect, QskSkinHintStatus *=nullptr) const
void setSkinStateFlag(QskAspect::State, bool on=true)
bool setFlagHint(QskAspect, int flag)
Sets a flag hint.
QskAspect::Subcontrol effectiveSubcontrol(QskAspect::Subcontrol) const