QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskFocusIndicator.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskFocusIndicator.h"
7#include "QskAspect.h"
8#include "QskAnimationHint.h"
9#include "QskEvent.h"
10#include "QskQuick.h"
11#include "QskInternalMacros.h"
12
13#include <qpointer.h>
14#include <qquickwindow.h>
15#include <qbasictimer.h>
16
17QSK_QT_PRIVATE_BEGIN
18#include <private/qquickitem_p.h>
19QSK_QT_PRIVATE_END
20
21QSK_SUBCONTROL( QskFocusIndicator, Panel )
22
23static inline QRectF qskFocusIndicatorRect( const QQuickItem* item )
24{
25 if ( auto control = qskControlCast( item ) )
26 return control->focusIndicatorRect();
27
28 const QVariant v = item->property( "focusIndicatorRect" );
29 if ( v.canConvert< QRectF >() )
30 return v.toRectF();
31
32 return qskItemRect( item );
33}
34
35static inline QRectF qskFocusIndicatorClipRect( const QQuickItem* item )
36{
37 QRectF rect( 0.0, 0.0, -1.0, -1.0 );
38
39 if ( item )
40 {
41 if ( auto control = qskControlCast( item ) )
42 rect = control->focusIndicatorClipRect();
43 else
44 rect = item->clipRect();
45 }
46
47 return rect;
48}
49
50static inline QskAspect::Section qskItemSection( const QQuickItem* item )
51{
52 if ( auto control = qskControlCast( item ) )
53 return control->section();
54
55 return QskAspect::Body;
56}
57
58static inline bool qskIsEnablingKey( const QKeyEvent* event )
59{
60 // what keys do we want have here ???
61 return qskIsButtonPressKey( event ) || qskFocusChainIncrement( event );
62}
63
64static inline bool qskIsEditing( const QQuickWindow* window )
65{
66 if ( !QGuiApplication::inputMethod()->isVisible() )
67 {
68 if ( auto item = window->activeFocusItem() )
69 {
70 const QVariant v = item->property( "editing" );
71 if ( v.value< bool >() )
72 return true;
73 }
74 }
75
76 return false;
77}
78
79class QskFocusIndicator::PrivateData
80{
81 public:
82 void resetConnections()
83 {
84 for ( const auto& connection : std::as_const( connections ) )
85 QObject::disconnect( connection );
86
87 connections.clear();
88 }
89
90 inline bool isAutoDisabling() const { return duration > 0; }
91 inline bool isAutoEnabling() const { return false; }
92
93 QPointer< QQuickItem > clippingItem;
94 QVector< QMetaObject::Connection > connections;
95
96 int duration = 0;
97 QBasicTimer timer;
98
99 bool blockAutoRepeatKeyEvents = false;
100};
101
102QskFocusIndicator::QskFocusIndicator( QQuickItem* parent )
103 : Inherited( parent ) // parentItem() might change, but parent() stays
104 , m_data( new PrivateData() )
105{
106 setPlacementPolicy( QskPlacementPolicy::Ignore );
107 connectWindow( window(), true );
108
109 setDuration( 4500 );
110}
111
112QskFocusIndicator::~QskFocusIndicator()
113{
114}
115
116void QskFocusIndicator::setDuration( int ms )
117{
118 ms = std::max( ms, 0 );
119 if ( ms == m_data->duration )
120 return;
121
122 m_data->duration = ms;
123
124 if ( m_data->isAutoDisabling() )
125 {
126 if ( auto w = window() )
127 w->installEventFilter( this );
128
129 if ( isEnabled() )
130 {
131 if ( isInitiallyPainted() )
132 m_data->timer.start( m_data->duration, this );
133 else
134 setEnabled( false );
135 }
136
137 connect( this, &QQuickItem::enabledChanged,
138 this, &QskFocusIndicator::resetTimer );
139 }
140 else
141 {
142 if ( auto w = window() )
143 w->removeEventFilter( this );
144
145 setEnabled( true );
146
147 disconnect( this, &QQuickItem::enabledChanged,
148 this, &QskFocusIndicator::resetTimer );
149 }
150
151 Q_EMIT durationChanged( ms );
152}
153
154int QskFocusIndicator::duration() const
155{
156 return m_data->duration;
157}
158
159void QskFocusIndicator::maybeEnable( bool on )
160{
161 if ( !m_data->isAutoEnabling() )
162 return;
163
164 if ( on )
165 {
166 if ( auto w = window() )
167 {
168 if ( w->isExposed() && w->isActive() )
169 setEnabled( true );
170 }
171 }
172 else
173 {
174 setEnabled( false );
175 }
176}
177
178void QskFocusIndicator::resetTimer()
179{
180 if ( m_data->isAutoDisabling() )
181 {
182 if ( isEnabled() )
183 {
184 const auto hint = animationHint( Panel | QskAspect::Color );
185 m_data->timer.start( m_data->duration + hint.duration, this );
186 }
187 else
188 {
189 m_data->timer.stop();
190 }
191 }
192}
193
194bool QskFocusIndicator::eventFilter( QObject* object, QEvent* event )
195{
196 if( ( object != window() ) || !m_data->isAutoDisabling() )
197 return Inherited::eventFilter( object, event );
198
199 switch( static_cast< int >( event->type() ) )
200 {
201 case QEvent::KeyPress:
202 case QEvent::KeyRelease:
203 case QEvent::ShortcutOverride:
204 {
205 if ( qskIsEditing( window() ) )
206 return false;
207
208 if ( m_data->timer.isActive() )
209 {
210 // renew the exposed period
211 m_data->timer.start( m_data->duration, this );
212 }
213 break;
214 }
215 }
216
217 switch( static_cast< int >( event->type() ) )
218 {
219 case QEvent::KeyPress:
220 {
221 const auto keyEvent = static_cast< QKeyEvent* >( event );
222
223 if( keyEvent->isAutoRepeat() && m_data->blockAutoRepeatKeyEvents )
224 {
225 /*
226 We swallow all auto repeated events to avoid running along
227 the tab chain by accident.
228 */
229 return true;
230 }
231
232 if ( !isEnabled() && qskIsEnablingKey( keyEvent ) )
233 {
234 setEnabled( true );
235 m_data->blockAutoRepeatKeyEvents = true;
236 return true;
237 }
238
239 m_data->blockAutoRepeatKeyEvents = false;
240 break;
241 }
242
243 case QEvent::KeyRelease:
244 {
245 if( m_data->blockAutoRepeatKeyEvents )
246 {
247 if( !static_cast< QKeyEvent* >( event )->isAutoRepeat() )
248 m_data->blockAutoRepeatKeyEvents = false;
249
250 return true;
251 }
252
253 break;
254 }
255
256 case QEvent::Expose:
257 case QEvent::FocusIn:
258 case QEvent::FocusOut:
259 {
260 maybeEnable( event->type() != QEvent::FocusOut );
261 break;
262 }
263 }
264
265 return Inherited::eventFilter( object, event );
266}
267
268void QskFocusIndicator::timerEvent( QTimerEvent* event )
269{
270 if ( m_data->isAutoDisabling() )
271 {
272 if( event->timerId() == m_data->timer.timerId() )
273 {
274 setEnabled( false );
275 return;
276 }
277 }
278
279 Inherited::timerEvent( event );
280}
281
282bool QskFocusIndicator::contains( const QPointF& ) const
283{
284 // so that tools like Squish do not see it
285 return false;
286}
287
288QRectF QskFocusIndicator::clipRect() const
289{
290 if ( m_data->clippingItem )
291 {
292 auto rect = qskFocusIndicatorClipRect( m_data->clippingItem );
293 rect = mapRectFromItem( m_data->clippingItem, rect );
294
295 return rect;
296 }
297
298 return Inherited::clipRect();
299}
300
301void QskFocusIndicator::onFocusItemGeometryChanged()
302{
303 updateFocusFrame();
304}
305
306void QskFocusIndicator::onWindowSizeChanged( int )
307{
308 updateFocusFrame();
309}
310
311void QskFocusIndicator::onFocusItemDestroyed()
312{
313 m_data->resetConnections();
314 setVisible( false );
315}
316
317void QskFocusIndicator::onFocusItemChanged()
318{
319 m_data->resetConnections();
320
321 if ( !( window() && window()->contentItem() ) )
322 return;
323
324 // We want to be on top, but do we cover all corner cases ???
325 setParentItem( window()->contentItem() );
326 setZ( 10e-6 );
327
328 const auto focusItem = window()->activeFocusItem();
329 QQuickItem* clippingItem = nullptr;
330
331 if ( focusItem && ( focusItem != window()->contentItem() ) )
332 {
333 setSection( qskItemSection( focusItem ) );
334
335 auto item = focusItem;
336 m_data->connections += connectItem( item );
337
338 while ( auto itemParent = item->parentItem() )
339 {
340 m_data->connections += connectItem( itemParent );
341
342 if ( clippingItem == nullptr && itemParent->clip() )
343 clippingItem = itemParent;
344
345 item = itemParent;
346 }
347 }
348
349 m_data->clippingItem = clippingItem;
350 updateFocusFrame();
351}
352
353void QskFocusIndicator::updateFocusFrame()
354{
355 QRectF r = focusRect();
356 setVisible( !r.isEmpty() );
357
358 if ( !r.isEmpty() )
359 {
360 r = r.marginsAdded( paddingHint( Panel ) );
361
362 if ( auto w = window() )
363 {
364 QRectF clipRect( 0, 0, w->width(), w->height() );
365 clipRect = parentItem()->mapRectFromScene( clipRect );
366
367 r = r.intersected( clipRect );
368 }
369
370 setGeometry( r );
371
372 const auto clipRect = qskFocusIndicatorClipRect( m_data->clippingItem );
373 setClip( !clipRect.isEmpty() );
374
375 if ( clip() )
376 {
377 /*
378 The clip node is updated on QQuickItemPrivate::Size
379 So we need to set it here even in situations, where
380 the size did not change. For now we always trigger an
381 update of the clipNode, but we could limit it to
382 changes of the clipRect(). TODO ...
383 */
384 QQuickItemPrivate::get( this )->dirty( QQuickItemPrivate::Size );
385 }
386 }
387
388 update();
389}
390
391QRectF QskFocusIndicator::focusRect() const
392{
393 if ( window() && parentItem() )
394 {
395 const auto item = window()->activeFocusItem();
396
397 if ( item && ( item != this ) && item->isVisible() &&
398 ( item != window()->contentItem() ) )
399 {
400 const auto rect = qskFocusIndicatorRect( item );
401 return parentItem()->mapRectFromItem( item, rect );
402 }
403 }
404
405 return QRectF();
406}
407
409{
411
412 connectWindow( event->oldWindow(), false );
413 connectWindow( event->window(), true );
414
415 onFocusItemChanged();
416
417 if ( m_data->isAutoDisabling() )
418 {
419 if ( auto w = event->oldWindow() )
420 w->removeEventFilter( this );
421
422 if( auto w = event->window() )
423 {
424 w->installEventFilter( this );
425 maybeEnable( true );
426 }
427 }
428}
429
430void QskFocusIndicator::connectWindow( const QQuickWindow* window, bool on )
431{
432 if ( window == nullptr )
433 return;
434
435 if ( on )
436 {
437 connect( window, &QQuickWindow::activeFocusItemChanged,
438 this, &QskFocusIndicator::onFocusItemChanged );
439
440 connect( window, &QQuickWindow::widthChanged,
441 this, &QskFocusIndicator::onWindowSizeChanged );
442
443 connect( window, &QQuickWindow::heightChanged,
444 this, &QskFocusIndicator::onWindowSizeChanged );
445 }
446 else
447 {
448 disconnect( window, &QQuickWindow::activeFocusItemChanged,
449 this, &QskFocusIndicator::onFocusItemChanged );
450
451 disconnect( window, &QQuickWindow::widthChanged,
452 this, &QskFocusIndicator::onWindowSizeChanged );
453
454 disconnect( window, &QQuickWindow::heightChanged,
455 this, &QskFocusIndicator::onWindowSizeChanged );
456 }
457}
458
459QVector< QMetaObject::Connection > QskFocusIndicator::connectItem( const QQuickItem* sender )
460{
461 QVector< QMetaObject::Connection > c;
462 c.reserve( 7 );
463
464 c += QObject::connect( sender, &QObject::destroyed,
465 this, &QskFocusIndicator::onFocusItemDestroyed );
466
467 const auto method = &QskFocusIndicator::onFocusItemGeometryChanged;
468
469 c += QObject::connect( sender, &QQuickItem::xChanged, this, method );
470 c += QObject::connect( sender, &QQuickItem::yChanged, this, method );
471 c += QObject::connect( sender, &QQuickItem::widthChanged, this, method );
472 c += QObject::connect( sender, &QQuickItem::heightChanged, this, method );
473 c += QObject::connect( sender, &QQuickItem::visibleChanged, this, method );
474
475 if ( const auto control = qskControlCast( sender ) )
476 {
477 c += QObject::connect( control, &QskControl::focusIndicatorRectChanged, this, method );
478 }
479 else
480 {
481 if ( sender->metaObject()->indexOfSignal( "focusIndicatorRectChanged()" ) >= 0 )
482 {
483 c += QObject::connect( sender, SIGNAL(focusIndicatorRectChanged()),
484 this, SLOT(onFocusItemGeometryChanged()) );
485 }
486 }
487
488 return c;
489}
490
491#include "moc_QskFocusIndicator.cpp"
void focusIndicatorRectChanged()
void windowChangeEvent(QskWindowChangeEvent *) override
QRectF rect
Definition QskItem.h:22
bool isInitiallyPainted() const
Definition QskItem.cpp:584
virtual void windowChangeEvent(QskWindowChangeEvent *)
Definition QskItem.cpp:862
void setGeometry(qreal x, qreal y, qreal width, qreal height)
Definition QskItem.cpp:332
QMarginsF paddingHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a padding hint.
QskAnimationHint animationHint(QskAspect, QskSkinHintStatus *=nullptr) const