qskinny/src/controls/QskVariantAnimator.cpp

236 lines
5.9 KiB
C++
Raw Normal View History

2017-07-21 18:21:34 +02:00
/******************************************************************************
* QSkinny - Copyright (C) 2016 Uwe Rathmann
* This file may be used under the terms of the QSkinny License, Version 1.0
*****************************************************************************/
#include "QskVariantAnimator.h"
2021-12-09 17:38:37 +01:00
#include "QskArcMetrics.h"
#include "QskBoxBorderColors.h"
2018-08-03 08:15:28 +02:00
#include "QskBoxBorderMetrics.h"
#include "QskBoxShapeMetrics.h"
2021-12-09 17:38:37 +01:00
#include "QskShadowMetrics.h"
2018-08-03 08:15:28 +02:00
#include "QskColorFilter.h"
#include "QskGradient.h"
2018-08-03 08:15:28 +02:00
#include "QskMargins.h"
2020-08-09 11:06:48 +02:00
#include "QskIntervalF.h"
2017-10-20 20:26:39 +02:00
#include "QskTextColors.h"
2017-07-21 18:21:34 +02:00
// Even if we don't use the standard Qt animation system we
// use its registry of interpolators: why adding our own ...
2018-07-19 14:10:48 +02:00
#include <qvariantanimation.h>
QSK_QT_PRIVATE_BEGIN
2021-09-21 08:51:01 +02:00
#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
#ifndef emit
#define emit
#include <private/qabstractanimation_p.h>
#undef emit
2021-09-21 08:51:01 +02:00
#endif
#endif
2017-07-21 18:21:34 +02:00
#include <private/qvariantanimation_p.h>
2018-07-19 14:10:48 +02:00
QSK_QT_PRIVATE_END
2017-07-21 18:21:34 +02:00
#if 1
static void qskRegisterInterpolator()
{
2018-08-03 08:15:28 +02:00
qRegisterAnimationInterpolator< QskColorFilter >( QskColorFilter::interpolate );
2020-08-09 11:06:48 +02:00
qRegisterAnimationInterpolator< QskIntervalF >( QskIntervalF::interpolate );
2018-08-03 08:15:28 +02:00
qRegisterAnimationInterpolator< QskMargins >( QskMargins::interpolate );
qRegisterAnimationInterpolator< QskGradient >( QskGradient::interpolate );
qRegisterAnimationInterpolator< QskBoxShapeMetrics >( QskBoxShapeMetrics::interpolate );
qRegisterAnimationInterpolator< QskBoxBorderMetrics >( QskBoxBorderMetrics::interpolate );
qRegisterAnimationInterpolator< QskBoxBorderColors >( QskBoxBorderColors::interpolate );
qRegisterAnimationInterpolator< QskTextColors >( QskTextColors::interpolate );
2021-12-09 17:38:37 +01:00
qRegisterAnimationInterpolator< QskShadowMetrics >( QskShadowMetrics::interpolate );
qRegisterAnimationInterpolator< QskArcMetrics >( QskArcMetrics::interpolate );
2017-07-21 18:21:34 +02:00
}
Q_CONSTRUCTOR_FUNCTION( qskRegisterInterpolator )
#endif
2018-08-03 08:15:28 +02:00
#if defined( Q_CC_CLANG )
#if __has_feature( address_sanitizer )
#define QSK_DECL_INSANE __attribute__( ( no_sanitize( "undefined" ) ) )
#endif
2017-12-07 11:54:06 +01:00
#endif
2017-12-07 12:59:05 +01:00
#if !defined( QSK_DECL_INSANE )
#define QSK_DECL_INSANE
2017-12-07 12:59:05 +01:00
#endif
2018-08-03 08:15:28 +02:00
QSK_DECL_INSANE static inline QVariant qskInterpolate(
void ( *interpolator )(), const QVariant& from, const QVariant& to, qreal progress )
2017-07-21 18:21:34 +02:00
{
2017-12-07 11:54:06 +01:00
#if 1
/*
how to get rid of the reported runtime error from the clang sanitizer,
when calling F( const T&, ... ) as G( const void* ... ); TODO ...
*/
#endif
2017-07-21 18:21:34 +02:00
auto f = reinterpret_cast< QVariantAnimation::Interpolator >( interpolator );
return f( from.constData(), to.constData(), progress );
}
2022-03-31 18:09:03 +02:00
#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
using QskMetaType = int;
static inline QskMetaType qskMetaType( const QVariant& v ) { return v.userType(); }
#else
using QskMetaType = QMetaType;
static inline QskMetaType qskMetaType( const QVariant& v ) { return v.metaType(); }
#endif
static inline QVariant qskDefaultVariant( QskMetaType type )
{
return QVariant( type, nullptr );
}
static inline QVariant qskConvertedVariant( const QVariant& from, QskMetaType type )
{
auto v = from;
v.convert( type );
return v;
}
2018-08-03 08:15:28 +02:00
QskVariantAnimator::QskVariantAnimator()
: m_interpolator( nullptr )
2017-07-21 18:21:34 +02:00
{
}
QskVariantAnimator::~QskVariantAnimator()
{
}
void QskVariantAnimator::setStartValue( const QVariant& value )
{
stop();
2017-07-21 18:21:34 +02:00
m_startValue = value;
}
void QskVariantAnimator::setEndValue( const QVariant& value )
{
stop();
2017-07-21 18:21:34 +02:00
m_endValue = value;
}
void QskVariantAnimator::setCurrentValue( const QVariant& value )
{
m_currentValue = value;
}
2022-03-31 18:09:03 +02:00
bool QskVariantAnimator::convertValues( QVariant& v1, QVariant& v2 )
2017-07-21 18:21:34 +02:00
{
2022-03-31 18:09:03 +02:00
if ( !v1.isValid() && !v2.isValid() )
return false;
const auto type1 = qskMetaType( v1 );
const auto type2 = qskMetaType( v2 );
2017-07-21 18:21:34 +02:00
2022-03-31 18:09:03 +02:00
if ( !v1.isValid() )
{
2022-03-31 18:09:03 +02:00
v1 = qskDefaultVariant( type2 );
return true;
}
2022-03-31 18:09:03 +02:00
if ( !v2.isValid() )
{
v2 = qskDefaultVariant( type1 );
return true;
}
2022-03-31 18:09:03 +02:00
if ( type1 != type2 )
{
if ( v1.canConvert( type2 ) )
{
2022-03-31 18:09:03 +02:00
v1.convert( type2 );
return true;
}
2022-03-31 18:09:03 +02:00
if ( v2.canConvert( type1 ) )
{
2022-03-31 18:09:03 +02:00
v2.convert( type1 );
return true;
}
2022-03-31 18:09:03 +02:00
return false;
}
2022-03-31 18:09:03 +02:00
return true;
}
void QskVariantAnimator::setup()
{
m_interpolator = nullptr;
if ( convertValues( m_startValue, m_endValue ) )
2017-07-21 18:21:34 +02:00
{
if ( m_startValue != m_endValue )
2022-03-31 18:09:03 +02:00
{
const auto id = m_startValue.userType();
// all what has been registered by qRegisterAnimationInterpolator
m_interpolator = reinterpret_cast< void ( * )() >(
QVariantAnimationPrivate::getInterpolator( id ) );
}
2017-07-21 18:21:34 +02:00
}
m_currentValue = m_interpolator ? m_startValue : m_endValue;
}
void QskVariantAnimator::advance( qreal progress )
{
if ( m_interpolator )
{
2017-12-22 14:15:24 +01:00
if ( qFuzzyCompare( progress, 1.0 ) )
progress = 1.0;
Q_ASSERT( qskMetaType( m_startValue ) == qskMetaType( m_endValue ) );
2017-07-21 18:21:34 +02:00
m_currentValue = qskInterpolate( m_interpolator,
m_startValue, m_endValue, progress );
}
}
void QskVariantAnimator::done()
{
m_interpolator = nullptr;
}
2022-03-31 18:09:03 +02:00
bool QskVariantAnimator::maybeInterpolate(
const QVariant& value1, const QVariant& value2 )
{
if ( !value1.isValid() && !value2.isValid() )
return false;
const auto type1 = qskMetaType( value1 );
const auto type2 = qskMetaType( value2 );
if ( !value1.isValid() )
return value2 != qskDefaultVariant( type2 );
if ( !value2.isValid() )
return value1 != qskDefaultVariant( type1 );
if ( type1 != type2 )
{
if ( value1.canConvert( type2 ) )
return value2 != qskConvertedVariant( value1, type2 );
if ( value2.canConvert( type1 ) )
return value1 != qskConvertedVariant( value2, type1 );
return false;
}
return value1 != value2;
}