2011-06-30 17:31:41 +02:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2012-01-26 18:33:46 +01:00
|
|
|
** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies).
|
2011-06-30 17:31:41 +02:00
|
|
|
**
|
2011-11-02 15:59:12 +01:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2011-06-30 17:31:41 +02:00
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
|
**
|
|
|
|
|
** This file may be used under the terms of the GNU Lesser General Public
|
|
|
|
|
** License version 2.1 as published by the Free Software Foundation and
|
|
|
|
|
** appearing in the file LICENSE.LGPL included in the packaging of this file.
|
|
|
|
|
** Please review the following information to ensure the GNU Lesser General
|
|
|
|
|
** Public License version 2.1 requirements will be met:
|
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
|
|
|
**
|
|
|
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
** Other Usage
|
|
|
|
|
**
|
|
|
|
|
** Alternatively, this file may be used in accordance with the terms and
|
|
|
|
|
** conditions contained in a signed written agreement between you and Nokia.
|
|
|
|
|
**
|
|
|
|
|
** If you have questions regarding the use of this file, please contact
|
2011-11-02 15:59:12 +01:00
|
|
|
** Nokia at qt-info@nokia.com.
|
2011-06-30 17:31:41 +02:00
|
|
|
**
|
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
2010-11-11 16:49:17 +01:00
|
|
|
#include "exampleslistmodel.h"
|
|
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QFile>
|
|
|
|
|
#include <QDir>
|
|
|
|
|
#include <QXmlStreamReader>
|
2010-11-11 16:49:17 +01:00
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
|
|
|
|
|
|
#include <coreplugin/icore.h>
|
2011-07-01 18:28:56 +02:00
|
|
|
#include <coreplugin/helpmanager.h>
|
2010-11-11 16:49:17 +01:00
|
|
|
#include <qtsupport/qtversionmanager.h>
|
2012-01-31 16:36:52 +01:00
|
|
|
#include <utils/qtcassert.h>
|
2010-11-11 16:49:17 +01:00
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
|
|
using QtSupport::QtVersionManager;
|
|
|
|
|
using QtSupport::BaseQtVersion;
|
|
|
|
|
|
|
|
|
|
namespace QtSupport {
|
|
|
|
|
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
ExamplesListModel::ExamplesListModel(QObject *parent) :
|
2012-01-31 16:36:52 +01:00
|
|
|
QAbstractListModel(parent),
|
2012-02-01 13:15:08 +01:00
|
|
|
m_updateOnQtVersionsChanged(false),
|
|
|
|
|
m_initialized(false),
|
|
|
|
|
m_helpInitialized(false)
|
2010-11-11 16:49:17 +01:00
|
|
|
{
|
|
|
|
|
QHash<int, QByteArray> roleNames;
|
|
|
|
|
roleNames[Name] = "name";
|
|
|
|
|
roleNames[ProjectPath] = "projectPath";
|
|
|
|
|
roleNames[ImageUrl] = "imageUrl";
|
|
|
|
|
roleNames[Description] = "description";
|
|
|
|
|
roleNames[DocUrl] = "docUrl";
|
|
|
|
|
roleNames[FilesToOpen] = "filesToOpen";
|
|
|
|
|
roleNames[Tags] = "tags";
|
|
|
|
|
roleNames[Difficulty] = "difficulty";
|
|
|
|
|
roleNames[Type] = "type";
|
|
|
|
|
roleNames[HasSourceCode] = "hasSourceCode";
|
2011-11-25 13:40:10 +01:00
|
|
|
roleNames[Dependencies] = "dependencies";
|
2011-11-23 12:25:32 +01:00
|
|
|
roleNames[IsVideo] = "isVideo";
|
|
|
|
|
roleNames[VideoUrl] = "videoUrl";
|
|
|
|
|
roleNames[VideoLength] = "videoLength";
|
2012-03-06 13:14:42 +01:00
|
|
|
roleNames[Platforms] = "platforms";
|
2010-11-11 16:49:17 +01:00
|
|
|
setRoleNames(roleNames);
|
|
|
|
|
|
2011-07-01 18:28:56 +02:00
|
|
|
connect(Core::HelpManager::instance(), SIGNAL(setupFinished()),
|
|
|
|
|
SLOT(helpInitialized()));
|
2012-02-24 18:57:33 +01:00
|
|
|
connect(QtVersionManager::instance(), SIGNAL(qtVersionsChanged(QList<int>,QList<int>,QList<int>)),
|
2012-01-31 16:36:52 +01:00
|
|
|
this, SLOT(handleQtVersionsChanged()));
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
2011-11-25 13:40:10 +01:00
|
|
|
static inline QString fixStringForTags(const QString &string)
|
2011-09-28 11:30:30 +02:00
|
|
|
{
|
|
|
|
|
QString returnString = string;
|
2011-09-28 15:25:07 +02:00
|
|
|
returnString.remove(QLatin1String("<i>"));
|
|
|
|
|
returnString.remove(QLatin1String("</i>"));
|
|
|
|
|
returnString.remove(QLatin1String("<tt>"));
|
|
|
|
|
returnString.remove(QLatin1String("</tt>"));
|
2011-09-28 11:30:30 +02:00
|
|
|
return returnString;
|
|
|
|
|
}
|
|
|
|
|
|
2012-03-06 13:14:42 +01:00
|
|
|
static inline QStringList trimStringList(const QStringList &stringlist)
|
|
|
|
|
{
|
|
|
|
|
QStringList returnList;
|
|
|
|
|
foreach (const QString &string, stringlist)
|
|
|
|
|
returnList << string.trimmed();
|
|
|
|
|
|
|
|
|
|
return returnList;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-11 16:49:17 +01:00
|
|
|
QList<ExampleItem> ExamplesListModel::parseExamples(QXmlStreamReader* reader, const QString& projectsOffset)
|
|
|
|
|
{
|
|
|
|
|
QList<ExampleItem> examples;
|
|
|
|
|
ExampleItem item;
|
2012-01-06 17:43:27 +01:00
|
|
|
const QChar slash = QLatin1Char('/');
|
2010-11-11 16:49:17 +01:00
|
|
|
while (!reader->atEnd()) {
|
|
|
|
|
switch (reader->readNext()) {
|
|
|
|
|
case QXmlStreamReader::StartElement:
|
|
|
|
|
if (reader->name() == QLatin1String("example")) {
|
|
|
|
|
item = ExampleItem();
|
|
|
|
|
item.type = Example;
|
|
|
|
|
QXmlStreamAttributes attributes = reader->attributes();
|
|
|
|
|
item.name = attributes.value(QLatin1String("name")).toString();
|
|
|
|
|
item.projectPath = attributes.value(QLatin1String("projectPath")).toString();
|
|
|
|
|
item.hasSourceCode = !item.projectPath.isEmpty();
|
2012-01-06 17:43:27 +01:00
|
|
|
item.projectPath.prepend(slash);
|
2010-11-11 16:49:17 +01:00
|
|
|
item.projectPath.prepend(projectsOffset);
|
2011-08-16 14:56:18 +02:00
|
|
|
item.imageUrl = attributes.value(QLatin1String("imageUrl")).toString();
|
2010-11-11 16:49:17 +01:00
|
|
|
item.docUrl = attributes.value(QLatin1String("docUrl")).toString();
|
|
|
|
|
} else if (reader->name() == QLatin1String("fileToOpen")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.filesToOpen.append(projectsOffset + slash + reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
2010-11-11 16:49:17 +01:00
|
|
|
} else if (reader->name() == QLatin1String("description")) {
|
2011-11-25 13:40:10 +01:00
|
|
|
item.description = fixStringForTags(reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
|
|
|
|
} else if (reader->name() == QLatin1String("dependency")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.dependencies.append(projectsOffset + slash + reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
2010-11-11 16:49:17 +01:00
|
|
|
} else if (reader->name() == QLatin1String("tags")) {
|
2012-03-06 13:14:42 +01:00
|
|
|
item.tags = trimStringList(reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement).split(QLatin1Char(','), QString::SkipEmptyParts));
|
2010-11-11 16:49:17 +01:00
|
|
|
m_tags.append(item.tags);
|
2012-03-06 13:14:42 +01:00
|
|
|
} else if (reader->name() == QLatin1String("platforms")) {
|
|
|
|
|
item.platforms = trimStringList(reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement).split(QLatin1Char(','), QString::SkipEmptyParts));
|
|
|
|
|
}
|
2010-11-11 16:49:17 +01:00
|
|
|
break;
|
|
|
|
|
case QXmlStreamReader::EndElement:
|
2012-03-27 18:39:31 +02:00
|
|
|
if (reader->name() == QLatin1String("example")) {
|
2012-03-27 19:00:11 +02:00
|
|
|
if (item.projectPath.isEmpty() || !QFileInfo(item.projectPath).exists())
|
2012-07-05 11:06:03 +03:00
|
|
|
item.tags.append(QLatin1String("broken"));
|
2012-03-27 19:00:11 +02:00
|
|
|
examples.append(item);
|
2012-03-27 18:39:31 +02:00
|
|
|
} else if (reader->name() == QLatin1String("examples")) {
|
2010-11-11 16:49:17 +01:00
|
|
|
return examples;
|
2012-03-27 18:39:31 +02:00
|
|
|
}
|
2010-11-11 16:49:17 +01:00
|
|
|
break;
|
|
|
|
|
default: // nothing
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return examples;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<ExampleItem> ExamplesListModel::parseDemos(QXmlStreamReader* reader, const QString& projectsOffset)
|
|
|
|
|
{
|
|
|
|
|
QList<ExampleItem> demos;
|
|
|
|
|
ExampleItem item;
|
2012-01-06 17:43:27 +01:00
|
|
|
const QChar slash = QLatin1Char('/');
|
2010-11-11 16:49:17 +01:00
|
|
|
while (!reader->atEnd()) {
|
|
|
|
|
switch (reader->readNext()) {
|
|
|
|
|
case QXmlStreamReader::StartElement:
|
|
|
|
|
if (reader->name() == QLatin1String("demo")) {
|
|
|
|
|
item = ExampleItem();
|
|
|
|
|
item.type = Demo;
|
|
|
|
|
QXmlStreamAttributes attributes = reader->attributes();
|
|
|
|
|
item.name = attributes.value(QLatin1String("name")).toString();
|
|
|
|
|
item.projectPath = attributes.value(QLatin1String("projectPath")).toString();
|
|
|
|
|
item.hasSourceCode = !item.projectPath.isEmpty();
|
2012-01-06 17:43:27 +01:00
|
|
|
item.projectPath.prepend(slash);
|
2010-11-11 16:49:17 +01:00
|
|
|
item.projectPath.prepend(projectsOffset);
|
|
|
|
|
item.imageUrl = attributes.value(QLatin1String("imageUrl")).toString();
|
|
|
|
|
item.docUrl = attributes.value(QLatin1String("docUrl")).toString();
|
|
|
|
|
} else if (reader->name() == QLatin1String("fileToOpen")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.filesToOpen.append(projectsOffset + slash + reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
2010-11-11 16:49:17 +01:00
|
|
|
} else if (reader->name() == QLatin1String("description")) {
|
2011-11-25 13:40:10 +01:00
|
|
|
item.description = fixStringForTags(reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
|
|
|
|
} else if (reader->name() == QLatin1String("dependency")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.dependencies.append(projectsOffset + slash + reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
2010-11-11 16:49:17 +01:00
|
|
|
} else if (reader->name() == QLatin1String("tags")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.tags = reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement).split(QLatin1Char(','));
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case QXmlStreamReader::EndElement:
|
|
|
|
|
if (reader->name() == QLatin1String("demo"))
|
|
|
|
|
demos.append(item);
|
|
|
|
|
else if (reader->name() == QLatin1String("demos"))
|
|
|
|
|
return demos;
|
|
|
|
|
break;
|
|
|
|
|
default: // nothing
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return demos;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<ExampleItem> ExamplesListModel::parseTutorials(QXmlStreamReader* reader, const QString& projectsOffset)
|
|
|
|
|
{
|
|
|
|
|
QList<ExampleItem> tutorials;
|
|
|
|
|
ExampleItem item;
|
2012-01-06 17:43:27 +01:00
|
|
|
const QChar slash = QLatin1Char('/');
|
2010-11-11 16:49:17 +01:00
|
|
|
while (!reader->atEnd()) {
|
|
|
|
|
switch (reader->readNext()) {
|
|
|
|
|
case QXmlStreamReader::StartElement:
|
|
|
|
|
if (reader->name() == QLatin1String("tutorial")) {
|
|
|
|
|
item = ExampleItem();
|
|
|
|
|
item.type = Tutorial;
|
|
|
|
|
QXmlStreamAttributes attributes = reader->attributes();
|
|
|
|
|
item.name = attributes.value(QLatin1String("name")).toString();
|
|
|
|
|
item.projectPath = attributes.value(QLatin1String("projectPath")).toString();
|
|
|
|
|
item.hasSourceCode = !item.projectPath.isEmpty();
|
2012-01-06 17:43:27 +01:00
|
|
|
item.projectPath.prepend(slash);
|
2010-11-11 16:49:17 +01:00
|
|
|
item.projectPath.prepend(projectsOffset);
|
2011-11-23 12:25:32 +01:00
|
|
|
if (attributes.hasAttribute(QLatin1String("imageUrl")))
|
|
|
|
|
item.imageUrl = attributes.value(QLatin1String("imageUrl")).toString();
|
|
|
|
|
if (attributes.hasAttribute(QLatin1String("docUrl")))
|
|
|
|
|
item.docUrl = attributes.value(QLatin1String("docUrl")).toString();
|
|
|
|
|
if (attributes.hasAttribute(QLatin1String("isVideo")))
|
|
|
|
|
item.isVideo = attributes.value(QLatin1String("isVideo")).toString() == QLatin1String("true");
|
|
|
|
|
if (attributes.hasAttribute(QLatin1String("videoUrl")))
|
|
|
|
|
item.videoUrl = attributes.value(QLatin1String("videoUrl")).toString();
|
|
|
|
|
if (attributes.hasAttribute(QLatin1String("videoLength")))
|
|
|
|
|
item.videoLength = attributes.value(QLatin1String("videoLength")).toString();
|
2010-11-11 16:49:17 +01:00
|
|
|
} else if (reader->name() == QLatin1String("fileToOpen")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.filesToOpen.append(projectsOffset + slash + reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
2010-11-11 16:49:17 +01:00
|
|
|
} else if (reader->name() == QLatin1String("description")) {
|
2011-11-25 13:40:10 +01:00
|
|
|
item.description = fixStringForTags(reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
|
|
|
|
} else if (reader->name() == QLatin1String("dependency")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.dependencies.append(projectsOffset + slash + reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement));
|
2010-11-11 16:49:17 +01:00
|
|
|
} else if (reader->name() == QLatin1String("tags")) {
|
2012-01-06 17:43:27 +01:00
|
|
|
item.tags = reader->readElementText(QXmlStreamReader::ErrorOnUnexpectedElement).split(QLatin1Char(','));
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case QXmlStreamReader::EndElement:
|
|
|
|
|
if (reader->name() == QLatin1String("tutorial"))
|
|
|
|
|
tutorials.append(item);
|
|
|
|
|
else if (reader->name() == QLatin1String("tutorials"))
|
|
|
|
|
return tutorials;
|
|
|
|
|
break;
|
|
|
|
|
default: // nothing
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return tutorials;
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-31 16:36:52 +01:00
|
|
|
void ExamplesListModel::handleQtVersionsChanged()
|
|
|
|
|
{
|
|
|
|
|
if (m_updateOnQtVersionsChanged)
|
|
|
|
|
updateExamples();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ExamplesListModel::updateExamples()
|
2010-11-11 16:49:17 +01:00
|
|
|
{
|
|
|
|
|
clear();
|
2012-01-31 16:36:52 +01:00
|
|
|
QString examplesFallback;
|
|
|
|
|
QString demosFallback;
|
|
|
|
|
QString sourceFallback;
|
|
|
|
|
foreach (const QString &exampleSource,
|
|
|
|
|
exampleSources(&examplesFallback, &demosFallback, &sourceFallback)) {
|
2010-11-11 16:49:17 +01:00
|
|
|
QFile exampleFile(exampleSource);
|
|
|
|
|
if (!exampleFile.open(QIODevice::ReadOnly)) {
|
|
|
|
|
qDebug() << Q_FUNC_INFO << "Could not open file" << exampleSource;
|
2012-01-31 16:36:52 +01:00
|
|
|
continue;
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QFileInfo fi(exampleSource);
|
|
|
|
|
QString offsetPath = fi.path();
|
|
|
|
|
QDir examplesDir(offsetPath);
|
|
|
|
|
QDir demosDir(offsetPath);
|
2012-01-31 16:36:52 +01:00
|
|
|
if (!examplesFallback.isEmpty()) {
|
|
|
|
|
// Look at Qt source directory at first,
|
|
|
|
|
// since examplesPath() / demosPath() points at the build directory
|
|
|
|
|
examplesDir = sourceFallback + QLatin1String("/examples");
|
|
|
|
|
demosDir = sourceFallback + QLatin1String("/demos");
|
|
|
|
|
// if examples or demos don't exist in source, try the directories
|
|
|
|
|
// that qmake -query gave (i.e. in the build directory)
|
2011-07-05 15:22:38 +02:00
|
|
|
if (!examplesDir.exists() || !demosDir.exists()) {
|
2012-01-31 16:36:52 +01:00
|
|
|
examplesDir = examplesFallback;
|
|
|
|
|
demosDir = demosFallback;
|
2011-07-05 15:22:38 +02:00
|
|
|
}
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QXmlStreamReader reader(&exampleFile);
|
|
|
|
|
while (!reader.atEnd())
|
|
|
|
|
switch (reader.readNext()) {
|
|
|
|
|
case QXmlStreamReader::StartElement:
|
|
|
|
|
if (reader.name() == QLatin1String("examples"))
|
|
|
|
|
addItems(parseExamples(&reader, examplesDir.path()));
|
|
|
|
|
else if (reader.name() == QLatin1String("demos"))
|
|
|
|
|
addItems(parseDemos(&reader, demosDir.path()));
|
|
|
|
|
else if (reader.name() == QLatin1String("tutorials"))
|
|
|
|
|
addItems(parseTutorials(&reader, examplesDir.path()));
|
|
|
|
|
break;
|
|
|
|
|
default: // nothing
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (reader.hasError())
|
|
|
|
|
qDebug() << "error parsing file" << exampleSource << "as XML document";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_tags.sort();
|
|
|
|
|
m_tags.erase(std::unique(m_tags.begin(), m_tags.end()), m_tags.end());
|
|
|
|
|
emit tagsUpdated();
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-31 16:36:52 +01:00
|
|
|
QStringList ExamplesListModel::exampleSources(QString *examplesFallback, QString *demosFallback,
|
|
|
|
|
QString *sourceFallback)
|
2010-11-11 16:49:17 +01:00
|
|
|
{
|
2012-01-31 16:36:52 +01:00
|
|
|
QTC_CHECK(examplesFallback);
|
|
|
|
|
QTC_CHECK(demosFallback);
|
|
|
|
|
QTC_CHECK(sourceFallback);
|
|
|
|
|
QStringList sources;
|
|
|
|
|
QString resourceDir = Core::ICore::resourcePath() + QLatin1String("/welcomescreen/");
|
|
|
|
|
|
|
|
|
|
// Qt Creator shipped tutorials
|
|
|
|
|
sources << (resourceDir + QLatin1String("/qtcreator_tutorials.xml"));
|
2010-11-11 16:49:17 +01:00
|
|
|
|
2011-08-03 14:19:10 +02:00
|
|
|
// Read keys from SDK installer
|
2012-01-24 15:36:40 +01:00
|
|
|
QSettings *settings = Core::ICore::settings(QSettings::SystemScope);
|
2012-01-06 17:43:27 +01:00
|
|
|
int size = settings->beginReadArray(QLatin1String("ExampleManifests"));
|
2011-08-03 14:19:10 +02:00
|
|
|
for (int i = 0; i < size; ++i) {
|
|
|
|
|
settings->setArrayIndex(i);
|
2012-01-06 17:43:27 +01:00
|
|
|
sources.append(settings->value(QLatin1String("Location")).toString());
|
2011-08-03 14:19:10 +02:00
|
|
|
}
|
|
|
|
|
settings->endArray();
|
2012-01-31 16:36:52 +01:00
|
|
|
// if the installer set something, that's enough for us
|
|
|
|
|
if (size > 0)
|
|
|
|
|
return sources;
|
|
|
|
|
|
|
|
|
|
// try to find a suitable Qt version
|
|
|
|
|
m_updateOnQtVersionsChanged = true; // this must be updated when the qt versions change
|
|
|
|
|
// fallbacks are passed back if no example manifest is found
|
|
|
|
|
// and we fallback to Qt Creator's shipped manifest (e.g. only old Qt Versions found)
|
|
|
|
|
QString potentialExamplesFallback;
|
|
|
|
|
QString potentialDemosFallback;
|
|
|
|
|
QString potentialSourceFallback;
|
|
|
|
|
bool potentialFallbackHasDeclarative = false; // we prefer Qt's with declarative as fallback
|
|
|
|
|
const QStringList pattern(QLatin1String("*.xml"));
|
2010-11-11 16:49:17 +01:00
|
|
|
|
2012-01-31 16:36:52 +01:00
|
|
|
QtVersionManager *versionManager = QtVersionManager::instance();
|
|
|
|
|
foreach (BaseQtVersion *version, versionManager->validVersions()) {
|
|
|
|
|
// There is no good solution for Qt 5 yet
|
|
|
|
|
if (version->qtVersion().majorVersion != 4)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
QFileInfoList fis;
|
|
|
|
|
if (version->hasExamples())
|
|
|
|
|
fis << QDir(version->examplesPath()).entryInfoList(pattern);
|
|
|
|
|
if (version->hasDemos())
|
|
|
|
|
fis << QDir(version->demosPath()).entryInfoList(pattern);
|
|
|
|
|
if (!fis.isEmpty()) {
|
|
|
|
|
foreach (const QFileInfo &fi, fis)
|
|
|
|
|
sources.append(fi.filePath());
|
|
|
|
|
return sources;
|
|
|
|
|
}
|
|
|
|
|
// check if this Qt version would be the preferred fallback
|
|
|
|
|
if (version->hasExamples() && version->hasDemos()) { // cached, so no performance hit
|
|
|
|
|
bool hasDeclarative = QDir(version->examplesPath() + QLatin1String("/declarative")).exists();
|
|
|
|
|
if (potentialExamplesFallback.isEmpty()
|
|
|
|
|
|| (!potentialFallbackHasDeclarative && hasDeclarative)) {
|
|
|
|
|
potentialFallbackHasDeclarative = hasDeclarative;
|
|
|
|
|
potentialExamplesFallback = version->examplesPath();
|
|
|
|
|
potentialDemosFallback = version->demosPath();
|
|
|
|
|
potentialSourceFallback = version->sourcePath().toString();
|
2011-10-27 10:32:06 +02:00
|
|
|
}
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-31 16:36:52 +01:00
|
|
|
if (!potentialExamplesFallback.isEmpty()) {
|
|
|
|
|
// We didn't find a manifest, use Creator-provided XML file with fall back Qt version
|
2011-06-30 17:37:02 +02:00
|
|
|
// qDebug() << Q_FUNC_INFO << "falling through to Creator-provided XML file";
|
2010-11-11 16:49:17 +01:00
|
|
|
sources << QString(resourceDir + QLatin1String("/examples_fallback.xml"));
|
2012-01-31 16:36:52 +01:00
|
|
|
if (examplesFallback)
|
|
|
|
|
*examplesFallback = potentialExamplesFallback;
|
|
|
|
|
if (demosFallback)
|
|
|
|
|
*demosFallback = potentialDemosFallback;
|
|
|
|
|
if (sourceFallback)
|
|
|
|
|
*sourceFallback = potentialSourceFallback;
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
2012-01-31 16:36:52 +01:00
|
|
|
return sources;
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ExamplesListModel::clear()
|
|
|
|
|
{
|
|
|
|
|
if (exampleItems.count() > 0) {
|
|
|
|
|
beginRemoveRows(QModelIndex(), 0, exampleItems.size()-1);
|
|
|
|
|
exampleItems.clear();
|
|
|
|
|
endRemoveRows();
|
|
|
|
|
}
|
|
|
|
|
m_tags.clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ExamplesListModel::addItems(const QList<ExampleItem> &newItems)
|
|
|
|
|
{
|
|
|
|
|
beginInsertRows(QModelIndex(), exampleItems.size(), exampleItems.size() - 1 + newItems.size());
|
|
|
|
|
exampleItems.append(newItems);
|
|
|
|
|
endInsertRows();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int ExamplesListModel::rowCount(const QModelIndex &) const
|
|
|
|
|
{
|
2012-02-01 13:15:08 +01:00
|
|
|
ensureInitialized();
|
2010-11-11 16:49:17 +01:00
|
|
|
return exampleItems.size();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariant ExamplesListModel::data(const QModelIndex &index, int role) const
|
|
|
|
|
{
|
2012-02-01 13:15:08 +01:00
|
|
|
ensureInitialized();
|
2010-11-11 16:49:17 +01:00
|
|
|
if (!index.isValid() || index.row()+1 > exampleItems.count()) {
|
|
|
|
|
qDebug() << Q_FUNC_INFO << "invalid index requested";
|
|
|
|
|
return QVariant();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ExampleItem item = exampleItems.at(index.row());
|
|
|
|
|
switch (role)
|
|
|
|
|
{
|
|
|
|
|
case Qt::DisplayRole: // for search only
|
2012-01-06 17:43:27 +01:00
|
|
|
return QString(item.name + QLatin1Char(' ') + item.tags.join(QLatin1String(" ")));
|
2010-11-11 16:49:17 +01:00
|
|
|
case Name:
|
|
|
|
|
return item.name;
|
|
|
|
|
case ProjectPath:
|
|
|
|
|
return item.projectPath;
|
|
|
|
|
case Description:
|
|
|
|
|
return item.description;
|
|
|
|
|
case ImageUrl:
|
|
|
|
|
return item.imageUrl;
|
|
|
|
|
case DocUrl:
|
|
|
|
|
return item.docUrl;
|
|
|
|
|
case FilesToOpen:
|
|
|
|
|
return item.filesToOpen;
|
|
|
|
|
case Tags:
|
|
|
|
|
return item.tags;
|
|
|
|
|
case Difficulty:
|
|
|
|
|
return item.difficulty;
|
2011-11-25 13:40:10 +01:00
|
|
|
case Dependencies:
|
|
|
|
|
return item.dependencies;
|
2010-11-11 16:49:17 +01:00
|
|
|
case HasSourceCode:
|
|
|
|
|
return item.hasSourceCode;
|
|
|
|
|
case Type:
|
|
|
|
|
return item.type;
|
2011-11-23 12:25:32 +01:00
|
|
|
case IsVideo:
|
|
|
|
|
return item.isVideo;
|
|
|
|
|
case VideoUrl:
|
|
|
|
|
return item.videoUrl;
|
|
|
|
|
case VideoLength:
|
|
|
|
|
return item.videoLength;
|
2012-03-06 13:14:42 +01:00
|
|
|
case Platforms:
|
|
|
|
|
return item.platforms;
|
2010-11-11 16:49:17 +01:00
|
|
|
default:
|
|
|
|
|
qDebug() << Q_FUNC_INFO << "role type not supported";
|
|
|
|
|
return QVariant();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-01 13:15:08 +01:00
|
|
|
QStringList ExamplesListModel::tags() const
|
|
|
|
|
{
|
|
|
|
|
ensureInitialized();
|
|
|
|
|
return m_tags;
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-01 18:28:56 +02:00
|
|
|
void ExamplesListModel::helpInitialized()
|
|
|
|
|
{
|
2012-02-01 13:15:08 +01:00
|
|
|
m_helpInitialized = true;
|
|
|
|
|
if (m_initialized) // if we are already initialized we need to update nevertheless
|
|
|
|
|
updateExamples();
|
2011-07-01 18:28:56 +02:00
|
|
|
}
|
|
|
|
|
|
2012-02-01 13:15:08 +01:00
|
|
|
void ExamplesListModel::ensureInitialized() const
|
|
|
|
|
{
|
|
|
|
|
if (m_initialized || !m_helpInitialized)
|
|
|
|
|
return;
|
|
|
|
|
ExamplesListModel *that = const_cast<ExamplesListModel *>(this);
|
|
|
|
|
that->m_initialized = true;
|
|
|
|
|
that->updateExamples();
|
|
|
|
|
}
|
2011-07-01 18:28:56 +02:00
|
|
|
|
2012-02-01 13:15:08 +01:00
|
|
|
ExamplesListModelFilter::ExamplesListModelFilter(ExamplesListModel *sourceModel, QObject *parent) :
|
2012-05-22 16:52:43 +02:00
|
|
|
QSortFilterProxyModel(parent), m_showTutorialsOnly(true), m_sourceModel(sourceModel), m_timerId(0)
|
2010-11-11 16:49:17 +01:00
|
|
|
{
|
|
|
|
|
connect(this, SIGNAL(showTutorialsOnlyChanged()), SLOT(updateFilter()));
|
2012-02-01 13:15:08 +01:00
|
|
|
setSourceModel(m_sourceModel);
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ExamplesListModelFilter::updateFilter()
|
|
|
|
|
{
|
2012-05-22 15:28:00 +02:00
|
|
|
ExamplesListModel *exampleListModel = qobject_cast<ExamplesListModel*>(sourceModel());
|
|
|
|
|
if (exampleListModel) {
|
|
|
|
|
exampleListModel->beginReset();
|
|
|
|
|
invalidateFilter();
|
|
|
|
|
exampleListModel->endReset();
|
|
|
|
|
}
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
2011-07-22 14:29:14 +02:00
|
|
|
bool containsSubString(const QStringList& list, const QString& substr, Qt::CaseSensitivity cs)
|
|
|
|
|
{
|
|
|
|
|
foreach (const QString &elem, list) {
|
|
|
|
|
if (elem.contains(substr, cs))
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-11 16:49:17 +01:00
|
|
|
bool ExamplesListModelFilter::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const
|
|
|
|
|
{
|
|
|
|
|
if (m_showTutorialsOnly) {
|
|
|
|
|
int type = sourceModel()->index(sourceRow, 0, sourceParent).data(Type).toInt();
|
|
|
|
|
if (type != Tutorial)
|
|
|
|
|
return false;
|
2011-07-22 14:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
2011-11-23 12:25:32 +01:00
|
|
|
if (!m_showTutorialsOnly) {
|
|
|
|
|
int type = sourceModel()->index(sourceRow, 0, sourceParent).data(Type).toInt();
|
|
|
|
|
if (type != Example)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-22 14:29:14 +02:00
|
|
|
const QStringList tags = sourceModel()->index(sourceRow, 0, sourceParent).data(Tags).toStringList();
|
|
|
|
|
|
|
|
|
|
if (!m_filterTags.isEmpty()) {
|
|
|
|
|
foreach(const QString &tag, m_filterTags) {
|
|
|
|
|
if (!tags.contains(tag, Qt::CaseInsensitive))
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!m_searchString.isEmpty()) {
|
|
|
|
|
const QString description = sourceModel()->index(sourceRow, 0, sourceParent).data(Description).toString();
|
|
|
|
|
const QString name = sourceModel()->index(sourceRow, 0, sourceParent).data(Name).toString();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
foreach(const QString &subString, m_searchString) {
|
|
|
|
|
bool wordMatch = false;
|
|
|
|
|
wordMatch |= (bool)name.contains(subString, Qt::CaseInsensitive);
|
|
|
|
|
if (wordMatch)
|
|
|
|
|
continue;
|
|
|
|
|
wordMatch |= containsSubString(tags, subString, Qt::CaseInsensitive);
|
|
|
|
|
if (wordMatch)
|
|
|
|
|
continue;
|
|
|
|
|
wordMatch |= (bool)description.contains(subString, Qt::CaseInsensitive);
|
|
|
|
|
if (!wordMatch)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2010-11-11 16:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ok = QSortFilterProxyModel::filterAcceptsRow(sourceRow, sourceParent);
|
|
|
|
|
if (!ok)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-01 13:15:08 +01:00
|
|
|
int ExamplesListModelFilter::rowCount(const QModelIndex &parent) const
|
|
|
|
|
{
|
|
|
|
|
m_sourceModel->ensureInitialized();
|
|
|
|
|
return QSortFilterProxyModel::rowCount(parent);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariant ExamplesListModelFilter::data(const QModelIndex &index, int role) const
|
|
|
|
|
{
|
|
|
|
|
m_sourceModel->ensureInitialized();
|
|
|
|
|
return QSortFilterProxyModel::data(index, role);
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-11 16:49:17 +01:00
|
|
|
void ExamplesListModelFilter::setShowTutorialsOnly(bool showTutorialsOnly)
|
|
|
|
|
{
|
|
|
|
|
m_showTutorialsOnly = showTutorialsOnly;
|
|
|
|
|
emit showTutorialsOnlyChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-22 16:52:43 +02:00
|
|
|
void ExamplesListModelFilter::delayedUpdateFilter()
|
|
|
|
|
{
|
|
|
|
|
if (m_timerId != 0)
|
|
|
|
|
killTimer(m_timerId);
|
|
|
|
|
|
|
|
|
|
m_timerId = startTimer(320);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ExamplesListModelFilter::timerEvent(QTimerEvent *timerEvent)
|
|
|
|
|
{
|
|
|
|
|
if (m_timerId == timerEvent->timerId()) {
|
|
|
|
|
updateFilter();
|
|
|
|
|
killTimer(m_timerId);
|
|
|
|
|
m_timerId = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-22 14:29:14 +02:00
|
|
|
struct SearchStringLexer {
|
|
|
|
|
QString code;
|
|
|
|
|
const QChar *codePtr;
|
|
|
|
|
QChar yychar;
|
|
|
|
|
QString yytext;
|
|
|
|
|
|
|
|
|
|
enum TokenKind {
|
|
|
|
|
END_OF_STRING = 0,
|
|
|
|
|
TAG,
|
|
|
|
|
STRING_LITERAL,
|
|
|
|
|
UNKNOWN
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
inline void yyinp() { yychar = *codePtr++; }
|
|
|
|
|
|
|
|
|
|
SearchStringLexer(const QString &code)
|
|
|
|
|
: code(code)
|
|
|
|
|
, codePtr(code.unicode())
|
2012-01-06 17:43:27 +01:00
|
|
|
, yychar(QLatin1Char(' ')) { }
|
2011-07-22 14:29:14 +02:00
|
|
|
|
|
|
|
|
int operator()() { return yylex(); }
|
|
|
|
|
|
|
|
|
|
int yylex() {
|
|
|
|
|
while (yychar.isSpace())
|
|
|
|
|
yyinp(); // skip all the spaces
|
|
|
|
|
|
|
|
|
|
yytext.clear();
|
|
|
|
|
|
|
|
|
|
if (yychar.isNull())
|
|
|
|
|
return END_OF_STRING;
|
|
|
|
|
|
|
|
|
|
QChar ch = yychar;
|
|
|
|
|
yyinp();
|
|
|
|
|
|
|
|
|
|
switch (ch.unicode()) {
|
|
|
|
|
case '"':
|
|
|
|
|
case '\'':
|
|
|
|
|
{
|
|
|
|
|
const QChar quote = ch;
|
|
|
|
|
yytext.clear();
|
|
|
|
|
while (!yychar.isNull()) {
|
|
|
|
|
if (yychar == quote) {
|
|
|
|
|
yyinp();
|
|
|
|
|
break;
|
2012-01-06 17:43:27 +01:00
|
|
|
} if (yychar == QLatin1Char('\\')) {
|
2011-07-22 14:29:14 +02:00
|
|
|
yyinp();
|
|
|
|
|
switch (yychar.unicode()) {
|
2012-01-06 17:43:27 +01:00
|
|
|
case '"': yytext += QLatin1Char('"'); yyinp(); break;
|
|
|
|
|
case '\'': yytext += QLatin1Char('\''); yyinp(); break;
|
|
|
|
|
case '\\': yytext += QLatin1Char('\\'); yyinp(); break;
|
2011-07-22 14:29:14 +02:00
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
yytext += yychar;
|
|
|
|
|
yyinp();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return STRING_LITERAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
default:
|
2012-01-06 17:43:27 +01:00
|
|
|
if (ch.isLetterOrNumber() || ch == QLatin1Char('_')) {
|
2011-07-22 14:29:14 +02:00
|
|
|
yytext.clear();
|
|
|
|
|
yytext += ch;
|
2012-01-06 17:43:27 +01:00
|
|
|
while (yychar.isLetterOrNumber() || yychar == QLatin1Char('_')) {
|
2011-07-22 14:29:14 +02:00
|
|
|
yytext += yychar;
|
|
|
|
|
yyinp();
|
|
|
|
|
}
|
2012-01-06 17:43:27 +01:00
|
|
|
if (yychar == QLatin1Char(':') && yytext == QLatin1String("tag")) {
|
2011-07-22 14:29:14 +02:00
|
|
|
yyinp();
|
|
|
|
|
return TAG;
|
|
|
|
|
}
|
|
|
|
|
return STRING_LITERAL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
yytext += ch;
|
|
|
|
|
return UNKNOWN;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void ExamplesListModelFilter::parseSearchString(const QString &arg)
|
|
|
|
|
{
|
|
|
|
|
QStringList tags;
|
|
|
|
|
QStringList searchTerms;
|
|
|
|
|
SearchStringLexer lex(arg);
|
|
|
|
|
bool isTag = false;
|
|
|
|
|
while (int tk = lex()) {
|
|
|
|
|
if (tk == SearchStringLexer::TAG) {
|
|
|
|
|
isTag = true;
|
|
|
|
|
searchTerms.append(lex.yytext);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tk == SearchStringLexer::STRING_LITERAL) {
|
|
|
|
|
if (isTag) {
|
|
|
|
|
searchTerms.pop_back();
|
|
|
|
|
tags.append(lex.yytext);
|
|
|
|
|
isTag = false;
|
|
|
|
|
} else {
|
|
|
|
|
searchTerms.append(lex.yytext);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setSearchStrings(searchTerms);
|
|
|
|
|
setFilterTags(tags);
|
2012-05-22 16:52:43 +02:00
|
|
|
delayedUpdateFilter();
|
2011-07-22 14:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
2010-11-11 16:49:17 +01:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace QtSupport
|