qskinny/src/controls/QskComboBoxSkinlet.cpp

242 lines
6.8 KiB
C++
Raw Normal View History

/******************************************************************************
* QSkinny - Copyright (C) 2023 Uwe Rathmann
* This file may be used under the terms of the QSkinny License, Version 1.0
*****************************************************************************/
#include "QskComboBoxSkinlet.h"
#include "QskComboBox.h"
#include "QskGraphic.h"
#include "QskSkin.h"
#include "QskSGNode.h"
#include "QskStandardSymbol.h"
#include "QskSubcontrolLayoutEngine.h"
namespace
{
#if 1 // unify with the implementation from QskMenu
template< class T >
static inline QVariant qskSampleAt( const QskComboBox* box )
{
if( std::is_same< T, QString >() )
{
return box->text();
}
const int index = box->currentIndex();
if( index < 0 )
return QVariant::fromValue( T() );
const auto list = box->optionAt( index );
for ( const auto& value : list )
{
if ( value.canConvert< T >() )
return value;
}
return QVariant();
}
template< class T >
static inline T qskValueAt( const QskComboBox* box )
{
const auto sample = qskSampleAt< T >( box );
return sample.template value< T >();
}
#endif
class LayoutEngine : public QskSubcontrolLayoutEngine
{
public:
LayoutEngine( const QskComboBox* box )
: QskSubcontrolLayoutEngine( Qt::Horizontal )
{
setSpacing( box->spacingHint( QskComboBox::Panel ) );
setGraphicTextElements( box,
QskComboBox::Text, qskValueAt< QString >( box ),
QskComboBox::Graphic, qskValueAt< QskGraphic >( box ).defaultSize() );
const auto alignment = box->alignmentHint( QskComboBox::Panel, Qt::AlignLeft );
setFixedContent( QskComboBox::Text, Qt::Horizontal, alignment );
}
};
}
QskComboBoxSkinlet::QskComboBoxSkinlet( QskSkin* skin )
: Inherited( skin )
{
setNodeRoles( { PanelRole, GraphicRole, TextRole, OpenMenuGraphicRole, RippleRole } );
}
QskComboBoxSkinlet::~QskComboBoxSkinlet() = default;
QRectF QskComboBoxSkinlet::subControlRect( const QskSkinnable* skinnable,
const QRectF& contentsRect, QskAspect::Subcontrol subControl ) const
{
using Q = QskComboBox;
const auto* box = static_cast< const QskComboBox* >( skinnable );
if ( subControl == Q::Text || subControl == Q::Graphic )
{
const auto r = box->subControlContentsRect( contentsRect, Q::Panel );
LayoutEngine layoutEngine( box );
layoutEngine.setGeometries( r );
return layoutEngine.subControlRect( subControl );
}
if( subControl == Q::OpenMenuGraphic )
{
auto rect = box->innerBox( Q::Panel, contentsRect );
const auto size = box->strutSizeHint( Q::OpenMenuGraphic );
rect.setLeft( rect.right() - size.width() );
return rect;
}
if ( subControl == Q::Panel )
{
return contentsRect;
}
if ( subControl == Q::Ripple )
{
return rippleRect( box, contentsRect );
}
return Inherited::subControlRect( skinnable, contentsRect, subControl );
}
QSGNode* QskComboBoxSkinlet::updateSubNode(
const QskSkinnable* skinnable, quint8 nodeRole, QSGNode* node ) const
{
using Q = QskComboBox;
const auto bar = static_cast< const QskComboBox* >( skinnable );
switch ( nodeRole )
{
case PanelRole:
{
return updateBoxNode( bar, node, Q::Panel );
}
case GraphicRole:
{
return updateGraphicNode( bar, node, bar->graphic(), Q::Graphic );
}
case TextRole:
{
return updateTextNode( bar, node );
}
case OpenMenuGraphicRole:
{
const auto symbol = bar->isPopupOpen() ? QskStandardSymbol::ComboBoxSymbolPopupOpen
: QskStandardSymbol::ComboBoxSymbolPopupClosed;
const auto graphic = bar->effectiveSkin()->symbol( symbol );
return updateGraphicNode( bar, node, graphic, Q::OpenMenuGraphic );
}
}
return Inherited::updateSubNode( skinnable, nodeRole, node );
}
QRectF QskComboBoxSkinlet::rippleRect(
const QskComboBox* bar, const QRectF& contentsRect ) const
{
using Q = QskComboBox;
QRectF rect;
const auto ratio = bar->metric( Q::Ripple | QskAspect::Size );
if ( ratio > 0.0 )
{
const auto pos = bar->effectiveSkinHint(
Q::Ripple | QskAspect::Metric | QskAspect::Position ).toPointF();
const auto panelRect = subControlRect( bar, contentsRect, Q::Panel );
rect.setSize( 2.0 * panelRect.size() * ratio );
rect.moveCenter( pos );
}
return rect;
}
QSGNode* QskComboBoxSkinlet::updateTextNode(
const QskComboBox* box, QSGNode* node ) const
{
using Q = QskComboBox;
const auto rect = box->subControlRect( Q::Text ).toAlignedRect();
const auto textHeight = box->effectiveFontHeight( Q::Text );
if ( !box->clip() && ( rect.height() < textHeight ) )
return nullptr;
const auto alignment = box->alignmentHint( Q::Text, Qt::AlignLeft | Qt::AlignVCenter );
return QskSkinlet::updateTextNode( box, node, rect,
alignment, box->text(), Q::Text );
}
QSGNode* QskComboBoxSkinlet::updateRippleNode(
const QskComboBox* button, QSGNode* node ) const
{
using Q = QskComboBox;
const auto rippleRect = button->subControlRect( Q::Ripple );
if ( rippleRect.isEmpty() )
return nullptr;
auto clipNode = updateBoxClipNode( button, node,
button->subControlRect( Q::Panel ), Q::Panel );
if ( clipNode )
{
auto boxNode = QskSGNode::findChildNode( clipNode, RippleRole );
boxNode = updateBoxNode( button, boxNode, rippleRect, Q::Ripple );
if ( boxNode == nullptr )
return nullptr;
QskSGNode::setNodeRole( boxNode, RippleRole );
if ( boxNode->parent() != clipNode )
clipNode->appendChildNode( boxNode );
}
return clipNode;
}
QSizeF QskComboBoxSkinlet::sizeHint( const QskSkinnable* skinnable,
Qt::SizeHint which, const QSizeF& ) const
{
using Q = QskComboBox;
if ( which != Qt::PreferredSize )
return QSizeF();
const auto box = static_cast< const QskComboBox* >( skinnable );
LayoutEngine layoutEngine( box );
auto size = layoutEngine.sizeHint( which, QSizeF() );
const auto spacingHint = box->spacingHint( Q::Panel );
const auto menuGraphicHint = box->strutSizeHint( Q::OpenMenuGraphic );
size.rwidth() += spacingHint + menuGraphicHint.width();
size = box->outerBoxSize( Q::Panel, size );
size = size.expandedTo( box->strutSizeHint( Q::Panel ) );
size = size.grownBy( skinnable->marginHint( Q::Panel ) );
return size;
}
#include "moc_QskComboBoxSkinlet.cpp"