/* === This file is part of Calamares - === * * SPDX-FileCopyrightText: 2020 Adriaan de Groot * SPDX-License-Identifier: GPL-3.0-or-later * * Calamares is Free Software: see the License-Identifier above. * */ #include "ThemeInfo.h" #include #include /** @brief describes a single plasma LnF theme. * * A theme description has an id, which is really the name of the desktop * file (e.g. org.kde.breeze.desktop), a name which is human-readable and * translated, and an optional image Page, which points to a local screenshot * of that theme. */ struct ThemeInfo { QString id; QString name; QString description; QString imagePath; bool show = true; ThemeInfo() {} explicit ThemeInfo( const QString& _id ) : id( _id ) { } explicit ThemeInfo( const QString& _id, const QString& image ) : id( _id ) , imagePath( image ) { } explicit ThemeInfo( const KPluginMetaData& ); bool isValid() const { return !id.isEmpty(); } }; class ThemeInfoList : public QList< ThemeInfo > { public: std::pair< int, const ThemeInfo* > indexById( const QString& id ) const { int index = 0; for ( const ThemeInfo& i : *this ) { if ( i.id == id ) { return { index, &i }; } } return { -1, nullptr }; } std::pair< int, ThemeInfo* > indexById( const QString& id ) { // Call the const version and then munge the types auto [ i, p ] = const_cast< const ThemeInfoList* >( this )->indexById( id ); return { i, const_cast< ThemeInfo* >( p ) }; } /** @brief Looks for a given @p id in the list of themes, returns nullptr if not found. */ ThemeInfo* findById( const QString& id ) { auto [ i, p ] = indexById( id ); return p; } /** @brief Looks for a given @p id in the list of themes, returns nullptr if not found. */ const ThemeInfo* findById( const QString& id ) const { auto [ i, p ] = indexById( id ); return p; } /** @brief Checks if a given @p id is in the list of themes. */ bool contains( const QString& id ) const { return findById( id ) != nullptr; } }; ThemesModel::ThemesModel( QObject* parent ) : QAbstractListModel( parent ) , m_themes( new ThemeInfoList ) { auto packages = KPackage::PackageLoader::self()->listPackages( "Plasma/LookAndFeel" ); m_themes->reserve( packages.length() ); for ( const auto& p : packages ) { m_themes->append( ThemeInfo { p } ); } } int ThemesModel::rowCount( const QModelIndex& ) const { return m_themes->count(); } QVariant ThemesModel::data( const QModelIndex& index, int role ) const { if ( !index.isValid() ) { return QVariant(); } if ( index.row() < 0 || index.row() >= m_themes->count() ) { return QVariant(); } const auto& item = m_themes->at( index.row() ); switch ( role ) { case LabelRole: return item.name; case KeyRole: return item.id; case ShownRole: return item.show; case DescriptionRole: return item.description; default: return QVariant(); } __builtin_unreachable(); } QHash< int, QByteArray > ThemesModel::roleNames() const { return { { LabelRole, "label" }, { KeyRole, "key" }, { ShownRole, "show" }, { ImageRole, "image" } }; } void ThemesModel::setThemeImage( const QString& id, const QString& imagePath ) { auto [ i, theme ] = m_themes->indexById( id ); if ( theme ) { theme->imagePath = imagePath; emit dataChanged( index( i, 0 ), index( i, 0 ), { ImageRole } ); } } void ThemesModel::setThemeImage( const QMap< QString, QString >& images ) { if ( m_themes->isEmpty() ) { return; } // Don't emit signals from each call, aggregate to one call (below this block) { QSignalBlocker b( this ); for ( const auto& k : images ) { setThemeImage( k, images[ k ] ); } } emit dataChanged( index( 0, 0 ), index( m_themes->count() - 1 ), { ImageRole } ); } void ThemesModel::showTheme( const QString& id, bool show ) { auto [ i, theme ] = m_themes->indexById( id ); if ( theme ) { theme->show = show; emit dataChanged( index( i, 0 ), index( i, 0 ), { ShownRole } ); } } void ThemesModel::showOnlyThemes( const QMap< QString, QString >& onlyThese ) { if ( m_themes->isEmpty() ) { return; } // No signal blocker block needed here because we're not calling showTheme() // QSignalBlocker b( this ); for ( auto& t : *m_themes ) { t.show = onlyThese.contains( t.id ); } emit dataChanged( index( 0, 0 ), index( m_themes->count() - 1 ), { ShownRole } ); } ThemeInfo::ThemeInfo( const KPluginMetaData& data ) : id( data.pluginId() ) , name( data.name() ) , description( data.description() ) { }