171 lines
4.6 KiB
C++
171 lines
4.6 KiB
C++
#include "AppListModel.hpp"
|
|
#include <QDebug>
|
|
#include <QProcess>
|
|
#include <QRegularExpression>
|
|
#include <QIcon>
|
|
#include <QPixmap>
|
|
#include <QUrl>
|
|
#include <QFile>
|
|
#include <QStandardPaths>
|
|
#include <QDir>
|
|
|
|
AppListModel::AppListModel(QObject *parent)
|
|
: QAbstractListModel(parent)
|
|
{
|
|
loadApps();
|
|
}
|
|
|
|
int AppListModel::rowCount(const QModelIndex &parent) const
|
|
{
|
|
Q_UNUSED(parent);
|
|
return static_cast<int>(m_filteredIndexes.size());
|
|
}
|
|
|
|
QVariant AppListModel::data(const QModelIndex &index, int role) const
|
|
{
|
|
if (!index.isValid() || !m_apps || index.row() >= static_cast<int>(m_filteredIndexes.size()))
|
|
return QVariant();
|
|
|
|
int appIndex = m_filteredIndexes[index.row()];
|
|
const dmenu::DesktopEntry &app = (*m_apps)[appIndex];
|
|
|
|
switch (role) {
|
|
case NameRole:
|
|
return QString::fromStdString(app.name);
|
|
case ExecRole:
|
|
return QString::fromStdString(app.exec);
|
|
case IdRole:
|
|
return QString::fromStdString(app.id);
|
|
case IconRole:
|
|
return getIconUrl(app);
|
|
default:
|
|
return QVariant();
|
|
}
|
|
}
|
|
|
|
QHash<int, QByteArray> AppListModel::roleNames() const
|
|
{
|
|
QHash<int, QByteArray> roles;
|
|
roles[NameRole] = "name";
|
|
roles[ExecRole] = "exec";
|
|
roles[IdRole] = "id";
|
|
roles[IconRole] = "icon";
|
|
return roles;
|
|
}
|
|
|
|
void AppListModel::loadApps()
|
|
{
|
|
beginResetModel();
|
|
m_apps = dmenu::get_dmenu_app_data();
|
|
updateFilteredApps();
|
|
endResetModel();
|
|
}
|
|
|
|
void AppListModel::launchApp(int index)
|
|
{
|
|
if (!m_apps || index < 0 || index >= static_cast<int>(m_filteredIndexes.size()))
|
|
return;
|
|
|
|
int appIndex = m_filteredIndexes[index];
|
|
const dmenu::DesktopEntry &app = (*m_apps)[appIndex];
|
|
|
|
// Parse exec command (remove %f, %u, %F, %U field codes if present)
|
|
QString command = QString::fromStdString(app.exec);
|
|
QRegularExpression fieldCodes("%[fuFU]");
|
|
command = command.replace(fieldCodes, "").trimmed();
|
|
|
|
qDebug() << "Launching:" << command;
|
|
|
|
// Use nohup and redirect output to /dev/null for proper detachment
|
|
QString detachedCommand = QString("nohup %1 >/dev/null 2>&1 &").arg(command);
|
|
QProcess::startDetached("/bin/sh", QStringList() << "-c" << detachedCommand);
|
|
}
|
|
|
|
QString AppListModel::searchText() const
|
|
{
|
|
return m_searchText;
|
|
}
|
|
|
|
void AppListModel::setSearchText(const QString &searchText)
|
|
{
|
|
if (m_searchText == searchText)
|
|
return;
|
|
|
|
m_searchText = searchText;
|
|
emit searchTextChanged();
|
|
|
|
beginResetModel();
|
|
updateFilteredApps();
|
|
endResetModel();
|
|
}
|
|
|
|
void AppListModel::updateFilteredApps()
|
|
{
|
|
m_filteredIndexes.clear();
|
|
|
|
if (!m_apps)
|
|
return;
|
|
|
|
for (size_t i = 0; i < m_apps->size(); ++i) {
|
|
const dmenu::DesktopEntry &app = (*m_apps)[i];
|
|
|
|
if (m_searchText.isEmpty() ||
|
|
QString::fromStdString(app.name).contains(m_searchText, Qt::CaseInsensitive)) {
|
|
m_filteredIndexes.push_back(static_cast<int>(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
QUrl AppListModel::getIconUrl(const dmenu::DesktopEntry &app) const
|
|
{
|
|
QString iconName = QString::fromStdString(app.icon_path);
|
|
|
|
if (iconName.isEmpty())
|
|
return QUrl();
|
|
|
|
// If it's already a full path, use it directly
|
|
if (iconName.startsWith('/')) {
|
|
if (QFile::exists(iconName)) {
|
|
return QUrl::fromLocalFile(iconName);
|
|
}
|
|
return QUrl();
|
|
}
|
|
|
|
// Use Qt's proper icon theme search which follows XDG spec
|
|
QIcon icon = QIcon::fromTheme(iconName);
|
|
if (icon.isNull()) {
|
|
return QUrl();
|
|
}
|
|
|
|
// Qt doesn't expose the resolved file path directly, so let's use QStandardPaths
|
|
// to search in the proper system directories
|
|
QStringList dataDirs = QStandardPaths::standardLocations(QStandardPaths::GenericDataLocation);
|
|
|
|
// Common icon subdirectories and sizes (prioritized)
|
|
QStringList iconSubDirs = {
|
|
"icons/hicolor/scalable/apps",
|
|
"icons/hicolor/48x48/apps",
|
|
"icons/hicolor/64x64/apps",
|
|
"icons/hicolor/32x32/apps",
|
|
"icons/hicolor/128x128/apps",
|
|
"icons/Adwaita/scalable/apps",
|
|
"icons/Adwaita/48x48/apps",
|
|
"pixmaps"
|
|
};
|
|
|
|
QStringList extensions = {"", ".png", ".svg", ".xpm"};
|
|
|
|
for (const QString &dataDir : dataDirs) {
|
|
for (const QString &iconSubDir : iconSubDirs) {
|
|
QString basePath = dataDir + "/" + iconSubDir + "/";
|
|
for (const QString &ext : extensions) {
|
|
QString fullPath = basePath + iconName + ext;
|
|
if (QFile::exists(fullPath)) {
|
|
return QUrl::fromLocalFile(fullPath);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return QUrl();
|
|
} |