8#include "QskGraphicProvider.h"
9#include "QskLabelData.h"
10#include "QskTextOptions.h"
11#include "QskGraphic.h"
12#include "QskColorFilter.h"
13#include "QskSkinlet.h"
15#include "QskPlatform.h"
16#include "QskInternalMacros.h"
20#include <qeventloop.h>
23#include <private/qquickitem_p.h>
27QSK_SUBCONTROL(
QskMenu, Segment )
28QSK_SUBCONTROL(
QskMenu, Cursor )
31QSK_SUBCONTROL(
QskMenu, Separator )
36static inline
int qskActionIndex( const QVector<
int >& actions,
int index )
41 auto it = std::lower_bound( actions.constBegin(), actions.constEnd(), index );
42 return it - actions.constBegin();
45class QskMenu::PrivateData
50 QVector< QskLabelData > options;
52 QVector< int > separators;
53 QVector< int > actions;
55 int triggeredIndex = -1;
56 int currentIndex = -1;
59 bool isPressed =
false;
62QskMenu::QskMenu( QQuickItem* parent )
64 , m_data( new PrivateData )
75 setPopupFlag( QskPopup::CloseOnPressOutside,
true );
76 setPopupFlag( QskPopup::DeleteOnClose,
true );
79 setSubcontrolProxy( Inherited::Overlay, Overlay );
81 initSizePolicy( QskSizePolicy::Fixed, QskSizePolicy::Fixed );
84 connect(
this, &QskPopup::fadingChanged,
87 connect(
this, &QskPopup::fadingChanged,
88 this, &QQuickItem::setClip );
90 connect(
this, &QskPopup::opened,
this,
91 [
this]() { m_data->triggeredIndex = -1; } );
93 setAcceptHoverEvents(
true );
100QRectF QskMenu::clipRect()
const
104 constexpr qreal d = 1e6;
105 return QRectF( -d, m_data->origin.y() - y(), 2.0 * d, d );
108 return Inherited::clipRect();
111bool QskMenu::isWrapping()
const
113 return m_data->wrapping;
116void QskMenu::setWrapping(
bool on )
118 if ( m_data->wrapping != on )
120 m_data->wrapping = on;
121 Q_EMIT wrappingChanged( on );
128bool QskMenu::isCascading()
const
133void QskMenu::setCascading(
bool on )
136 Q_EMIT cascadingChanged( on );
139void QskMenu::resetCascading()
142 Q_EMIT cascadingChanged( isCascading() );
147void QskMenu::setOrigin(
const QPointF& origin )
149 if ( origin != m_data->origin )
151 m_data->origin = origin;
152 Q_EMIT originChanged( origin );
156QPointF QskMenu::origin()
const
158 return m_data->origin;
163 setTextOptionsHint( Text, textOptions );
168 return textOptionsHint( Text );
171int QskMenu::addOption(
const QString& graphicSource,
const QString& text )
173 return addOption(
QskLabelData( text, graphicSource ) );
176int QskMenu::addOption(
const QUrl& graphicSource,
const QString& text )
178 return addOption(
QskLabelData( text, graphicSource ) );
183 const int index = m_data->options.count();
185 m_data->options += option;
187 if ( option.isEmpty() )
188 m_data->separators += index;
190 m_data->actions += index;
195 if ( isComponentComplete() )
196 Q_EMIT optionsChanged();
201void QskMenu::setOptions(
const QStringList& options )
203 setOptions( qskCreateLabelData( options ) );
206void QskMenu::setOptions(
const QVector< QskLabelData >& options )
208 m_data->options = options;
210 for (
int i = 0; i < options.count(); i++ )
212 if ( options[i].isEmpty() )
213 m_data->separators += i;
215 m_data->actions += i;
218 if ( m_data->currentIndex >= 0 )
220 m_data->currentIndex = -1;
222 if ( isComponentComplete() )
223 Q_EMIT currentIndexChanged( m_data->currentIndex );
229 if ( isComponentComplete() )
230 Q_EMIT optionsChanged();
235 setOptions( QVector< QskLabelData >() );
238QVector< QskLabelData > QskMenu::options()
const
240 return m_data->options;
245 return m_data->options.value( index );
248void QskMenu::addSeparator()
253QVector< int > QskMenu::separators()
const
255 return m_data->separators;
258QVector< int > QskMenu::actions()
const
260 return m_data->actions;
263int QskMenu::currentIndex()
const
265 return m_data->currentIndex;
268void QskMenu::setCurrentIndex(
int index )
270 if( index < 0 || index >= m_data->options.count() )
276 if ( m_data->options[index].isEmpty() )
280 if( index != m_data->currentIndex )
282 setPositionHint( Cursor, index );
284 m_data->currentIndex = index;
287 Q_EMIT currentIndexChanged( index );
292QString QskMenu::currentText()
const
294 return optionAt( m_data->currentIndex ).text();
297int QskMenu::triggeredIndex()
const
299 return m_data->triggeredIndex;
302QString QskMenu::triggeredText()
const
304 return optionAt( m_data->triggeredIndex ).text();
307void QskMenu::updateResources()
311 dy = ( 1.0 - fadingFactor() ) * height();
313 setPosition( m_data->origin.x(), m_data->origin.y() - dy );
315 Inherited::updateResources();
320 if ( isFading() && clip() )
322 if (
auto clipNode = QQuickItemPrivate::get(
this )->clipNode() )
329 const auto r = clipRect();
330 if ( r != clipNode->rect() )
332 clipNode->setRect( r );
341void QskMenu::keyPressEvent( QKeyEvent* event )
343 if( m_data->currentIndex < 0 )
346 switch( event->key() )
365 m_data->isPressed =
true;
371 if (
const int steps = qskFocusChainIncrement( event ) )
379 return Inherited::keyPressEvent( event );
382void QskMenu::keyReleaseEvent( QKeyEvent* )
386 m_data->isPressed =
false;
388 if ( m_data->currentIndex >= 0 )
391 trigger( m_data->currentIndex );
396void QskMenu::hoverEnterEvent( QHoverEvent* event )
400 setSkinHint( Segment |
Hovered | A::Metric | A::Position, qskHoverPosition( event ) );
404void QskMenu::hoverMoveEvent( QHoverEvent* event )
408 setSkinHint( Segment |
Hovered | A::Metric | A::Position, qskHoverPosition( event ) );
412void QskMenu::hoverLeaveEvent( QHoverEvent* )
420#ifndef QT_NO_WHEELEVENT
422void QskMenu::wheelEvent( QWheelEvent* event )
424 const auto steps = qskWheelSteps( event );
430void QskMenu::traverse(
int steps )
432 const auto& actions = m_data->actions;
433 const auto count = actions.count();
436 if ( actions.isEmpty() || ( steps % count == 0 ) )
439 int action1 = qskActionIndex( actions, m_data->currentIndex );
440 int action2 = action1 + steps;
442 if ( !m_data->wrapping )
443 action2 = qBound( 0, action2, count - 1 );
449 index1 = m_data->options.count();
450 else if ( action2 >= count )
453 index1 = actions[ action1 ];
459 const auto index2 = actions[ action2 ];
461 movePositionHint( Cursor, index1, index2 );
462 setCurrentIndex( index2 );
465void QskMenu::mousePressEvent( QMouseEvent* event )
469 if ( event->button() == Qt::LeftButton )
471 const auto index = indexAtPosition( qskMousePosition( event ) );
474 setCurrentIndex( index );
475 m_data->isPressed =
true;
481 Inherited::mousePressEvent( event );
484void QskMenu::mouseUngrabEvent()
486 m_data->isPressed =
false;
490void QskMenu::mouseReleaseEvent( QMouseEvent* event )
492 if ( event->button() == Qt::LeftButton )
496 m_data->isPressed =
false;
498 const auto index = m_data->currentIndex;
501 && ( index == indexAtPosition( qskMousePosition( event ) ) ) )
504 trigger( m_data->currentIndex );
511 Inherited::mouseReleaseEvent( event );
518 if ( m_data->currentIndex < 0 )
520 if ( !m_data->actions.isEmpty() )
521 setCurrentIndex( m_data->actions.first() );
532 if( currentIndex() >= 0 )
534 auto actionIndex = qskActionIndex( m_data->actions, currentIndex() );
543QRectF QskMenu::cellRect(
int index )
const
545 const auto actionIndex = qskActionIndex( m_data->actions, index );
551int QskMenu::indexAtPosition(
const QPointF& pos )
const
556 return m_data->actions.value( index, -1 );
559bool QskMenu::isPressed()
const
561 return m_data->isPressed;
564void QskMenu::trigger(
int index )
566 if ( index >= 0 && index < m_data->options.count() )
568 m_data->triggeredIndex = index;
569 Q_EMIT triggered( index );
581 return m_data->triggeredIndex;
584#include "moc_QskMenu.cpp"
Lookup key for a QskSkinHintTable.
void focusIndicatorRectChanged()
static const QskAspect::State Hovered
virtual QRectF focusIndicatorRect() const
QRectF contentsRect() const
void mouseUngrabEvent() override
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.