qskinny/src/controls/QskSliderSkinlet.cpp
2019-04-25 14:23:39 +02:00

247 lines
6.7 KiB
C++

/******************************************************************************
* QSkinny - Copyright (C) 2016 Uwe Rathmann
* This file may be used under the terms of the QSkinny License, Version 1.0
*****************************************************************************/
#include "QskSliderSkinlet.h"
#include "QskSlider.h"
#include "QskAspect.h"
#include "QskBoxBorderMetrics.h"
#include "QskFunctions.h"
static QMarginsF qskPadding(
const QskSlider* slider, QskAspect::Subcontrol subControl )
{
return slider->marginsHint( subControl | QskAspect::Padding ) +
slider->boxBorderMetricsHint( subControl ).widths();
}
QskSliderSkinlet::QskSliderSkinlet( QskSkin* skin )
: Inherited( skin )
{
setNodeRoles( { PanelRole, GrooveRole, FillRole, HandleRole } );
}
QskSliderSkinlet::~QskSliderSkinlet() = default;
QRectF QskSliderSkinlet::subControlRect( const QskSkinnable* skinnable,
const QRectF& contentsRect, QskAspect::Subcontrol subControl ) const
{
const auto slider = static_cast< const QskSlider* >( skinnable );
if ( subControl == QskSlider::Panel )
{
return panelRect( slider, contentsRect );
}
if ( subControl == QskSlider::Groove )
{
return grooveRect( slider, contentsRect );
}
if ( subControl == QskSlider::Fill )
{
return fillRect( slider, contentsRect );
}
if ( subControl == QskSlider::Handle )
{
return handleRect( slider, contentsRect );
}
if ( subControl == QskSlider::Scale )
{
return scaleRect( slider, contentsRect );
}
return Inherited::subControlRect( skinnable, contentsRect, subControl );
}
QSGNode* QskSliderSkinlet::updateSubNode(
const QskSkinnable* skinnable, quint8 nodeRole, QSGNode* node ) const
{
const auto slider = static_cast< const QskSlider* >( skinnable );
switch ( nodeRole )
{
case PanelRole:
{
return updateBoxNode( slider, node, QskSlider::Panel );
}
case GrooveRole:
{
return updateBoxNode( slider, node, QskSlider::Groove );
}
case FillRole:
{
return updateBoxNode( slider, node, QskSlider::Fill );
}
case HandleRole:
{
return updateBoxNode( slider, node, QskSlider::Handle );
}
}
return Inherited::updateSubNode( skinnable, nodeRole, node );
}
QRectF QskSliderSkinlet::panelRect(
const QskSlider* slider, const QRectF& contentsRect ) const
{
using namespace QskAspect;
const QskAspect::Subcontrol subControl = QskSlider::Panel;
qreal size = slider->metric( subControl | Size ); // 0: no hint
{
/*
We use MinimumHeight/MaximumHeight/Size for the "thickness" of
to slider even if "Height" is not the expected value for
vertical sliders. Maybe we should have the orientation as an aspect bit ?
*/
const qreal minSize = slider->metric( subControl | MinimumHeight );
const qreal maxSize = slider->metric( subControl | MaximumHeight );
size = qMax( size, minSize );
if ( maxSize > minSize )
size = qMin( size, maxSize );
}
QRectF r = contentsRect;
if ( size > 0 && size < r.height() )
{
const Qt::Alignment alignment =
slider->flagHint< Qt::Alignment >( subControl | Alignment );
if ( slider->orientation() == Qt::Horizontal )
r = qskAlignedRectF( r, r.width(), size, alignment & Qt::AlignVertical_Mask );
else
r = qskAlignedRectF( r, size, r.height(), alignment & Qt::AlignHorizontal_Mask );
}
return r;
}
QRectF QskSliderSkinlet::innerRect( const QskSlider* slider,
const QRectF& contentsRect, QskAspect::Subcontrol subControl ) const
{
using namespace QskAspect;
QRectF r = subControlRect( slider, contentsRect, QskSlider::Panel );
r = r.marginsRemoved( qskPadding( slider, QskSlider::Panel ) );
QskSkinHintStatus status;
qreal thickness = slider->metric( subControl | Size, &status );
if ( slider->orientation() == Qt::Horizontal )
{
if ( status.isValid() && thickness < r.height() )
{
r.setTop( r.center().y() - 0.5 * thickness );
r.setHeight( thickness );
}
}
else
{
if ( status.isValid() && thickness < r.width() )
{
r.setLeft( r.center().x() - 0.5 * thickness );
r.setWidth( thickness );
}
}
return r;
}
QRectF QskSliderSkinlet::grooveRect(
const QskSlider* slider, const QRectF& rect ) const
{
return innerRect( slider, rect, QskSlider::Groove );
}
QRectF QskSliderSkinlet::scaleRect(
const QskSlider* slider, const QRectF& rect ) const
{
return innerRect( slider, rect, QskSlider::Groove );
}
QRectF QskSliderSkinlet::fillRect(
const QskSlider* slider, const QRectF& contentsRect ) const
{
using namespace QskAspect;
QRectF r = subControlRect( slider, contentsRect, QskSlider::Panel );
r = r.marginsRemoved( qskPadding( slider, QskSlider::Panel ) );
qreal pos = slider->metric( QskSlider::Handle | Position );
pos = qBound( 0.0, pos, 1.0 );
QRectF fillRect = innerRect( slider, contentsRect, QskSlider::Fill );
if ( slider->orientation() == Qt::Horizontal )
{
fillRect.setLeft( r.left() );
fillRect.setRight( r.left() + pos * r.width() );
}
else
{
fillRect.setBottom( r.bottom() );
fillRect.setTop( r.bottom() - pos * r.height() );
}
return fillRect;
}
QRectF QskSliderSkinlet::handleRect(
const QskSlider* slider, const QRectF& contentsRect ) const
{
using namespace QskAspect;
QRectF r = subControlRect( slider, contentsRect, QskSlider::Panel );
r = r.marginsRemoved( qskPadding( slider, QskSlider::Panel ) );
const bool isHorizontal = slider->orientation() == Qt::Horizontal;
qreal thickness;
{
QskSkinHintStatus status;
thickness = slider->metric( QskSlider::Handle | Size, &status );
if ( !status.isValid() )
thickness = isHorizontal ? r.height() : r.width();
}
QRectF handleRect( 0, 0, thickness, thickness );
if ( thickness > 0 )
{
handleRect = handleRect.marginsRemoved(
slider->marginsHint( QskSlider::Handle | QskAspect::Margin ) );
}
qreal pos = slider->metric( QskSlider::Handle | Position );
pos = qBound( 0.0, pos, 1.0 );
if ( slider->orientation() == Qt::Horizontal )
{
pos = r.left() + pos * r.width();
handleRect.moveCenter( QPointF( pos, r.center().y() ) );
}
else
{
pos = r.bottom() - pos * r.height();
handleRect.moveCenter( QPointF( r.center().x(), pos ) );
}
return handleRect;
}
#include "moc_QskSliderSkinlet.cpp"