QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskMenu.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskMenu.h"
7
8#include "QskGraphicProvider.h"
9#include "QskLabelData.h"
10#include "QskTextOptions.h"
11#include "QskGraphic.h"
12#include "QskColorFilter.h"
13#include "QskSkinlet.h"
14#include "QskEvent.h"
15#include "QskPlatform.h"
16#include "QskInternalMacros.h"
17
18#include <qvector.h>
19#include <qvariant.h>
20#include <qeventloop.h>
21
22QSK_QT_PRIVATE_BEGIN
23#include <private/qquickitem_p.h>
24QSK_QT_PRIVATE_END
25
26QSK_SUBCONTROL( QskMenu, Panel )
27QSK_SUBCONTROL( QskMenu, Segment )
28QSK_SUBCONTROL( QskMenu, Cursor )
29QSK_SUBCONTROL( QskMenu, Text )
30QSK_SUBCONTROL( QskMenu, Icon )
31QSK_SUBCONTROL( QskMenu, Separator )
32
33QSK_SYSTEM_STATE( QskMenu, Selected, QskAspect::FirstSystemState << 2 )
34QSK_SYSTEM_STATE( QskMenu, Pressed, QskAspect::FirstSystemState << 3 )
35
36static inline int qskActionIndex( const QVector< int >& actions, int index )
37{
38 if ( index < 0 )
39 return -1;
40
41 auto it = std::lower_bound( actions.constBegin(), actions.constEnd(), index );
42 return it - actions.constBegin();
43}
44
45class QskMenu::PrivateData
46{
47 public:
48 QPointF origin;
49
50 QVector< QskLabelData > options;
51
52 QVector< int > separators;
53 QVector< int > actions;
54
55 int triggeredIndex = -1;
56 int currentIndex = -1;
57
58 bool wrapping = true;
59 bool isPressed = false;
60};
61
62QskMenu::QskMenu( QQuickItem* parent )
63 : Inherited( parent )
64 , m_data( new PrivateData )
65{
66#if 1
67 /*
68 The overlay is clipped from the drop down fading effect.
69 Until it is fixed we simply disable it. TODO ...
70 */
71 setOverlay( false );
72#endif
73 setModal( true );
74
75 setPopupFlag( QskPopup::CloseOnPressOutside, true );
76 setPopupFlag( QskPopup::DeleteOnClose, true );
77
78 setPlacementPolicy( QskPlacementPolicy::Ignore );
79 setSubcontrolProxy( Inherited::Overlay, Overlay );
80
81 initSizePolicy( QskSizePolicy::Fixed, QskSizePolicy::Fixed );
82
83 // we hide the focus indicator while sliding
84 connect( this, &QskPopup::fadingChanged,
86
87 connect( this, &QskPopup::fadingChanged,
88 this, &QQuickItem::setClip );
89
90 connect( this, &QskPopup::opened, this,
91 [this]() { m_data->triggeredIndex = -1; } );
92
93 setAcceptHoverEvents( true );
94}
95
96QskMenu::~QskMenu()
97{
98}
99
100QRectF QskMenu::clipRect() const
101{
102 if ( isFading() )
103 {
104 constexpr qreal d = 1e6;
105 return QRectF( -d, m_data->origin.y() - y(), 2.0 * d, d );
106 }
107
108 return Inherited::clipRect();
109}
110
111bool QskMenu::isWrapping() const
112{
113 return m_data->wrapping;
114}
115
116void QskMenu::setWrapping( bool on )
117{
118 if ( m_data->wrapping != on )
119 {
120 m_data->wrapping = on;
121 Q_EMIT wrappingChanged( on );
122 }
123}
124
125#if 1
126
127// has no effect as we do not offer submenus yet. TODO ...
128bool QskMenu::isCascading() const
129{
130 return flagHint( QskMenu::Panel | QskAspect::Style, qskMaybeDesktopPlatform() );
131}
132
133void QskMenu::setCascading( bool on )
134{
135 if ( setFlagHint( QskMenu::Panel | QskAspect::Style, on ) )
136 Q_EMIT cascadingChanged( on );
137}
138
139void QskMenu::resetCascading()
140{
141 if ( resetSkinHint( QskMenu::Panel | QskAspect::Style ) )
142 Q_EMIT cascadingChanged( isCascading() );
143}
144
145#endif
146
147void QskMenu::setOrigin( const QPointF& origin )
148{
149 if ( origin != m_data->origin )
150 {
151 m_data->origin = origin;
152 Q_EMIT originChanged( origin );
153 }
154}
155
156QPointF QskMenu::origin() const
157{
158 return m_data->origin;
159}
160
161void QskMenu::setTextOptions( const QskTextOptions& textOptions )
162{
163 setTextOptionsHint( Text, textOptions );
164}
165
166QskTextOptions QskMenu::textOptions() const
167{
168 return textOptionsHint( Text );
169}
170
171int QskMenu::addOption( const QString& graphicSource, const QString& text )
172{
173 return addOption( QskLabelData( text, graphicSource ) );
174}
175
176int QskMenu::addOption( const QUrl& graphicSource, const QString& text )
177{
178 return addOption( QskLabelData( text, graphicSource ) );
179}
180
181int QskMenu::addOption( const QskLabelData& option )
182{
183 const int index = m_data->options.count();
184
185 m_data->options += option;
186
187 if ( option.isEmpty() )
188 m_data->separators += index;
189 else
190 m_data->actions += index;
191
193 update();
194
195 if ( isComponentComplete() )
196 Q_EMIT optionsChanged();
197
198 return index;
199}
200
201void QskMenu::setOptions( const QStringList& options )
202{
203 setOptions( qskCreateLabelData( options ) );
204}
205
206void QskMenu::setOptions( const QVector< QskLabelData >& options )
207{
208 m_data->options = options;
209
210 for ( int i = 0; i < options.count(); i++ )
211 {
212 if ( options[i].isEmpty() )
213 m_data->separators += i;
214 else
215 m_data->actions += i;
216 }
217
218 if ( m_data->currentIndex >= 0 )
219 {
220 m_data->currentIndex = -1;
221
222 if ( isComponentComplete() )
223 Q_EMIT currentIndexChanged( m_data->currentIndex );
224 }
225
227 update();
228
229 if ( isComponentComplete() )
230 Q_EMIT optionsChanged();
231}
232
233void QskMenu::clear()
234{
235 setOptions( QVector< QskLabelData >() );
236}
237
238QVector< QskLabelData > QskMenu::options() const
239{
240 return m_data->options;
241}
242
243QskLabelData QskMenu::optionAt( int index ) const
244{
245 return m_data->options.value( index );
246}
247
248void QskMenu::addSeparator()
249{
250 addOption( QskLabelData() );
251}
252
253QVector< int > QskMenu::separators() const
254{
255 return m_data->separators;
256}
257
258QVector< int > QskMenu::actions() const
259{
260 return m_data->actions;
261}
262
263int QskMenu::currentIndex() const
264{
265 return m_data->currentIndex;
266}
267
268void QskMenu::setCurrentIndex( int index )
269{
270 if( index < 0 || index >= m_data->options.count() )
271 {
272 index = -1;
273 }
274 else
275 {
276 if ( m_data->options[index].isEmpty() ) // a seperator
277 index = -1;
278 }
279
280 if( index != m_data->currentIndex )
281 {
282 setPositionHint( Cursor, index );
283
284 m_data->currentIndex = index;
285 update();
286
287 Q_EMIT currentIndexChanged( index );
289 }
290}
291
292QString QskMenu::currentText() const
293{
294 return optionAt( m_data->currentIndex ).text();
295}
296
297int QskMenu::triggeredIndex() const
298{
299 return m_data->triggeredIndex;
300}
301
302QString QskMenu::triggeredText() const
303{
304 return optionAt( m_data->triggeredIndex ).text();
305}
306
307void QskMenu::updateResources()
308{
309 qreal dy = 0.0;
310 if ( isFading() )
311 dy = ( 1.0 - fadingFactor() ) * height();
312
313 setPosition( m_data->origin.x(), m_data->origin.y() - dy );
314
315 Inherited::updateResources();
316}
317
318void QskMenu::updateNode( QSGNode* node )
319{
320 if ( isFading() && clip() )
321 {
322 if ( auto clipNode = QQuickItemPrivate::get( this )->clipNode() )
323 {
324 /*
325 The clipRect is changing while fading. Couldn't
326 find a way how to trigger updates - maybe be enabling/disabling
327 the clip. So we do the updates manually. TODO ...
328 */
329 const auto r = clipRect();
330 if ( r != clipNode->rect() )
331 {
332 clipNode->setRect( r );
333 clipNode->update();
334 }
335 }
336 }
337
338 Inherited::updateNode( node );
339}
340
341void QskMenu::keyPressEvent( QKeyEvent* event )
342{
343 if( m_data->currentIndex < 0 )
344 return;
345
346 switch( event->key() )
347 {
348 case Qt::Key_Up:
349 {
350 traverse( -1 );
351 return;
352 }
353
354 case Qt::Key_Down:
355 {
356 traverse( 1 );
357 return;
358 }
359
360 case Qt::Key_Select:
361 case Qt::Key_Space:
362 case Qt::Key_Return:
363 case Qt::Key_Enter:
364 {
365 m_data->isPressed = true;
366 return;
367 }
368
369 default:
370 {
371 if ( const int steps = qskFocusChainIncrement( event ) )
372 {
373 traverse( steps );
374 return;
375 }
376 }
377 }
378
379 return Inherited::keyPressEvent( event );
380}
381
382void QskMenu::keyReleaseEvent( QKeyEvent* )
383{
384 if( isPressed() )
385 {
386 m_data->isPressed = false;
387
388 if ( m_data->currentIndex >= 0 )
389 {
390 close();
391 trigger( m_data->currentIndex );
392 }
393 }
394}
395
396void QskMenu::hoverEnterEvent( QHoverEvent* event )
397{
398 using A = QskAspect;
399
400 setSkinHint( Segment | Hovered | A::Metric | A::Position, qskHoverPosition( event ) );
401 update();
402}
403
404void QskMenu::hoverMoveEvent( QHoverEvent* event )
405{
406 using A = QskAspect;
407
408 setSkinHint( Segment | Hovered | A::Metric | A::Position, qskHoverPosition( event ) );
409 update();
410}
411
412void QskMenu::hoverLeaveEvent( QHoverEvent* )
413{
414 using A = QskAspect;
415
416 setSkinHint( Segment | Hovered | A::Metric | A::Position, QPointF() );
417 update();
418}
419
420#ifndef QT_NO_WHEELEVENT
421
422void QskMenu::wheelEvent( QWheelEvent* event )
423{
424 const auto steps = qskWheelSteps( event );
425 traverse( -steps );
426}
427
428#endif
429
430void QskMenu::traverse( int steps )
431{
432 const auto& actions = m_data->actions;
433 const auto count = actions.count();
434
435 // -1 -> only one entry ?
436 if ( actions.isEmpty() || ( steps % count == 0 ) )
437 return;
438
439 int action1 = qskActionIndex( actions, m_data->currentIndex );
440 int action2 = action1 + steps;
441
442 if ( !m_data->wrapping )
443 action2 = qBound( 0, action2, count - 1 );
444
445 // when cycling we want to slide in
446 int index1;
447
448 if ( action2 < 0 )
449 index1 = m_data->options.count();
450 else if ( action2 >= count )
451 index1 = -1;
452 else
453 index1 = actions[ action1 ];
454
455 action2 %= count;
456 if ( action2 < 0 )
457 action2 += count;
458
459 const auto index2 = actions[ action2 ];
460
461 movePositionHint( Cursor, index1, index2 );
462 setCurrentIndex( index2 );
463}
464
465void QskMenu::mousePressEvent( QMouseEvent* event )
466{
467 // QGuiApplication::styleHints()->setFocusOnTouchRelease ??
468
469 if ( event->button() == Qt::LeftButton )
470 {
471 const auto index = indexAtPosition( qskMousePosition( event ) );
472 if ( index >= 0 )
473 {
474 setCurrentIndex( index );
475 m_data->isPressed = true;
476 }
477
478 return;
479 }
480
481 Inherited::mousePressEvent( event );
482}
483
484void QskMenu::mouseUngrabEvent()
485{
486 m_data->isPressed = false;
488}
489
490void QskMenu::mouseReleaseEvent( QMouseEvent* event )
491{
492 if ( event->button() == Qt::LeftButton )
493 {
494 if( isPressed() )
495 {
496 m_data->isPressed = false;
497
498 const auto index = m_data->currentIndex;
499
500 if ( ( index >= 0 )
501 && ( index == indexAtPosition( qskMousePosition( event ) ) ) )
502 {
503 close();
504 trigger( m_data->currentIndex );
505 }
506 }
507
508 return;
509 }
510
511 Inherited::mouseReleaseEvent( event );
512}
513
515{
516 setSize( sizeConstraint() );
517
518 if ( m_data->currentIndex < 0 )
519 {
520 if ( !m_data->actions.isEmpty() )
521 setCurrentIndex( m_data->actions.first() );
522 }
523
525}
526
528{
529 if ( isFading() )
530 return QRectF();
531
532 if( currentIndex() >= 0 )
533 {
534 auto actionIndex = qskActionIndex( m_data->actions, currentIndex() );
535
536 return effectiveSkinlet()->sampleRect( this,
537 contentsRect(), Segment, actionIndex );
538 }
539
541}
542
543QRectF QskMenu::cellRect( int index ) const
544{
545 const auto actionIndex = qskActionIndex( m_data->actions, index );
546
547 return effectiveSkinlet()->sampleRect(
548 this, contentsRect(), QskMenu::Segment, actionIndex );
549}
550
551int QskMenu::indexAtPosition( const QPointF& pos ) const
552{
553 const auto index = effectiveSkinlet()->sampleIndexAt(
554 this, contentsRect(), QskMenu::Segment, pos );
555
556 return m_data->actions.value( index, -1 );
557}
558
559bool QskMenu::isPressed() const
560{
561 return m_data->isPressed;
562}
563
564void QskMenu::trigger( int index )
565{
566 if ( index >= 0 && index < m_data->options.count() )
567 {
568 m_data->triggeredIndex = index;
569 Q_EMIT triggered( index );
570 }
571}
572
573QskAspect QskMenu::fadingAspect() const
574{
575 return QskMenu::Panel | QskAspect::Position;
576}
577
578int QskMenu::exec()
579{
580 (void) execPopup();
581 return m_data->triggeredIndex;
582}
583
584#include "moc_QskMenu.cpp"
Lookup key for a QskSkinHintTable.
Definition QskAspect.h:15
@ FirstSystemState
Definition QskAspect.h:115
void focusIndicatorRectChanged()
static const QskAspect::State Hovered
Definition QskControl.h:56
virtual QRectF focusIndicatorRect() const
QRectF contentsRect() const
QSizeF sizeConstraint
Definition QskControl.h:50
void mouseUngrabEvent() override
Definition QskItem.cpp:1052
void resetImplicitSize()
Definition QskItem.cpp:727
void updateNode(QSGNode *) override
Definition QskMenu.cpp:318
void aboutToShow() override
Definition QskMenu.cpp:514
QRectF focusIndicatorRect() const override
Definition QskMenu.cpp:527
void aboutToShow() override
Definition QskPopup.cpp:600
bool resetSkinHint(QskAspect)
Remove a hint from the local hint table.
bool setSkinHint(QskAspect, const QVariant &)
Insert a hint into the local hint table.
const QskSkinlet * effectiveSkinlet() const
bool setFlagHint(QskAspect, int flag)
Sets a flag hint.
virtual void updateNode(QSGNode *)
T flagHint(QskAspect, T=T()) const
Retrieves a flag hint.