shift-shell/containments/homescreens/folio/applicationlistmodel.cpp
Marco Allegretti dd7f087fac Add category filter panel to the app drawer
A panel between the drawer and the power panel lists the XDG
categories installed on the device as icon-and-label tiles.
Tapping one filters the app list; text search still applies
on top. The filter resets each time the drawer opens. Audio,
Video and Settings fold into their parent groups to match
Kickoff.
2026-04-18 13:20:13 +02:00

283 lines
8.1 KiB
C++

// SPDX-FileCopyrightText: 2014 Antonis Tsiapaliokas <antonis.tsiapaliokas@kde.org>
// SPDX-FileCopyrightText: 2022-2024 Devin Lin <devin@kde.org>
// SPDX-License-Identifier: GPL-2.0-or-later
#include "applicationlistmodel.h"
#include <QByteArray>
#include <QDebug>
#include <QModelIndex>
#include <QProcess>
#include <QQuickWindow>
#include <KApplicationTrader>
#include <KConfigGroup>
#include <KIO/ApplicationLauncherJob>
#include <KNotificationJobUiDelegate>
#include <KService>
#include <KSharedConfig>
#include <KSycoca>
#include <chrono>
using namespace std::chrono_literals;
ApplicationListModel::ApplicationListModel(HomeScreen *parent)
: QAbstractListModel(parent)
, m_homeScreen{parent}
, m_reloadAppsTimer{new QTimer{this}}
{
m_reloadAppsTimer->setSingleShot(true);
m_reloadAppsTimer->setInterval(100ms);
connect(m_reloadAppsTimer, &QTimer::timeout, this, &ApplicationListModel::sycocaDbChanged);
connect(KSycoca::self(), &KSycoca::databaseChanged, m_reloadAppsTimer, static_cast<void (QTimer::*)()>(&QTimer::start));
// initialize wayland window checking
KWayland::Client::ConnectionThread *connection = KWayland::Client::ConnectionThread::fromApplication(this);
if (!connection) {
return;
}
load();
}
ApplicationListModel::~ApplicationListModel() = default;
QHash<int, QByteArray> ApplicationListModel::roleNames() const
{
return {
{DelegateRole, QByteArrayLiteral("delegate")},
{NameRole, QByteArrayLiteral("name")},
{CategoriesRole, QByteArrayLiteral("categories")},
};
}
void ApplicationListModel::sycocaDbChanged()
{
load();
}
KService::List ApplicationListModel::queryApplications()
{
auto cfg = KSharedConfig::openConfig(QStringLiteral("applications-blacklistrc"));
auto blgroup = KConfigGroup(cfg, QStringLiteral("Applications"));
const QStringList blacklist = blgroup.readEntry("blacklist", QStringList());
auto filter = [blacklist](const KService::Ptr &service) -> bool {
if (service->noDisplay()) {
return false;
}
if (!service->showOnCurrentPlatform()) {
return false;
}
if (blacklist.contains(service->desktopEntryName())) {
return false;
}
return true;
};
return KApplicationTrader::query(filter);
}
void ApplicationListModel::load()
{
qDebug() << "Reloading folio app list...";
// This function supports dynamic insertions and deletions to the existing
// list depending on what is given from queryApplications().
QMap<QString, int> storageIdMap; // <storageId, index>
for (int i = 0; i < m_delegates.size(); ++i) {
const auto &delegate = m_delegates[i];
storageIdMap.insert(delegate->application()->storageId(), i);
}
const KService::List currentApps = queryApplications();
QList<KService::Ptr> toInsert;
for (const KService::Ptr &service : currentApps) {
auto it = storageIdMap.find(service->storageId());
if (it != storageIdMap.end()) {
// Service already in m_delegates
storageIdMap.erase(it);
} else {
// Service needs to be inserted into m_delegates
toInsert.append(std::move(service));
}
}
QList<int> toRemove;
for (int index : storageIdMap.values()) {
toRemove.append(index);
}
std::sort(toRemove.begin(), toRemove.end());
// Remove indices first, from end to start to avoid indices changing
for (int i = toRemove.size() - 1; i >= 0; --i) {
int ind = toRemove[i];
QString storageId;
if (m_delegates[ind]->application()) {
storageId = m_delegates[ind]->application()->storageId();
}
beginRemoveRows({}, ind, ind);
m_delegates.removeAt(ind);
endRemoveRows();
Q_EMIT applicationRemoved(storageId);
}
// Append new elements
for (const KService::Ptr &service : toInsert) {
FolioApplication::Ptr app = std::make_shared<FolioApplication>(service);
FolioDelegate::Ptr delegate = std::make_shared<FolioDelegate>(app);
beginInsertRows({}, m_delegates.size(), m_delegates.size());
m_delegates.append(delegate);
endInsertRows();
}
}
QVariant ApplicationListModel::data(const QModelIndex &index, int role) const
{
if (!index.isValid()) {
return QVariant();
}
FolioDelegate::Ptr delegate = m_delegates.at(index.row());
switch (role) {
case Qt::DisplayRole:
case DelegateRole:
return QVariant::fromValue(delegate.get());
case NameRole:
if (!delegate->application()) {
return QVariant();
}
return delegate->application()->name();
case CategoriesRole:
if (!delegate->application()) {
return QVariant();
}
return QVariant::fromValue(delegate->application()->categories());
default:
return QVariant();
}
}
int ApplicationListModel::rowCount(const QModelIndex &parent) const
{
if (parent.isValid()) {
return 0;
}
return m_delegates.count();
}
// Sub-categories merged into their canonical parent, mirroring Kickoff's grouping.
static QString normalizeCategory(const QString &cat)
{
if (cat == QLatin1String("Audio") || cat == QLatin1String("Video"))
return QStringLiteral("AudioVideo");
if (cat == QLatin1String("Settings"))
return QStringLiteral("System");
return cat;
}
static const QSet<QString> &mainCategories()
{
static const QSet<QString> s = {
QStringLiteral("AudioVideo"),
QStringLiteral("Development"),
QStringLiteral("Education"),
QStringLiteral("Game"),
QStringLiteral("Graphics"),
QStringLiteral("Network"),
QStringLiteral("Office"),
QStringLiteral("Science"),
QStringLiteral("System"),
QStringLiteral("Utility"),
};
return s;
}
QStringList ApplicationListModel::allCategories() const
{
QSet<QString> found;
for (const auto &del : m_delegates) {
if (!del->application())
continue;
for (const QString &raw : del->application()->categories()) {
const QString cat = normalizeCategory(raw);
if (mainCategories().contains(cat))
found.insert(cat);
}
}
QStringList result = found.values();
result.sort();
return result;
}
ApplicationListSearchModel::ApplicationListSearchModel(HomeScreen *parent, ApplicationListModel *model)
: QSortFilterProxyModel(parent)
, m_homeScreen{parent}
{
setSourceModel(model);
setFilterRole(ApplicationListModel::NameRole);
setFilterCaseSensitivity(Qt::CaseInsensitive);
setSortRole(ApplicationListModel::NameRole);
setSortCaseSensitivity(Qt::CaseInsensitive);
setSortLocaleAware(true);
sort(0, Qt::AscendingOrder);
}
QString ApplicationListSearchModel::categoryFilter() const
{
return m_categoryFilter;
}
void ApplicationListSearchModel::setCategoryFilter(const QString &filter)
{
if (m_categoryFilter == filter)
return;
m_categoryFilter = filter;
Q_EMIT categoryFilterChanged();
beginFilterChange();
endFilterChange(QSortFilterProxyModel::Direction::Rows);
}
bool ApplicationListSearchModel::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const
{
if (!QSortFilterProxyModel::filterAcceptsRow(sourceRow, sourceParent))
return false;
if (m_categoryFilter.isEmpty())
return true;
auto *src = static_cast<ApplicationListModel *>(sourceModel());
const QModelIndex idx = src->index(sourceRow, 0, sourceParent);
auto *del = src->data(idx, ApplicationListModel::DelegateRole).value<FolioDelegate *>();
if (!del || !del->application())
return false;
if (m_categoryFilter == QLatin1String("__favorites__"))
return m_homeScreen->favouritesModel()->containsApplication(del->application()->storageId());
// Match both the canonical name and any raw aliases it absorbs.
const QStringList &cats = del->application()->categories();
for (const QString &raw : cats) {
if (normalizeCategory(raw) == m_categoryFilter)
return true;
}
return false;
}