You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

498 lines
16 KiB

/*
SPDX-FileCopyrightText: 2003-2007 Fredrik Höglund <fredrik@kde.org>
SPDX-FileCopyrightText: 2019 Benjamin Port <benjamin.port@enioka.com>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include <config-X11.h>
#include "cursorthemedata.h"
#include "kcmcursortheme.h"
#include "../kcms-common_p.h"
#include "krdb.h"
#include "xcursor/cursortheme.h"
#include "xcursor/previewwidget.h"
#include "xcursor/sortproxymodel.h"
#include "xcursor/themeapplicator.h"
#include "xcursor/thememodel.h"
#include <KIO/CopyJob>
#include <KIO/DeleteJob>
#include <KIO/Job>
#include <KIO/JobUiDelegate>
#include <KLocalizedString>
#include <KMessageBox>
#include <KPluginFactory>
#include <KTar>
#include <KUrlRequesterDialog>
#include <QStandardItemModel>
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
#include <private/qtx11extras_p.h>
#else
#include <QX11Info>
#endif
#include <X11/Xcursor/Xcursor.h>
#include <X11/Xlib.h>
#include <updatelaunchenvjob.h>
#ifdef HAVE_XFIXES
#include <X11/extensions/Xfixes.h>
#endif
K_PLUGIN_FACTORY_WITH_JSON(CursorThemeConfigFactory, "kcm_cursortheme.json", registerPlugin<CursorThemeConfig>(); registerPlugin<CursorThemeData>();)
CursorThemeConfig::CursorThemeConfig(QObject *parent, const KPluginMetaData &data, const QVariantList &args)
: KQuickAddons::ManagedConfigModule(parent, data, args)
, m_data(new CursorThemeData(this))
, m_canInstall(true)
, m_canResize(true)
, m_canConfigure(true)
{
m_preferredSize = cursorThemeSettings()->cursorSize();
connect(cursorThemeSettings(), &CursorThemeSettings::cursorThemeChanged, this, &CursorThemeConfig::updateSizeComboBox);
qmlRegisterType<PreviewWidget>("org.kde.private.kcm_cursortheme", 1, 0, "PreviewWidget");
qmlRegisterAnonymousType<SortProxyModel>("SortProxyModel",1);
qmlRegisterAnonymousType<CursorThemeSettings>("CursorThemeSettings",1);
m_themeModel = new CursorThemeModel(this);
m_themeProxyModel = new SortProxyModel(this);
m_themeProxyModel->setSourceModel(m_themeModel);
// sort ordering is already case-insensitive; match that for filtering too
m_themeProxyModel->setFilterCaseSensitivity(Qt::CaseInsensitive);
m_themeProxyModel->sort(0, Qt::AscendingOrder);
m_sizesModel = new QStandardItemModel(this);
// Disable the install button if we can't install new themes to ~/.icons,
// or Xcursor isn't set up to look for cursor themes there.
if (!m_themeModel->searchPaths().contains(QDir::homePath() + "/.icons") || !iconsIsWritable()) {
setCanInstall(false);
}
connect(m_themeModel, &QAbstractItemModel::dataChanged, this, &CursorThemeConfig::settingsChanged);
connect(m_themeModel, &QAbstractItemModel::dataChanged, this, [this](const QModelIndex &start, const QModelIndex &end, const QVector<int> &roles) {
const QModelIndex currentThemeIndex = m_themeModel->findIndex(cursorThemeSettings()->cursorTheme());
if (roles.contains(CursorTheme::PendingDeletionRole) && currentThemeIndex.data(CursorTheme::PendingDeletionRole) == true
&& start.row() <= currentThemeIndex.row() && currentThemeIndex.row() <= end.row()) {
cursorThemeSettings()->setCursorTheme(m_themeModel->theme(m_themeModel->defaultIndex())->name());
}
});
}
CursorThemeConfig::~CursorThemeConfig()
{
}
CursorThemeSettings *CursorThemeConfig::cursorThemeSettings() const
{
return m_data->settings();
}
void CursorThemeConfig::setCanInstall(bool can)
{
if (m_canInstall == can) {
return;
}
m_canInstall = can;
Q_EMIT canInstallChanged();
}
bool CursorThemeConfig::canInstall() const
{
return m_canInstall;
}
void CursorThemeConfig::setCanResize(bool can)
{
if (m_canResize == can) {
return;
}
m_canResize = can;
Q_EMIT canResizeChanged();
}
bool CursorThemeConfig::canResize() const
{
return m_canResize;
}
void CursorThemeConfig::setCanConfigure(bool can)
{
if (m_canConfigure == can) {
return;
}
m_canConfigure = can;
Q_EMIT canConfigureChanged();
}
int CursorThemeConfig::preferredSize() const
{
return m_preferredSize;
}
void CursorThemeConfig::setPreferredSize(int size)
{
if (m_preferredSize == size) {
return;
}
m_preferredSize = size;
Q_EMIT preferredSizeChanged();
}
bool CursorThemeConfig::canConfigure() const
{
return m_canConfigure;
}
bool CursorThemeConfig::downloadingFile() const
{
return m_tempCopyJob;
}
QAbstractItemModel *CursorThemeConfig::cursorsModel()
{
return m_themeProxyModel;
}
QAbstractItemModel *CursorThemeConfig::sizesModel()
{
return m_sizesModel;
}
bool CursorThemeConfig::iconsIsWritable() const
{
const QFileInfo icons = QFileInfo(QDir::homePath() + "/.icons");
const QFileInfo home = QFileInfo(QDir::homePath());
return ((icons.exists() && icons.isDir() && icons.isWritable()) || (!icons.exists() && home.isWritable()));
}
void CursorThemeConfig::updateSizeComboBox()
{
// clear the combo box
m_sizesModel->clear();
// refill the combo box and adopt its icon size
int row = cursorThemeIndex(cursorThemeSettings()->cursorTheme());
QModelIndex selected = m_themeProxyModel->index(row, 0);
int maxIconWidth = 0;
int maxIconHeight = 0;
if (selected.isValid()) {
const CursorTheme *theme = m_themeProxyModel->theme(selected);
const QList<int> sizes = theme->availableSizes();
// only refill the combobox if there is more that 1 size
if (sizes.size() > 1) {
int i;
QList<int> comboBoxList;
QPixmap m_pixmap;
// insert the items
m_pixmap = theme->createIcon(0);
if (m_pixmap.width() > maxIconWidth) {
maxIconWidth = m_pixmap.width();
}
if (m_pixmap.height() > maxIconHeight) {
maxIconHeight = m_pixmap.height();
}
foreach (i, sizes) {
m_pixmap = theme->createIcon(i);
if (m_pixmap.width() > maxIconWidth) {
maxIconWidth = m_pixmap.width();
}
if (m_pixmap.height() > maxIconHeight) {
maxIconHeight = m_pixmap.height();
}
QStandardItem *item = new QStandardItem(QIcon(m_pixmap), QString::number(i));
item->setData(i);
m_sizesModel->appendRow(item);
comboBoxList << i;
}
// select an item
int size = m_preferredSize;
int selectItem = comboBoxList.indexOf(size);
// cursor size not available for this theme
if (selectItem < 0) {
/* Search the value next to cursor size. The first entry (0)
is ignored. (If cursor size would have been 0, then we
would had found it yet. As cursor size is not 0, we won't
default to "automatic size".)*/
int j;
int distance;
int smallestDistance;
selectItem = 1;
j = comboBoxList.value(selectItem);
size = j;
smallestDistance = qAbs(m_preferredSize - j);
for (int i = 2; i < comboBoxList.size(); ++i) {
j = comboBoxList.value(i);
distance = qAbs(m_preferredSize - j);
if (distance < smallestDistance || (distance == smallestDistance && j > m_preferredSize)) {
smallestDistance = distance;
selectItem = i;
size = j;
}
}
}
cursorThemeSettings()->setCursorSize(size);
}
}
// enable or disable the combobox
if (cursorThemeSettings()->isImmutable("cursorSize")) {
setCanResize(false);
} else {
setCanResize(m_sizesModel->rowCount() > 0);
}
// We need to Q_EMIT a cursorSizeChanged in all case to refresh UI
Q_EMIT cursorThemeSettings()->cursorSizeChanged();
}
int CursorThemeConfig::cursorSizeIndex(int cursorSize) const
{
if (m_sizesModel->rowCount() > 0) {
const auto items = m_sizesModel->findItems(QString::number(cursorSize));
if (items.count() == 1) {
return items.first()->row();
}
}
return -1;
}
int CursorThemeConfig::cursorSizeFromIndex(int index)
{
Q_ASSERT(index < m_sizesModel->rowCount() && index >= 0);
return m_sizesModel->item(index)->data().toInt();
}
int CursorThemeConfig::cursorThemeIndex(const QString &cursorTheme) const
{
auto results = m_themeProxyModel->findIndex(cursorTheme);
return results.row();
}
QString CursorThemeConfig::cursorThemeFromIndex(int index) const
{
QModelIndex idx = m_themeProxyModel->index(index, 0);
return idx.isValid() ? m_themeProxyModel->theme(idx)->name() : QString();
}
void CursorThemeConfig::save()
{
ManagedConfigModule::save();
setPreferredSize(cursorThemeSettings()->cursorSize());
int row = cursorThemeIndex(cursorThemeSettings()->cursorTheme());
QModelIndex selected = m_themeProxyModel->index(row, 0);
const CursorTheme *theme = selected.isValid() ? m_themeProxyModel->theme(selected) : nullptr;
if (!applyTheme(theme, cursorThemeSettings()->cursorSize())) {
Q_EMIT showInfoMessage(i18n("You have to restart the Plasma session for these changes to take effect."));
}
removeThemes();
notifyKcmChange(GlobalChangeType::CursorChanged);
}
void CursorThemeConfig::load()
{
ManagedConfigModule::load();
setPreferredSize(cursorThemeSettings()->cursorSize());
// Disable the listview and the buttons if we're in kiosk mode
if (cursorThemeSettings()->isImmutable(QStringLiteral("cursorTheme"))) {
setCanConfigure(false);
setCanInstall(false);
}
updateSizeComboBox(); // This handles also the kiosk mode
setNeedsSave(false);
}
void CursorThemeConfig::defaults()
{
ManagedConfigModule::defaults();
m_preferredSize = cursorThemeSettings()->cursorSize();
}
bool CursorThemeConfig::isSaveNeeded() const
{
return !m_themeModel->match(m_themeModel->index(0, 0), CursorTheme::PendingDeletionRole, true).isEmpty();
}
void CursorThemeConfig::ghnsEntryChanged(KNSCore::EntryWrapper *entry)
{
if (entry->entry().status() == KNS3::Entry::Deleted) {
for (const QString &deleted : entry->entry().uninstalledFiles()) {
auto list = QStringView(deleted).split(QLatin1Char('/'));
if (list.last() == QLatin1Char('*')) {
list.takeLast();
}
QModelIndex idx = m_themeModel->findIndex(list.last().toString());
if (idx.isValid()) {
m_themeModel->removeTheme(idx);
}
}
} else if (entry->entry().status() == KNS3::Entry::Installed) {
for (const QString &created : entry->entry().installedFiles()) {
QStringList list = created.split(QLatin1Char('/'));
if (list.last() == QLatin1Char('*')) {
list.takeLast();
}
// Because we sometimes get some extra slashes in the installed files list
list.removeAll({});
// Because we'll also get the containing folder, if it was not already there
// we need to ignore it.
if (list.last() == QLatin1String(".icons")) {
continue;
}
m_themeModel->addTheme(list.join(QLatin1Char('/')));
}
}
}
void CursorThemeConfig::installThemeFromFile(const QUrl &url)
{
if (url.isLocalFile()) {
installThemeFile(url.toLocalFile());
return;
}
if (m_tempCopyJob) {
return;
}
m_tempInstallFile.reset(new QTemporaryFile());
if (!m_tempInstallFile->open()) {
Q_EMIT showErrorMessage(i18n("Unable to create a temporary file."));
m_tempInstallFile.reset();
return;
}
m_tempCopyJob = KIO::file_copy(url, QUrl::fromLocalFile(m_tempInstallFile->fileName()), -1, KIO::Overwrite);
m_tempCopyJob->uiDelegate()->setAutoErrorHandlingEnabled(true);
Q_EMIT downloadingFileChanged();
connect(m_tempCopyJob, &KIO::FileCopyJob::result, this, [this, url](KJob *job) {
if (job->error() != KJob::NoError) {
Q_EMIT showErrorMessage(i18n("Unable to download the icon theme archive: %1", job->errorText()));
return;
}
installThemeFile(m_tempInstallFile->fileName());
m_tempInstallFile.reset();
});
connect(m_tempCopyJob, &QObject::destroyed, this, &CursorThemeConfig::downloadingFileChanged);
}
void CursorThemeConfig::installThemeFile(const QString &path)
{
KTar archive(path);
archive.open(QIODevice::ReadOnly);
const KArchiveDirectory *archiveDir = archive.directory();
QStringList themeDirs;
// Extract the dir names of the cursor themes in the archive, and
// append them to themeDirs
foreach (const QString &name, archiveDir->entries()) {
const KArchiveEntry *entry = archiveDir->entry(name);
if (entry->isDirectory() && entry->name().toLower() != "default") {
const KArchiveDirectory *dir = static_cast<const KArchiveDirectory *>(entry);
if (dir->entry("index.theme") && dir->entry("cursors")) {
themeDirs << dir->name();
}
}
}
if (themeDirs.isEmpty()) {
Q_EMIT showErrorMessage(i18n("The file is not a valid icon theme archive."));
return;
}
// The directory we'll install the themes to
QString destDir = QDir::homePath() + "/.icons/";
if (!QDir().mkpath(destDir)) {
Q_EMIT showErrorMessage(i18n("Failed to create 'icons' folder."));
return;
}
// Process each cursor theme in the archive
foreach (const QString &dirName, themeDirs) {
QDir dest(destDir + dirName);
if (dest.exists()) {
QString question = i18n(
"A theme named %1 already exists in your icon "
"theme folder. Do you want replace it with this one?",
dirName);
int answer = KMessageBox::warningContinueCancel(nullptr, question, i18n("Overwrite Theme?"), KStandardGuiItem::overwrite());
if (answer != KMessageBox::Continue) {
continue;
}
// ### If the theme that's being replaced is the current theme, it
// will cause cursor inconsistencies in newly started apps.
}
// ### Should we check if a theme with the same name exists in a global theme dir?
// If that's the case it will effectively replace it, even though the global theme
// won't be deleted. Checking for this situation is easy, since the global theme
// will be in the listview. Maybe this should never be allowed since it might
// result in strange side effects (from the average users point of view). OTOH
// a user might want to do this 'upgrade' a global theme.
const KArchiveDirectory *dir = static_cast<const KArchiveDirectory *>(archiveDir->entry(dirName));
dir->copyTo(dest.path());
m_themeModel->addTheme(dest);
}
archive.close();
Q_EMIT showSuccessMessage(i18n("Theme installed successfully."));
m_themeModel->refreshList();
}
void CursorThemeConfig::removeThemes()
{
const QModelIndexList indices = m_themeModel->match(m_themeModel->index(0, 0), CursorTheme::PendingDeletionRole, true, -1);
QList<QPersistentModelIndex> persistentIndices;
persistentIndices.reserve(indices.count());
std::transform(indices.constBegin(), indices.constEnd(), std::back_inserter(persistentIndices), [](const QModelIndex index) {
return QPersistentModelIndex(index);
});
for (const auto &idx : qAsConst(persistentIndices)) {
const CursorTheme *theme = m_themeModel->theme(idx);
// Delete the theme from the harddrive
KIO::del(QUrl::fromLocalFile(theme->path())); // async
// Remove the theme from the model
m_themeModel->removeTheme(idx);
}
// TODO:
// Since it's possible to substitute cursors in a system theme by adding a local
// theme with the same name, we shouldn't remove the theme from the list if it's
// still available elsewhere. We could add a
// bool CursorThemeModel::tryAddTheme(const QString &name), and call that, but
// since KIO::del() is an asynchronos operation, the theme we're deleting will be
// readded to the list again before KIO has removed it.
}
#include "kcmcursortheme.moc"