qskinny/src/controls/QskStatusIndicator.cpp
2022-02-08 08:59:34 +01:00

262 lines
5.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 "QskStatusIndicator.h"
#include "QskColorFilter.h"
#include "QskGraphic.h"
#include "QskGraphicProvider.h"
#include <qdebug.h>
QSK_SUBCONTROL( QskStatusIndicator, Graphic )
namespace
{
class StatusData
{
public:
StatusData( const QskGraphic& graphic )
: graphic( graphic )
, isDirty( false )
{
}
StatusData( const QUrl& url )
: source( url )
, isDirty( !url.isEmpty() )
{
}
void ensureGraphic( const QskStatusIndicator* indicator )
{
if ( !source.isEmpty() && isDirty )
{
graphic = indicator->loadSource( source );
isDirty = false;
}
}
QUrl source;
QskGraphic graphic;
bool isDirty : 1;
};
}
class QskStatusIndicator::PrivateData
{
public:
PrivateData()
: currentStatus( -1 )
{
}
int currentStatus;
QMap< int, StatusData > map;
mutable QList< int > statusList;
};
QskStatusIndicator::QskStatusIndicator( QQuickItem* parent )
: Inherited( parent )
, m_data( new PrivateData() )
{
initSizePolicy( QskSizePolicy::Expanding, QskSizePolicy::Expanding );
}
QskStatusIndicator::~QskStatusIndicator()
{
}
QUrl QskStatusIndicator::source( int status ) const
{
const auto it = m_data->map.constFind( status );
if ( it != m_data->map.constEnd() )
return it->source;
return QUrl();
}
void QskStatusIndicator::setSource( int status, const QString& source )
{
setSource( status, QUrl( source ) );
}
void QskStatusIndicator::setSource( int status, const QUrl& url )
{
bool hasChanged = false;
const auto it = m_data->map.find( status );
if ( it != m_data->map.end() )
{
if ( it->source != url )
{
it->source = url;
it->graphic.reset();
it->isDirty = !url.isEmpty();
hasChanged = true;
}
}
else
{
m_data->map.insert( status, StatusData( url ) );
m_data->statusList.clear();
hasChanged = true;
}
if ( hasChanged )
{
resetImplicitSize();
if ( status == m_data->currentStatus )
update();
}
}
QskGraphic QskStatusIndicator::graphic( int status ) const
{
const auto it = m_data->map.find( status );
if ( it != m_data->map.end() )
{
it->ensureGraphic( this );
return it->graphic;
}
return QskGraphic();
}
void QskStatusIndicator::setGraphic( int status, const QskGraphic& graphic )
{
bool hasChanged = false;
const auto it = m_data->map.find( status );
if ( it != m_data->map.end() )
{
if ( !it->source.isEmpty() || graphic != it->graphic )
{
it->source.clear();
it->isDirty = false;
it->graphic = graphic;
hasChanged = true;
}
}
else
{
m_data->map.insert( status, StatusData( graphic ) );
m_data->statusList.clear();
hasChanged = true;
}
if ( hasChanged )
{
resetImplicitSize();
if ( status == m_data->currentStatus )
update();
}
}
QskColorFilter QskStatusIndicator::graphicFilter( int status ) const
{
Q_UNUSED( status )
return effectiveGraphicFilter( Graphic );
}
void QskStatusIndicator::setGraphicRole( int role )
{
if ( setGraphicRoleHint( Graphic, role ) )
Q_EMIT graphicRoleChanged( role );
}
void QskStatusIndicator::resetGraphicRole()
{
if ( resetGraphicRoleHint( Graphic ) )
Q_EMIT graphicRoleChanged( graphicRoleHint( Graphic ) );
}
int QskStatusIndicator::graphicRole() const
{
return graphicRoleHint( Graphic );
}
int QskStatusIndicator::status() const
{
return m_data->currentStatus;
}
bool QskStatusIndicator::hasStatus( int status ) const
{
return m_data->map.contains( status );
}
void QskStatusIndicator::setStatus( int status )
{
if ( status == m_data->currentStatus )
return;
const auto it = m_data->map.constFind( status );
if ( it == m_data->map.constEnd() )
{
qWarning() << "QskStatusIndicator: invalid status:" << status;
return;
}
m_data->currentStatus = status;
Q_EMIT statusChanged( m_data->currentStatus );
// we should have a mode to decide if we
// want to keep the hidden graphics in memory
if ( it->isDirty )
polish();
update();
}
QList< int > QskStatusIndicator::statusList() const
{
/*
We should be have a QMap< int, QskGraphic >, so that
users can iterate over all entries without having to
do extra lookups for each entry. TODO ...
*/
if ( m_data->statusList.isEmpty() && !m_data->map.isEmpty() )
m_data->statusList = m_data->map.keys();
return m_data->statusList;
}
void QskStatusIndicator::changeEvent( QEvent* event )
{
if ( event->type() == QEvent::StyleChange )
{
for ( auto& statusData : m_data->map )
{
if ( !statusData.source.isEmpty() )
{
statusData.graphic.reset();
statusData.isDirty = true;
}
}
}
Inherited::changeEvent( event );
}
void QskStatusIndicator::updateLayout()
{
const auto it = m_data->map.find( m_data->currentStatus );
if ( it != m_data->map.end() )
it->ensureGraphic( this );
}
QskGraphic QskStatusIndicator::loadSource( const QUrl& url ) const
{
return Qsk::loadGraphic( url );
}
#include "moc_QskStatusIndicator.cpp"