2021-01-18 08:28:54 +01:00
|
|
|
/* Copyright 2013-2021 MultiMC Contributors
|
2013-12-31 01:24:28 +01:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "IconList.h"
|
2015-10-05 01:47:27 +02:00
|
|
|
#include <FileSystem.h>
|
2013-12-31 01:24:28 +01:00
|
|
|
#include <QMap>
|
|
|
|
#include <QEventLoop>
|
|
|
|
#include <QMimeData>
|
|
|
|
#include <QUrl>
|
|
|
|
#include <QFileSystemWatcher>
|
2015-02-01 11:44:47 +01:00
|
|
|
#include <QSet>
|
2015-02-02 02:14:14 +01:00
|
|
|
#include <QDebug>
|
2013-12-31 01:24:28 +01:00
|
|
|
|
|
|
|
#define MAX_SIZE 1024
|
|
|
|
|
2016-11-10 02:54:53 +01:00
|
|
|
IconList::IconList(const QStringList &builtinPaths, QString path, QObject *parent) : QAbstractListModel(parent)
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
QSet<QString> builtinNames;
|
|
|
|
|
|
|
|
// add builtin icons
|
|
|
|
for(auto & builtinPath: builtinPaths)
|
|
|
|
{
|
|
|
|
QDir instance_icons(builtinPath);
|
|
|
|
auto file_info_list = instance_icons.entryInfoList(QDir::Files, QDir::Name);
|
|
|
|
for (auto file_info : file_info_list)
|
|
|
|
{
|
2022-05-13 17:21:35 -03:00
|
|
|
builtinNames.insert(file_info.completeBaseName());
|
2018-07-15 14:51:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
for(auto & builtinName : builtinNames)
|
|
|
|
{
|
|
|
|
addThemeIcon(builtinName);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_watcher.reset(new QFileSystemWatcher());
|
|
|
|
is_watching = false;
|
|
|
|
connect(m_watcher.get(), SIGNAL(directoryChanged(QString)),
|
|
|
|
SLOT(directoryChanged(QString)));
|
|
|
|
connect(m_watcher.get(), SIGNAL(fileChanged(QString)), SLOT(fileChanged(QString)));
|
|
|
|
|
|
|
|
directoryChanged(path);
|
2022-05-13 17:21:35 -03:00
|
|
|
|
|
|
|
// Forces the UI to update, so that lengthy icon names are shown properly from the start
|
|
|
|
emit iconUpdated({});
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void IconList::directoryChanged(const QString &path)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
QDir new_dir (path);
|
|
|
|
if(m_dir.absolutePath() != new_dir.absolutePath())
|
|
|
|
{
|
|
|
|
m_dir.setPath(path);
|
|
|
|
m_dir.refresh();
|
|
|
|
if(is_watching)
|
|
|
|
stopWatching();
|
|
|
|
startWatching();
|
|
|
|
}
|
|
|
|
if(!m_dir.exists())
|
|
|
|
if(!FS::ensureFolderPathExists(m_dir.absolutePath()))
|
|
|
|
return;
|
|
|
|
m_dir.refresh();
|
|
|
|
auto new_list = m_dir.entryList(QDir::Files, QDir::Name);
|
|
|
|
for (auto it = new_list.begin(); it != new_list.end(); it++)
|
|
|
|
{
|
|
|
|
QString &foo = (*it);
|
|
|
|
foo = m_dir.filePath(foo);
|
|
|
|
}
|
|
|
|
auto new_set = new_list.toSet();
|
|
|
|
QList<QString> current_list;
|
|
|
|
for (auto &it : icons)
|
|
|
|
{
|
|
|
|
if (!it.has(IconType::FileBased))
|
|
|
|
continue;
|
|
|
|
current_list.push_back(it.m_images[IconType::FileBased].filename);
|
|
|
|
}
|
|
|
|
QSet<QString> current_set = current_list.toSet();
|
|
|
|
|
|
|
|
QSet<QString> to_remove = current_set;
|
|
|
|
to_remove -= new_set;
|
|
|
|
|
|
|
|
QSet<QString> to_add = new_set;
|
|
|
|
to_add -= current_set;
|
|
|
|
|
|
|
|
for (auto remove : to_remove)
|
|
|
|
{
|
|
|
|
qDebug() << "Removing " << remove;
|
|
|
|
QFileInfo rmfile(remove);
|
2022-05-13 17:21:35 -03:00
|
|
|
QString key = rmfile.completeBaseName();
|
|
|
|
|
|
|
|
QString suffix = rmfile.suffix();
|
|
|
|
// The icon doesnt have a suffix, but it can have other .s in the name, so we account for those as well
|
|
|
|
if (suffix != "jpeg" && suffix != "png" && suffix != "jpg" && suffix != "ico" && suffix != "svg" && suffix != "gif")
|
|
|
|
key = rmfile.fileName();
|
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
int idx = getIconIndex(key);
|
|
|
|
if (idx == -1)
|
|
|
|
continue;
|
|
|
|
icons[idx].remove(IconType::FileBased);
|
|
|
|
if (icons[idx].type() == IconType::ToBeDeleted)
|
|
|
|
{
|
|
|
|
beginRemoveRows(QModelIndex(), idx, idx);
|
|
|
|
icons.remove(idx);
|
|
|
|
reindex();
|
|
|
|
endRemoveRows();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dataChanged(index(idx), index(idx));
|
|
|
|
}
|
|
|
|
m_watcher->removePath(remove);
|
|
|
|
emit iconUpdated(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto add : to_add)
|
|
|
|
{
|
|
|
|
qDebug() << "Adding " << add;
|
2022-05-13 17:21:35 -03:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
QFileInfo addfile(add);
|
2022-05-13 17:21:35 -03:00
|
|
|
QString key = addfile.completeBaseName();
|
|
|
|
|
|
|
|
QString suffix = addfile.suffix();
|
|
|
|
// The icon doesnt have a suffix, but it can have other .s in the name, so we account for those as well
|
|
|
|
if (suffix != "jpeg" && suffix != "png" && suffix != "jpg" && suffix != "ico" && suffix != "svg" && suffix != "gif")
|
|
|
|
key = addfile.fileName();
|
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
if (addIcon(key, QString(), addfile.filePath(), IconType::FileBased))
|
|
|
|
{
|
|
|
|
m_watcher->addPath(add);
|
|
|
|
emit iconUpdated(key);
|
|
|
|
}
|
|
|
|
}
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void IconList::fileChanged(const QString &path)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
qDebug() << "Checking " << path;
|
|
|
|
QFileInfo checkfile(path);
|
|
|
|
if (!checkfile.exists())
|
|
|
|
return;
|
2022-05-13 17:21:35 -03:00
|
|
|
QString key = checkfile.completeBaseName();
|
2018-07-15 14:51:05 +02:00
|
|
|
int idx = getIconIndex(key);
|
|
|
|
if (idx == -1)
|
|
|
|
return;
|
|
|
|
QIcon icon(path);
|
|
|
|
if (!icon.availableSizes().size())
|
|
|
|
return;
|
|
|
|
|
|
|
|
icons[idx].m_images[IconType::FileBased].icon = icon;
|
|
|
|
dataChanged(index(idx), index(idx));
|
|
|
|
emit iconUpdated(key);
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2014-07-01 01:48:09 +02:00
|
|
|
void IconList::SettingChanged(const Setting &setting, QVariant value)
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if(setting.id() != "IconsDir")
|
|
|
|
return;
|
2013-12-31 01:24:28 +01:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
directoryChanged(value.toString());
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void IconList::startWatching()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
auto abs_path = m_dir.absolutePath();
|
|
|
|
FS::ensureFolderPathExists(abs_path);
|
|
|
|
is_watching = m_watcher->addPath(abs_path);
|
|
|
|
if (is_watching)
|
|
|
|
{
|
|
|
|
qDebug() << "Started watching " << abs_path;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
qDebug() << "Failed to start watching " << abs_path;
|
|
|
|
}
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void IconList::stopWatching()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_watcher->removePaths(m_watcher->files());
|
|
|
|
m_watcher->removePaths(m_watcher->directories());
|
|
|
|
is_watching = false;
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QStringList IconList::mimeTypes() const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
QStringList types;
|
|
|
|
types << "text/uri-list";
|
|
|
|
return types;
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
Qt::DropActions IconList::supportedDropActions() const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
return Qt::CopyAction;
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2019-06-01 00:59:02 +02:00
|
|
|
bool IconList::dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent)
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if (action == Qt::IgnoreAction)
|
|
|
|
return true;
|
|
|
|
// check if the action is supported
|
|
|
|
if (!data || !(action & supportedDropActions()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// files dropped from outside?
|
|
|
|
if (data->hasUrls())
|
|
|
|
{
|
|
|
|
auto urls = data->urls();
|
|
|
|
QStringList iconFiles;
|
|
|
|
for (auto url : urls)
|
|
|
|
{
|
|
|
|
// only local files may be dropped...
|
|
|
|
if (!url.isLocalFile())
|
|
|
|
continue;
|
|
|
|
iconFiles += url.toLocalFile();
|
|
|
|
}
|
|
|
|
installIcons(iconFiles);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Qt::ItemFlags IconList::flags(const QModelIndex &index) const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
Qt::ItemFlags defaultFlags = QAbstractListModel::flags(index);
|
|
|
|
if (index.isValid())
|
|
|
|
return Qt::ItemIsDropEnabled | defaultFlags;
|
|
|
|
else
|
|
|
|
return Qt::ItemIsDropEnabled | defaultFlags;
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant IconList::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if (!index.isValid())
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
int row = index.row();
|
|
|
|
|
|
|
|
if (row < 0 || row >= icons.size())
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
switch (role)
|
|
|
|
{
|
|
|
|
case Qt::DecorationRole:
|
|
|
|
return icons[row].icon();
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
return icons[row].name();
|
|
|
|
case Qt::UserRole:
|
|
|
|
return icons[row].m_key;
|
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int IconList::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
return icons.size();
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
void IconList::installIcons(const QStringList &iconFiles)
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
for (QString file : iconFiles)
|
|
|
|
{
|
|
|
|
QFileInfo fileinfo(file);
|
|
|
|
if (!fileinfo.isReadable() || !fileinfo.isFile())
|
|
|
|
continue;
|
2022-05-27 09:15:32 -03:00
|
|
|
QString target = FS::PathCombine(getDirectory(), fileinfo.fileName());
|
2018-07-15 14:51:05 +02:00
|
|
|
|
|
|
|
QString suffix = fileinfo.suffix();
|
2018-10-31 21:54:22 +01:00
|
|
|
if (suffix != "jpeg" && suffix != "png" && suffix != "jpg" && suffix != "ico" && suffix != "svg" && suffix != "gif")
|
2018-07-15 14:51:05 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!QFile::copy(file, target))
|
|
|
|
continue;
|
|
|
|
}
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2018-04-07 16:15:58 +02:00
|
|
|
void IconList::installIcon(const QString &file, const QString &name)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
QFileInfo fileinfo(file);
|
|
|
|
if(!fileinfo.isReadable() || !fileinfo.isFile())
|
|
|
|
return;
|
2018-04-07 16:15:58 +02:00
|
|
|
|
2022-05-27 09:15:32 -03:00
|
|
|
QString target = FS::PathCombine(getDirectory(), name);
|
2018-04-07 16:15:58 +02:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
QFile::copy(file, target);
|
2018-04-07 16:15:58 +02:00
|
|
|
}
|
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
bool IconList::iconFileExists(const QString &key) const
|
2015-06-01 01:19:12 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
auto iconEntry = icon(key);
|
|
|
|
if(!iconEntry)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return iconEntry->has(IconType::FileBased);
|
2015-06-01 01:19:12 +02:00
|
|
|
}
|
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
const MMCIcon *IconList::icon(const QString &key) const
|
2015-06-01 01:19:12 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
int iconIdx = getIconIndex(key);
|
|
|
|
if (iconIdx == -1)
|
|
|
|
return nullptr;
|
|
|
|
return &icons[iconIdx];
|
2015-06-01 01:19:12 +02:00
|
|
|
}
|
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
bool IconList::deleteIcon(const QString &key)
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
int iconIdx = getIconIndex(key);
|
|
|
|
if (iconIdx == -1)
|
|
|
|
return false;
|
|
|
|
auto &iconEntry = icons[iconIdx];
|
|
|
|
if (iconEntry.has(IconType::FileBased))
|
|
|
|
{
|
|
|
|
return QFile::remove(iconEntry.m_images[IconType::FileBased].filename);
|
|
|
|
}
|
|
|
|
return false;
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2016-11-10 02:54:53 +01:00
|
|
|
bool IconList::addThemeIcon(const QString& key)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
auto iter = name_index.find(key);
|
|
|
|
if (iter != name_index.end())
|
|
|
|
{
|
|
|
|
auto &oldOne = icons[*iter];
|
|
|
|
oldOne.replace(Builtin, key);
|
|
|
|
dataChanged(index(*iter), index(*iter));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// add a new icon
|
|
|
|
beginInsertRows(QModelIndex(), icons.size(), icons.size());
|
|
|
|
{
|
|
|
|
MMCIcon mmc_icon;
|
|
|
|
mmc_icon.m_name = key;
|
|
|
|
mmc_icon.m_key = key;
|
|
|
|
mmc_icon.replace(Builtin, key);
|
|
|
|
icons.push_back(mmc_icon);
|
|
|
|
name_index[key] = icons.size() - 1;
|
|
|
|
}
|
|
|
|
endInsertRows();
|
|
|
|
return true;
|
|
|
|
}
|
2016-11-10 02:54:53 +01:00
|
|
|
}
|
|
|
|
|
2016-10-20 01:02:28 +02:00
|
|
|
bool IconList::addIcon(const QString &key, const QString &name, const QString &path, const IconType type)
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
// replace the icon even? is the input valid?
|
|
|
|
QIcon icon(path);
|
|
|
|
if (icon.isNull())
|
|
|
|
return false;
|
|
|
|
auto iter = name_index.find(key);
|
|
|
|
if (iter != name_index.end())
|
|
|
|
{
|
|
|
|
auto &oldOne = icons[*iter];
|
|
|
|
oldOne.replace(type, icon, path);
|
|
|
|
dataChanged(index(*iter), index(*iter));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// add a new icon
|
|
|
|
beginInsertRows(QModelIndex(), icons.size(), icons.size());
|
|
|
|
{
|
|
|
|
MMCIcon mmc_icon;
|
|
|
|
mmc_icon.m_name = name;
|
|
|
|
mmc_icon.m_key = key;
|
|
|
|
mmc_icon.replace(type, icon, path);
|
|
|
|
icons.push_back(mmc_icon);
|
|
|
|
name_index[key] = icons.size() - 1;
|
|
|
|
}
|
|
|
|
endInsertRows();
|
|
|
|
return true;
|
|
|
|
}
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2016-10-20 01:02:28 +02:00
|
|
|
void IconList::saveIcon(const QString &key, const QString &path, const char * format) const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
auto icon = getIcon(key);
|
|
|
|
auto pixmap = icon.pixmap(128, 128);
|
|
|
|
pixmap.save(path, format);
|
2016-10-20 01:02:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-31 01:24:28 +01:00
|
|
|
void IconList::reindex()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
name_index.clear();
|
|
|
|
int i = 0;
|
|
|
|
for (auto &iter : icons)
|
|
|
|
{
|
|
|
|
name_index[iter.m_key] = i;
|
|
|
|
i++;
|
|
|
|
}
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2016-10-20 01:02:28 +02:00
|
|
|
QIcon IconList::getIcon(const QString &key) const
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
int icon_index = getIconIndex(key);
|
2013-12-31 01:24:28 +01:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
if (icon_index != -1)
|
|
|
|
return icons[icon_index].icon();
|
2013-12-31 01:24:28 +01:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
// Fallback for icons that don't exist.
|
2021-10-16 00:42:01 +02:00
|
|
|
icon_index = getIconIndex("grass");
|
2013-12-31 01:24:28 +01:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
if (icon_index != -1)
|
|
|
|
return icons[icon_index].icon();
|
|
|
|
return QIcon();
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2016-10-20 01:02:28 +02:00
|
|
|
int IconList::getIconIndex(const QString &key) const
|
2013-12-31 01:24:28 +01:00
|
|
|
{
|
2021-10-16 00:42:01 +02:00
|
|
|
auto iter = name_index.find(key == "default" ? "grass" : key);
|
2018-07-15 14:51:05 +02:00
|
|
|
if (iter != name_index.end())
|
|
|
|
return *iter;
|
2013-12-31 01:24:28 +01:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
return -1;
|
2013-12-31 01:24:28 +01:00
|
|
|
}
|
|
|
|
|
2018-02-05 02:01:12 +01:00
|
|
|
QString IconList::getDirectory() const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
return m_dir.absolutePath();
|
2018-02-05 02:01:12 +01:00
|
|
|
}
|
|
|
|
|
2014-01-02 18:51:40 +01:00
|
|
|
//#include "IconList.moc"
|