2022-05-26 22:18:54 +01:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
2023-08-04 18:41:47 +01:00
|
|
|
* Prism Launcher - Minecraft Launcher
|
2022-05-26 22:18:54 +01:00
|
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
2015-08-19 00:10:17 +01:00
|
|
|
*
|
2022-05-26 22:18:54 +01:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, version 3.
|
2015-08-19 00:10:17 +01:00
|
|
|
*
|
2022-05-26 22:18:54 +01:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2015-08-19 00:10:17 +01:00
|
|
|
*
|
2022-05-26 22:18:54 +01:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* This file incorporates work covered by the following copyright and
|
|
|
|
* permission notice:
|
|
|
|
*
|
|
|
|
* Copyright 2013-2021 MultiMC Contributors
|
|
|
|
*
|
|
|
|
* 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.
|
2015-08-19 00:10:17 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "WorldList.h"
|
2022-03-31 17:45:17 +01:00
|
|
|
|
2015-10-05 00:47:27 +01:00
|
|
|
#include <FileSystem.h>
|
2023-08-14 17:16:53 +01:00
|
|
|
#include <QDebug>
|
|
|
|
#include <QFileSystemWatcher>
|
2015-08-19 00:10:17 +01:00
|
|
|
#include <QMimeData>
|
2023-08-14 17:16:53 +01:00
|
|
|
#include <QString>
|
2015-08-19 00:10:17 +01:00
|
|
|
#include <QUrl>
|
|
|
|
#include <QUuid>
|
2023-08-14 17:16:53 +01:00
|
|
|
#include <Qt>
|
|
|
|
#include "Application.h"
|
2015-08-19 00:10:17 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
WorldList::WorldList(const QString& dir, BaseInstance* instance) : QAbstractListModel(), m_instance(instance), m_dir(dir)
|
2015-08-19 00:10:17 +01:00
|
|
|
{
|
2015-10-05 00:47:27 +01:00
|
|
|
FS::ensureFolderPathExists(m_dir.absolutePath());
|
2021-09-08 23:27:46 +01:00
|
|
|
m_dir.setFilter(QDir::Readable | QDir::NoDotAndDotDot | QDir::Files | QDir::Dirs);
|
2015-08-19 00:10:17 +01:00
|
|
|
m_dir.setSorting(QDir::Name | QDir::IgnoreCase | QDir::LocaleAware);
|
|
|
|
m_watcher = new QFileSystemWatcher(this);
|
|
|
|
is_watching = false;
|
2023-04-18 01:51:34 +01:00
|
|
|
connect(m_watcher, &QFileSystemWatcher::directoryChanged, this, &WorldList::directoryChanged);
|
2015-08-19 00:10:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void WorldList::startWatching()
|
|
|
|
{
|
2023-08-14 17:16:53 +01:00
|
|
|
if (is_watching) {
|
2016-08-06 23:44:30 +01:00
|
|
|
return;
|
|
|
|
}
|
2015-08-19 00:10:17 +01:00
|
|
|
update();
|
|
|
|
is_watching = m_watcher->addPath(m_dir.absolutePath());
|
2023-08-14 17:16:53 +01:00
|
|
|
if (is_watching) {
|
2015-08-19 00:10:17 +01:00
|
|
|
qDebug() << "Started watching " << m_dir.absolutePath();
|
2023-08-14 17:16:53 +01:00
|
|
|
} else {
|
2015-08-19 00:10:17 +01:00
|
|
|
qDebug() << "Failed to start watching " << m_dir.absolutePath();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void WorldList::stopWatching()
|
|
|
|
{
|
2023-08-14 17:16:53 +01:00
|
|
|
if (!is_watching) {
|
2016-08-06 23:44:30 +01:00
|
|
|
return;
|
|
|
|
}
|
2015-08-19 00:10:17 +01:00
|
|
|
is_watching = !m_watcher->removePath(m_dir.absolutePath());
|
2023-08-14 17:16:53 +01:00
|
|
|
if (!is_watching) {
|
2015-08-19 00:10:17 +01:00
|
|
|
qDebug() << "Stopped watching " << m_dir.absolutePath();
|
2023-08-14 17:16:53 +01:00
|
|
|
} else {
|
2015-08-19 00:10:17 +01:00
|
|
|
qDebug() << "Failed to stop watching " << m_dir.absolutePath();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WorldList::update()
|
|
|
|
{
|
|
|
|
if (!isValid())
|
|
|
|
return false;
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2015-08-19 00:10:17 +01:00
|
|
|
QList<World> newWorlds;
|
|
|
|
m_dir.refresh();
|
|
|
|
auto folderContents = m_dir.entryInfoList();
|
|
|
|
// if there are any untracked files...
|
2023-08-14 17:16:53 +01:00
|
|
|
for (QFileInfo entry : folderContents) {
|
|
|
|
if (!entry.isDir())
|
2015-09-09 22:53:33 +01:00
|
|
|
continue;
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2015-09-09 22:53:33 +01:00
|
|
|
World w(entry);
|
2023-08-14 17:16:53 +01:00
|
|
|
if (w.isValid()) {
|
2015-09-09 22:53:33 +01:00
|
|
|
newWorlds.append(w);
|
2015-08-19 00:10:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
beginResetModel();
|
2015-09-09 22:53:33 +01:00
|
|
|
worlds.swap(newWorlds);
|
2015-08-19 00:10:17 +01:00
|
|
|
endResetModel();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WorldList::directoryChanged(QString path)
|
|
|
|
{
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WorldList::isValid()
|
|
|
|
{
|
|
|
|
return m_dir.exists() && m_dir.isReadable();
|
|
|
|
}
|
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
QString WorldList::instDirPath() const
|
|
|
|
{
|
2023-04-30 03:38:51 +01:00
|
|
|
return QFileInfo(m_instance->instanceRoot()).absoluteFilePath();
|
2023-02-12 09:44:39 +00:00
|
|
|
}
|
|
|
|
|
2015-08-19 00:10:17 +01:00
|
|
|
bool WorldList::deleteWorld(int index)
|
|
|
|
{
|
|
|
|
if (index >= worlds.size() || index < 0)
|
|
|
|
return false;
|
2023-08-14 17:16:53 +01:00
|
|
|
World& m = worlds[index];
|
|
|
|
if (m.destroy()) {
|
2015-08-19 00:10:17 +01:00
|
|
|
beginRemoveRows(QModelIndex(), index, index);
|
|
|
|
worlds.removeAt(index);
|
|
|
|
endRemoveRows();
|
|
|
|
emit changed();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WorldList::deleteWorlds(int first, int last)
|
|
|
|
{
|
2023-08-14 17:16:53 +01:00
|
|
|
for (int i = first; i <= last; i++) {
|
|
|
|
World& m = worlds[i];
|
2015-08-19 00:10:17 +01:00
|
|
|
m.destroy();
|
|
|
|
}
|
|
|
|
beginRemoveRows(QModelIndex(), first, last);
|
|
|
|
worlds.erase(worlds.begin() + first, worlds.begin() + last + 1);
|
|
|
|
endRemoveRows();
|
|
|
|
emit changed();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-08-22 00:34:55 +01:00
|
|
|
bool WorldList::resetIcon(int row)
|
|
|
|
{
|
|
|
|
if (row >= worlds.size() || row < 0)
|
|
|
|
return false;
|
2023-08-14 17:16:53 +01:00
|
|
|
World& m = worlds[row];
|
|
|
|
if (m.resetIcon()) {
|
|
|
|
emit dataChanged(index(row), index(row), { WorldList::IconFileRole });
|
2020-08-22 00:34:55 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
int WorldList::columnCount(const QModelIndex& parent) const
|
2015-08-19 00:10:17 +01:00
|
|
|
{
|
2023-08-14 17:16:53 +01:00
|
|
|
return parent.isValid() ? 0 : 5;
|
2015-08-19 00:10:17 +01:00
|
|
|
}
|
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
QVariant WorldList::data(const QModelIndex& index, int role) const
|
2015-08-19 00:10:17 +01:00
|
|
|
{
|
|
|
|
if (!index.isValid())
|
|
|
|
return QVariant();
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2015-08-19 00:10:17 +01:00
|
|
|
int row = index.row();
|
|
|
|
int column = index.column();
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2015-08-19 00:10:17 +01:00
|
|
|
if (row < 0 || row >= worlds.size())
|
|
|
|
return QVariant();
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2022-03-31 17:45:17 +01:00
|
|
|
QLocale locale;
|
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
auto& world = worlds[row];
|
|
|
|
switch (role) {
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
switch (column) {
|
|
|
|
case NameColumn:
|
|
|
|
return world.name();
|
|
|
|
|
|
|
|
case GameModeColumn:
|
|
|
|
return world.gameType().toTranslatedString();
|
|
|
|
|
|
|
|
case LastPlayedColumn:
|
|
|
|
return world.lastPlayed();
|
|
|
|
|
|
|
|
case SizeColumn:
|
|
|
|
return locale.formattedDataSize(world.bytes());
|
|
|
|
|
|
|
|
case InfoColumn:
|
|
|
|
if (world.isSymLinkUnder(instDirPath())) {
|
|
|
|
return tr("This world is symbolically linked from elsewhere.");
|
|
|
|
}
|
|
|
|
if (world.isMoreThanOneHardLink()) {
|
|
|
|
return tr("\nThis world is hard linked elsewhere.");
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
case Qt::UserRole:
|
|
|
|
switch (column) {
|
|
|
|
case SizeColumn:
|
|
|
|
return QVariant::fromValue<qlonglong>(world.bytes());
|
2022-03-31 17:45:17 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
default:
|
|
|
|
return data(index, Qt::DisplayRole);
|
2023-02-12 09:44:39 +00:00
|
|
|
}
|
2023-08-14 17:16:53 +01:00
|
|
|
|
|
|
|
case Qt::ToolTipRole: {
|
|
|
|
if (column == InfoColumn) {
|
|
|
|
if (world.isSymLinkUnder(instDirPath())) {
|
|
|
|
return tr("Warning: This world is symbolically linked from elsewhere. Editing it will also change the original."
|
|
|
|
"\nCanonical Path: %1")
|
|
|
|
.arg(world.canonicalFilePath());
|
|
|
|
}
|
|
|
|
if (world.isMoreThanOneHardLink()) {
|
|
|
|
return tr("Warning: This world is hard linked elsewhere. Editing it will also change the original.");
|
|
|
|
}
|
2023-02-12 09:44:39 +00:00
|
|
|
}
|
2023-08-14 17:16:53 +01:00
|
|
|
return world.folderName();
|
2015-08-19 00:10:17 +01:00
|
|
|
}
|
2023-08-14 17:16:53 +01:00
|
|
|
case ObjectRole: {
|
|
|
|
return QVariant::fromValue<void*>((void*)&world);
|
2022-04-19 14:07:14 +01:00
|
|
|
}
|
2023-08-14 17:16:53 +01:00
|
|
|
case FolderRole: {
|
|
|
|
return QDir::toNativeSeparators(dir().absoluteFilePath(world.folderName()));
|
2023-02-12 09:44:39 +00:00
|
|
|
}
|
2023-08-14 17:16:53 +01:00
|
|
|
case SeedRole: {
|
|
|
|
return QVariant::fromValue<qlonglong>(world.seed());
|
|
|
|
}
|
|
|
|
case NameRole: {
|
|
|
|
return world.name();
|
|
|
|
}
|
|
|
|
case LastPlayedRole: {
|
|
|
|
return world.lastPlayed();
|
|
|
|
}
|
|
|
|
case SizeRole: {
|
|
|
|
return QVariant::fromValue<qlonglong>(world.bytes());
|
|
|
|
}
|
|
|
|
case IconFileRole: {
|
|
|
|
return world.iconFile();
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return QVariant();
|
2015-08-19 00:10:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-01 07:51:15 +01:00
|
|
|
QVariant WorldList::headerData(int section, [[maybe_unused]] Qt::Orientation orientation, int role) const
|
2015-08-19 00:10:17 +01:00
|
|
|
{
|
2023-08-14 17:16:53 +01:00
|
|
|
switch (role) {
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
switch (section) {
|
|
|
|
case NameColumn:
|
|
|
|
return tr("Name");
|
|
|
|
case GameModeColumn:
|
|
|
|
return tr("Game Mode");
|
|
|
|
case LastPlayedColumn:
|
|
|
|
return tr("Last Played");
|
|
|
|
case SizeColumn:
|
|
|
|
//: World size on disk
|
|
|
|
return tr("Size");
|
|
|
|
case InfoColumn:
|
|
|
|
//: special warnings?
|
|
|
|
return tr("Info");
|
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
case Qt::ToolTipRole:
|
|
|
|
switch (section) {
|
|
|
|
case NameColumn:
|
|
|
|
return tr("The name of the world.");
|
|
|
|
case GameModeColumn:
|
|
|
|
return tr("Game mode of the world.");
|
|
|
|
case LastPlayedColumn:
|
|
|
|
return tr("Date and time the world was last played.");
|
|
|
|
case SizeColumn:
|
|
|
|
return tr("Size of the world on disk.");
|
|
|
|
case InfoColumn:
|
|
|
|
return tr("Information and warnings about the world.");
|
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
2015-08-19 00:10:17 +01:00
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList WorldList::mimeTypes() const
|
|
|
|
{
|
|
|
|
QStringList types;
|
2015-09-09 22:53:33 +01:00
|
|
|
types << "text/uri-list";
|
2015-08-19 00:10:17 +01:00
|
|
|
return types;
|
|
|
|
}
|
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
class WorldMimeData : public QMimeData {
|
|
|
|
Q_OBJECT
|
2015-08-19 00:10:17 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
public:
|
|
|
|
WorldMimeData(QList<World> worlds) { m_worlds = worlds; }
|
|
|
|
QStringList formats() const { return QMimeData::formats() << "text/uri-list"; }
|
2015-09-13 03:21:26 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
protected:
|
2022-05-02 20:34:55 +01:00
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
2023-08-14 17:16:53 +01:00
|
|
|
QVariant retrieveData(const QString& mimetype, QMetaType type) const
|
2022-05-02 20:34:55 +01:00
|
|
|
#else
|
2023-08-14 17:16:53 +01:00
|
|
|
QVariant retrieveData(const QString& mimetype, QVariant::Type type) const
|
2022-05-02 20:34:55 +01:00
|
|
|
#endif
|
2015-09-13 03:21:26 +01:00
|
|
|
{
|
|
|
|
QList<QUrl> urls;
|
2023-08-14 17:16:53 +01:00
|
|
|
for (auto& world : m_worlds) {
|
|
|
|
if (!world.isValid() || !world.isOnFS())
|
2015-09-13 03:21:26 +01:00
|
|
|
continue;
|
|
|
|
QString worldPath = world.container().absoluteFilePath();
|
|
|
|
qDebug() << worldPath;
|
|
|
|
urls.append(QUrl::fromLocalFile(worldPath));
|
|
|
|
}
|
|
|
|
const_cast<WorldMimeData*>(this)->setUrls(urls);
|
|
|
|
return QMimeData::retrieveData(mimetype, type);
|
|
|
|
}
|
2023-08-14 17:16:53 +01:00
|
|
|
|
|
|
|
private:
|
2015-09-13 03:21:26 +01:00
|
|
|
QList<World> m_worlds;
|
|
|
|
};
|
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
QMimeData* WorldList::mimeData(const QModelIndexList& indexes) const
|
2015-09-13 03:21:26 +01:00
|
|
|
{
|
|
|
|
if (indexes.size() == 0)
|
|
|
|
return new QMimeData();
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2023-07-01 07:51:15 +01:00
|
|
|
QList<World> worlds_;
|
2023-08-14 17:16:53 +01:00
|
|
|
for (auto idx : indexes) {
|
|
|
|
if (idx.column() != 0)
|
2015-09-13 03:21:26 +01:00
|
|
|
continue;
|
|
|
|
int row = idx.row();
|
|
|
|
if (row < 0 || row >= this->worlds.size())
|
|
|
|
continue;
|
2023-07-01 07:51:15 +01:00
|
|
|
worlds_.append(this->worlds[row]);
|
2015-09-13 03:21:26 +01:00
|
|
|
}
|
2023-08-14 17:16:53 +01:00
|
|
|
if (!worlds_.size()) {
|
2015-09-13 03:21:26 +01:00
|
|
|
return new QMimeData();
|
|
|
|
}
|
2023-07-01 07:51:15 +01:00
|
|
|
return new WorldMimeData(worlds_);
|
2015-08-19 00:10:17 +01:00
|
|
|
}
|
2015-09-09 22:53:33 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
Qt::ItemFlags WorldList::flags(const QModelIndex& index) const
|
2015-09-09 22:53:33 +01:00
|
|
|
{
|
|
|
|
Qt::ItemFlags defaultFlags = QAbstractListModel::flags(index);
|
|
|
|
if (index.isValid())
|
2023-08-14 17:16:53 +01:00
|
|
|
return Qt::ItemIsUserCheckable | Qt::ItemIsDragEnabled | Qt::ItemIsDropEnabled | defaultFlags;
|
2015-09-09 22:53:33 +01:00
|
|
|
else
|
|
|
|
return Qt::ItemIsDropEnabled | defaultFlags;
|
|
|
|
}
|
|
|
|
|
|
|
|
Qt::DropActions WorldList::supportedDragActions() const
|
|
|
|
{
|
|
|
|
// move to other mod lists or VOID
|
|
|
|
return Qt::MoveAction;
|
|
|
|
}
|
|
|
|
|
|
|
|
Qt::DropActions WorldList::supportedDropActions() const
|
|
|
|
{
|
|
|
|
// copy from outside, move from within and other mod lists
|
|
|
|
return Qt::CopyAction | Qt::MoveAction;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WorldList::installWorld(QFileInfo filename)
|
|
|
|
{
|
|
|
|
qDebug() << "installing: " << filename.absoluteFilePath();
|
|
|
|
World w(filename);
|
2023-08-14 17:16:53 +01:00
|
|
|
if (!w.isValid()) {
|
2015-09-09 22:53:33 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
w.install(m_dir.absolutePath());
|
|
|
|
}
|
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
bool WorldList::dropMimeData(const QMimeData* data,
|
|
|
|
Qt::DropAction action,
|
|
|
|
[[maybe_unused]] int row,
|
|
|
|
[[maybe_unused]] int column,
|
|
|
|
[[maybe_unused]] const QModelIndex& parent)
|
2015-09-09 22:53:33 +01:00
|
|
|
{
|
|
|
|
if (action == Qt::IgnoreAction)
|
|
|
|
return true;
|
|
|
|
// check if the action is supported
|
|
|
|
if (!data || !(action & supportedDropActions()))
|
|
|
|
return false;
|
|
|
|
// files dropped from outside?
|
2023-08-14 17:16:53 +01:00
|
|
|
if (data->hasUrls()) {
|
2015-09-09 22:53:33 +01:00
|
|
|
bool was_watching = is_watching;
|
|
|
|
if (was_watching)
|
|
|
|
stopWatching();
|
|
|
|
auto urls = data->urls();
|
2023-08-14 17:16:53 +01:00
|
|
|
for (auto url : urls) {
|
2015-09-09 22:53:33 +01:00
|
|
|
// only local files may be dropped...
|
|
|
|
if (!url.isLocalFile())
|
|
|
|
continue;
|
|
|
|
QString filename = url.toLocalFile();
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2015-09-09 22:53:33 +01:00
|
|
|
QFileInfo worldInfo(filename);
|
2018-07-15 13:51:05 +01:00
|
|
|
|
2023-08-14 17:16:53 +01:00
|
|
|
if (!m_dir.entryInfoList().contains(worldInfo)) {
|
2015-09-14 22:49:32 +01:00
|
|
|
installWorld(worldInfo);
|
|
|
|
}
|
2015-09-09 22:53:33 +01:00
|
|
|
}
|
|
|
|
if (was_watching)
|
|
|
|
startWatching();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2015-09-13 03:21:26 +01:00
|
|
|
|
|
|
|
#include "WorldList.moc"
|