sneedmc/logic/LwjglVersionList.cpp

201 lines
4.6 KiB
C++
Raw Normal View History

/* Copyright 2013 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
2013-10-05 23:13:40 +00:00
*
* 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 "LwjglVersionList.h"
#include "MultiMC.h"
#include <QtNetwork>
#include <QtXml>
#include <QRegExp>
#include "logger/QsLog.h"
2013-10-05 23:13:40 +00:00
#define RSS_URL "http://sourceforge.net/projects/java-game-lib/rss"
2013-10-05 23:13:40 +00:00
LWJGLVersionList::LWJGLVersionList(QObject *parent) : QAbstractListModel(parent)
{
setLoading(false);
}
QVariant LWJGLVersionList::data(const QModelIndex &index, int role) const
{
if (!index.isValid())
return QVariant();
2013-10-05 23:13:40 +00:00
if (index.row() > count())
return QVariant();
2013-10-05 23:13:40 +00:00
const PtrLWJGLVersion version = at(index.row());
2013-10-05 23:13:40 +00:00
switch (role)
{
case Qt::DisplayRole:
return version->name();
2013-10-05 23:13:40 +00:00
case Qt::ToolTipRole:
return version->url();
2013-10-05 23:13:40 +00:00
default:
return QVariant();
}
}
QVariant LWJGLVersionList::headerData(int section, Qt::Orientation orientation, int role) const
{
switch (role)
{
case Qt::DisplayRole:
return "Version";
2013-10-05 23:13:40 +00:00
case Qt::ToolTipRole:
return "LWJGL version name.";
2013-10-05 23:13:40 +00:00
default:
return QVariant();
}
}
int LWJGLVersionList::columnCount(const QModelIndex &parent) const
{
return 1;
}
bool LWJGLVersionList::isLoading() const
{
return m_loading;
}
void LWJGLVersionList::loadList()
{
Q_ASSERT_X(!m_loading, "loadList", "list is already loading (m_loading is true)");
2013-10-05 23:13:40 +00:00
setLoading(true);
auto worker = MMC->qnam();
QNetworkRequest req(QUrl(RSS_URL));
req.setRawHeader("Accept", "application/rss+xml, text/xml, */*");
req.setRawHeader("User-Agent", "MultiMC/5.0 (Uncached)");
reply = worker->get(req);
connect(reply, SIGNAL(finished()), SLOT(netRequestComplete()));
}
inline QDomElement getDomElementByTagName(QDomElement parent, QString tagname)
{
QDomNodeList elementList = parent.elementsByTagName(tagname);
if (elementList.count())
return elementList.at(0).toElement();
else
return QDomElement();
}
void LWJGLVersionList::netRequestComplete()
{
if (reply->error() == QNetworkReply::NoError)
{
QRegExp lwjglRegex("lwjgl-(([0-9]\\.?)+)\\.zip");
2013-10-05 23:13:40 +00:00
Q_ASSERT_X(lwjglRegex.isValid(), "load LWJGL list", "LWJGL regex is invalid");
QDomDocument doc;
2013-10-05 23:13:40 +00:00
QString xmlErrorMsg;
int errorLine;
auto rawData = reply->readAll();
if (!doc.setContent(rawData, false, &xmlErrorMsg, &errorLine))
{
2013-10-05 23:13:40 +00:00
failed("Failed to load LWJGL list. XML error: " + xmlErrorMsg + " at line " +
QString::number(errorLine));
setLoading(false);
return;
}
2013-10-05 23:13:40 +00:00
QDomNodeList items = doc.elementsByTagName("item");
2013-10-05 23:13:40 +00:00
QList<PtrLWJGLVersion> tempList;
2013-10-05 23:13:40 +00:00
for (int i = 0; i < items.length(); i++)
{
Q_ASSERT_X(items.at(i).isElement(), "load LWJGL list",
"XML element isn't an element... wat?");
2013-10-05 23:13:40 +00:00
QDomElement linkElement = getDomElementByTagName(items.at(i).toElement(), "link");
if (linkElement.isNull())
{
QLOG_INFO() << "Link element" << i << "in RSS feed doesn't exist! Skipping.";
continue;
}
2013-10-05 23:13:40 +00:00
QString link = linkElement.text();
2013-10-05 23:13:40 +00:00
// Make sure it's a download link.
if (link.endsWith("/download") && link.contains(lwjglRegex))
{
QString name = link.mid(lwjglRegex.indexIn(link) + 6);
// Subtract 4 here to remove the .zip file extension.
name = name.left(lwjglRegex.matchedLength() - 10);
2013-10-05 23:13:40 +00:00
QUrl url(link);
if (!url.isValid())
{
QLOG_WARN() << "LWJGL version URL isn't valid:" << link << "Skipping.";
continue;
}
QLOG_INFO() << "Discovered LWGL version" << name << "at" << link;
tempList.append(LWJGLVersion::Create(name, link));
}
}
2013-10-05 23:13:40 +00:00
beginResetModel();
m_vlist.swap(tempList);
endResetModel();
2013-10-05 23:13:40 +00:00
QLOG_INFO() << "Loaded LWJGL list.";
finished();
}
else
{
failed("Failed to load LWJGL list. Network error: " + reply->errorString());
}
2013-10-05 23:13:40 +00:00
setLoading(false);
reply->deleteLater();
}
const PtrLWJGLVersion LWJGLVersionList::getVersion(const QString &versionName)
{
for (int i = 0; i < count(); i++)
{
QString name = at(i)->name();
2013-10-05 23:13:40 +00:00
if (name == versionName)
return at(i);
}
return PtrLWJGLVersion();
}
void LWJGLVersionList::failed(QString msg)
{
QLOG_ERROR() << msg;
emit loadListFailed(msg);
}
void LWJGLVersionList::finished()
{
emit loadListFinished();
}
void LWJGLVersionList::setLoading(bool loading)
{
m_loading = loading;
emit loadingStateUpdated(m_loading);
}