qskinny/playground/dials/DialSkinlet.cpp
2024-01-17 14:31:45 +01:00

201 lines
5.5 KiB
C++

/******************************************************************************
* QSkinny - Copyright (C) The authors
* SPDX-License-Identifier: BSD-3-Clause
*****************************************************************************/
#include "DialSkinlet.h"
#include "Dial.h"
#include <QskBoxBorderColors.h>
#include <QskBoxBorderMetrics.h>
#include <QskBoxShapeMetrics.h>
#include <QskTextColors.h>
#include <QskTextOptions.h>
#include <QskLinesNode.h>
#include <QskFunctions.h>
#include <QskSGNode.h>
#include <QFontMetrics>
#include <QLineF>
#include <QtMath>
DialSkinlet::DialSkinlet( QskSkin* skin )
: QskSkinlet( skin )
{
setNodeRoles( { PanelRole, NeedleRole, KnobRole, LabelsRole } );
}
QRectF DialSkinlet::subControlRect( const QskSkinnable* skinnable,
const QRectF& contentsRect, QskAspect::Subcontrol subcontrol ) const
{
QRectF r;
if ( subcontrol == Dial::Knob )
{
const auto size = skinnable->strutSizeHint( Dial::Knob );
r.setSize( size );
}
else
{
const auto extent = qMin( contentsRect.width(), contentsRect.height() );
r.setSize( QSizeF( extent, extent ) );
}
r.moveCenter( contentsRect.center() );
return r;
}
QSGNode* DialSkinlet::updateSubNode(
const QskSkinnable* skinnable, quint8 nodeRole, QSGNode* node ) const
{
const auto speedometer = static_cast< const Dial* >( skinnable );
switch ( nodeRole )
{
case PanelRole:
return updateBoxNode( speedometer, node, Dial::Panel );
case KnobRole:
return updateBoxNode( speedometer, node, Dial::Knob );
case NeedleRole:
return updateNeedleNode( speedometer, node );
case LabelsRole:
return updateLabelsNode( speedometer, node );
default:
return nullptr;
}
}
QSGNode* DialSkinlet::updateLabelsNode(
const Dial* dial, QSGNode* node ) const
{
using Q = Dial;
const auto labels = dial->tickLabels();
// ### actually, we could draw labels with only one entry
if ( labels.count() <= 1 )
return nullptr;
auto ticksNode = QskSGNode::ensureNode< QskLinesNode >( node );
const auto color = dial->color( Q::TickLabels );
const auto startAngle = dial->minimum();
const auto endAngle = dial->maximum();
const auto step = ( endAngle - startAngle ) / ( labels.count() - 1 );
auto scaleRect = this->scaleRect( dial );
const auto center = scaleRect.center();
const auto radius = 0.5 * scaleRect.width();
const auto spacing = dial->spacingHint( Q::TickLabels );
const auto font = dial->effectiveFont( Q::TickLabels );
const QFontMetricsF fontMetrics( font );
auto angle = startAngle;
const auto tickSize = dial->strutSizeHint( Q::TickLabels );
const auto needleRadius = radius - tickSize.height();
// Create a series of tickmarks from minimum to maximum
auto labelNode = ticksNode->firstChild();
QVector< QLineF > ticks;
for ( int i = 0; i < labels.count(); ++i, angle += step )
{
const qreal cos = qFastCos( qDegreesToRadians( angle ) );
const qreal sin = qFastSin( qDegreesToRadians( angle ) );
const auto xStart = center.x() + radius * cos;
const auto yStart = center.y() + radius * sin;
const auto xEnd = center.x() + needleRadius * cos;
const auto yEnd = center.y() + needleRadius * sin;
ticks += QLineF( xStart, yStart, xEnd, yEnd );
const auto& text = labels.at( i );
if ( !text.isEmpty() )
{
const auto w = qskHorizontalAdvance( fontMetrics, text );
const auto h = fontMetrics.height();
const auto adjustX = ( -0.5 * cos - 0.5 ) * w;
const auto adjustY = ( -0.5 * sin - 0.5 ) * h;
const auto numbersX = xEnd - ( spacing * cos ) + adjustX;
const auto numbersY = yEnd - ( spacing * sin ) + adjustY;
const QRectF numbersRect( numbersX, numbersY, w, h );
labelNode = QskSkinlet::updateTextNode(
dial, labelNode, numbersRect, Qt::AlignCenter | Qt::AlignHCenter,
text, font, QskTextOptions(), QskTextColors( color ), Qsk::Normal );
if ( labelNode )
{
if ( labelNode->parent() != ticksNode )
ticksNode->appendChildNode( labelNode );
labelNode = labelNode->nextSibling();
}
}
}
ticksNode->updateLines( color, tickSize.width(), ticks );
return ticksNode;
}
QSGNode* DialSkinlet::updateNeedleNode(
const Dial* dial, QSGNode* node ) const
{
using Q = Dial;
const auto color = dial->color( Q::Needle );
const auto width = dial->metric( Q::Needle | QskAspect::Size );
const auto line = needlePoints( dial );
auto needleNode = QskSGNode::ensureNode< QskLinesNode >( node );
needleNode->updateLine( color, width * 2, line.p1(), line.p2() );
return needleNode;
}
QRectF DialSkinlet::scaleRect( const Dial* dial ) const
{
using Q = Dial;
const auto margins = dial->marginHint( Q::Panel );
auto r = dial->subControlRect( Q::Panel );
r = r.marginsRemoved( margins );
return r;
}
QLineF DialSkinlet::needlePoints( const Dial* dial ) const
{
const auto r = scaleRect( dial );
const auto margin = dial->metric( Dial::Needle | QskAspect::Margin );
QLineF line;
line.setP1( r.center() );
line.setLength( 0.5 * r.width() - margin );
line.setAngle( -dial->value() );
return line;
}
#include "moc_DialSkinlet.cpp"