/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** Commercial Usage ** ** Licensees holding valid Qt Commercial licenses may use this file in ** accordance with the Qt Commercial License Agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Nokia. ** ** GNU Lesser General Public License Usage ** ** Alternatively, 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. ** ** If you are unsure which license is appropriate for your use, please ** contact the sales department at http://qt.nokia.com/contact. ** **************************************************************************/ #include "qmlproject.h" #include "qmlprojectconstants.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace QmlProjectManager; using namespace QmlProjectManager::Internal; using namespace ProjectExplorer; //////////////////////////////////////////////////////////////////////////////////// // QmlProject //////////////////////////////////////////////////////////////////////////////////// QmlProject::QmlProject(Manager *manager, const QString &fileName) : m_manager(manager), m_fileName(fileName), m_modelManager(ExtensionSystem::PluginManager::instance()->getObject()) { QFileInfo fileInfo(m_fileName); m_projectName = fileInfo.completeBaseName(); m_file = new QmlProjectFile(this, fileName); m_rootNode = new QmlProjectNode(this, m_file); m_manager->registerProject(this); } QmlProject::~QmlProject() { m_manager->unregisterProject(this); delete m_rootNode; } QDir QmlProject::projectDir() const { return QFileInfo(file()->fileName()).dir(); } QString QmlProject::filesFileName() const { return m_fileName; } static QStringList readLines(const QString &absoluteFileName) { QStringList lines; QFile file(absoluteFileName); if (file.open(QFile::ReadOnly)) { QTextStream stream(&file); forever { QString line = stream.readLine(); if (line.isNull()) break; line = line.trimmed(); if (line.isEmpty()) continue; lines.append(line); } } return lines; } void QmlProject::parseProject(RefreshOptions options) { if (options & Files) { m_files = convertToAbsoluteFiles(readLines(filesFileName())); m_files.removeDuplicates(); m_modelManager->updateSourceFiles(m_files); } if (options & Configuration) { // update configuration } if (options & Files) emit fileListChanged(); } void QmlProject::refresh(RefreshOptions options) { QSet oldFileList; if (!(options & Configuration)) oldFileList = m_files.toSet(); parseProject(options); if (options & Files) m_rootNode->refresh(); } QStringList QmlProject::convertToAbsoluteFiles(const QStringList &paths) const { const QDir projectDir(QFileInfo(m_fileName).dir()); QStringList absolutePaths; foreach (const QString &file, paths) { QFileInfo fileInfo(projectDir, file); absolutePaths.append(fileInfo.absoluteFilePath()); } absolutePaths.removeDuplicates(); return absolutePaths; } QStringList QmlProject::files() const { return m_files; } QString QmlProject::name() const { return m_projectName; } Core::IFile *QmlProject::file() const { return m_file; } Manager *QmlProject::projectManager() const { return m_manager; } QList QmlProject::dependsOn() { return QList(); } bool QmlProject::isApplication() const { return true; } bool QmlProject::hasBuildSettings() const { return false; } ProjectExplorer::BuildConfigWidget *QmlProject::createConfigWidget() { return 0; } QList QmlProject::subConfigWidgets() { return QList(); } ProjectExplorer::IBuildConfigurationFactory *QmlProject::buildConfigurationFactory() const { return 0; } QmlProjectNode *QmlProject::rootProjectNode() const { return m_rootNode; } QStringList QmlProject::files(FilesMode) const { return m_files; } QStringList QmlProject::targets() const { QStringList targets; return targets; } bool QmlProject::restoreSettingsImpl(ProjectExplorer::PersistentSettingsReader &reader) { Project::restoreSettingsImpl(reader); if (runConfigurations().isEmpty()) { QmlRunConfiguration *runConf = new QmlRunConfiguration(this); addRunConfiguration(runConf); } refresh(Everything); return true; } void QmlProject::saveSettingsImpl(ProjectExplorer::PersistentSettingsWriter &writer) { Project::saveSettingsImpl(writer); } //////////////////////////////////////////////////////////////////////////////////// // QmlProjectFile //////////////////////////////////////////////////////////////////////////////////// QmlProjectFile::QmlProjectFile(QmlProject *parent, QString fileName) : Core::IFile(parent), m_project(parent), m_fileName(fileName) { } QmlProjectFile::~QmlProjectFile() { } bool QmlProjectFile::save(const QString &) { return false; } QString QmlProjectFile::fileName() const { return m_fileName; } QString QmlProjectFile::defaultPath() const { return QString(); } QString QmlProjectFile::suggestedFileName() const { return QString(); } QString QmlProjectFile::mimeType() const { return Constants::QMLMIMETYPE; } bool QmlProjectFile::isModified() const { return false; } bool QmlProjectFile::isReadOnly() const { return true; } bool QmlProjectFile::isSaveAsAllowed() const { return false; } void QmlProjectFile::modified(ReloadBehavior *) { } QmlRunConfiguration::QmlRunConfiguration(QmlProject *pro) : ProjectExplorer::RunConfiguration(pro), m_project(pro), m_type(Constants::QMLRUNCONFIGURATION), m_debugServerPort(3768) { setName(tr("QML Viewer")); // append creator/bin dir to search path (only useful for special creator-qml package) const QString searchPath = QString(qgetenv("PATH")) + Utils::SynchronousProcess::pathSeparator() + QCoreApplication::applicationDirPath() ; m_qmlViewerDefaultPath = Utils::SynchronousProcess::locateBinary(searchPath, QLatin1String("qmlviewer")); } QmlRunConfiguration::~QmlRunConfiguration() { } QString QmlRunConfiguration::type() const { return m_type; } QString QmlRunConfiguration::viewerPath() const { if (!m_qmlViewerCustomPath.isEmpty()) return m_qmlViewerCustomPath; return m_qmlViewerDefaultPath; } QStringList QmlRunConfiguration::viewerArguments() const { QStringList args; if (!m_qmlViewerArgs.isEmpty()) args.append(m_qmlViewerArgs); const QString s = mainScript(); if (! s.isEmpty()) args.append(s); return args; } QString QmlRunConfiguration::workingDirectory() const { QFileInfo projectFile(m_project->file()->fileName()); return projectFile.absolutePath(); } uint QmlRunConfiguration::debugServerPort() const { return m_debugServerPort; } QWidget *QmlRunConfiguration::configurationWidget() { QWidget *config = new QWidget; QFormLayout *form = new QFormLayout(config); QComboBox *combo = new QComboBox; QDir projectDir = m_project->projectDir(); QStringList files; files.append(tr("")); int currentIndex = -1; foreach (const QString &fn, m_project->files()) { QFileInfo fileInfo(fn); if (fileInfo.suffix() != QLatin1String("qml")) continue; QString fileName = projectDir.relativeFilePath(fn); if (fileName == m_scriptFile) currentIndex = files.size(); files.append(fileName); } combo->addItems(files); if (currentIndex != -1) combo->setCurrentIndex(currentIndex); connect(combo, SIGNAL(activated(QString)), this, SLOT(setMainScript(QString))); Utils::PathChooser *qmlViewer = new Utils::PathChooser; qmlViewer->setExpectedKind(Utils::PathChooser::Command); qmlViewer->setPath(viewerPath()); connect(qmlViewer, SIGNAL(changed(QString)), this, SLOT(onQmlViewerChanged())); QLineEdit *qmlViewerArgs = new QLineEdit; qmlViewerArgs->setText(m_qmlViewerArgs); connect(qmlViewerArgs, SIGNAL(textChanged(QString)), this, SLOT(onQmlViewerArgsChanged())); QSpinBox *debugPort = new QSpinBox; debugPort->setMinimum(1024); // valid registered/dynamic/free ports according to http://www.iana.org/assignments/port-numbers debugPort->setMaximum(65535); debugPort->setValue(m_debugServerPort); connect(debugPort, SIGNAL(valueChanged(int)), this, SLOT(onDebugServerPortChanged())); form->addRow(tr("QML Viewer"), qmlViewer); form->addRow(tr("QML Viewer arguments:"), qmlViewerArgs); form->addRow(tr("Main QML File:"), combo); form->addRow(tr("Debugging Port:"), debugPort); return config; } QString QmlRunConfiguration::mainScript() const { if (m_scriptFile.isEmpty() || m_scriptFile == tr("")) { Core::EditorManager *editorManager = Core::ICore::instance()->editorManager(); if (Core::IEditor *editor = editorManager->currentEditor()) { return editor->file()->fileName(); } } return m_project->projectDir().absoluteFilePath(m_scriptFile); } void QmlRunConfiguration::setMainScript(const QString &scriptFile) { m_scriptFile = scriptFile; } void QmlRunConfiguration::onQmlViewerChanged() { if (Utils::PathChooser *chooser = qobject_cast(sender())) { m_qmlViewerCustomPath = chooser->path(); } } void QmlRunConfiguration::onQmlViewerArgsChanged() { if (QLineEdit *lineEdit = qobject_cast(sender())) m_qmlViewerArgs = lineEdit->text(); } void QmlRunConfiguration::onDebugServerPortChanged() { if (QSpinBox *spinBox = qobject_cast(sender())) { m_debugServerPort = spinBox->value(); } } void QmlRunConfiguration::save(ProjectExplorer::PersistentSettingsWriter &writer) const { ProjectExplorer::RunConfiguration::save(writer); writer.saveValue(QLatin1String("qmlviewer"), m_qmlViewerCustomPath); writer.saveValue(QLatin1String("qmlviewerargs"), m_qmlViewerArgs); writer.saveValue(QLatin1String("mainscript"), m_scriptFile); writer.saveValue(QLatin1String("debugserverport"), m_debugServerPort); } void QmlRunConfiguration::restore(const ProjectExplorer::PersistentSettingsReader &reader) { ProjectExplorer::RunConfiguration::restore(reader); m_qmlViewerCustomPath = reader.restoreValue(QLatin1String("qmlviewer")).toString(); m_qmlViewerArgs = reader.restoreValue(QLatin1String("qmlviewerargs")).toString(); m_scriptFile = reader.restoreValue(QLatin1String("mainscript")).toString(); m_debugServerPort = reader.restoreValue(QLatin1String("debugserverport")).toUInt(); if (m_scriptFile.isEmpty()) m_scriptFile = tr(""); if (m_debugServerPort == 0) m_debugServerPort = 3768; } QmlRunConfigurationFactory::QmlRunConfigurationFactory() : m_type(Constants::QMLRUNCONFIGURATION) { } QmlRunConfigurationFactory::~QmlRunConfigurationFactory() { } bool QmlRunConfigurationFactory::canRestore(const QString &type) const { if (type.startsWith(m_type)) return true; return false; } QStringList QmlRunConfigurationFactory::availableCreationTypes(ProjectExplorer::Project *) const { return QStringList(); } QString QmlRunConfigurationFactory::displayNameForType(const QString &type) const { return type; } ProjectExplorer::RunConfiguration *QmlRunConfigurationFactory::create(ProjectExplorer::Project *project, const QString &) { QmlProject *pro = qobject_cast(project); return new QmlRunConfiguration(pro); } QmlRunControl::QmlRunControl(QmlRunConfiguration *runConfiguration, bool debugMode) : RunControl(runConfiguration), m_debugMode(debugMode) { Environment environment = ProjectExplorer::Environment::systemEnvironment(); if (debugMode) environment.set("QML_DEBUG_SERVER_PORT", QString::number(runConfiguration->debugServerPort())); m_applicationLauncher.setEnvironment(environment.toStringList()); m_applicationLauncher.setWorkingDirectory(runConfiguration->workingDirectory()); m_executable = runConfiguration->viewerPath(); m_commandLineArguments = runConfiguration->viewerArguments(); connect(&m_applicationLauncher, SIGNAL(applicationError(QString)), this, SLOT(slotError(QString))); connect(&m_applicationLauncher, SIGNAL(appendOutput(QString)), this, SLOT(slotAddToOutputWindow(QString))); connect(&m_applicationLauncher, SIGNAL(processExited(int)), this, SLOT(processExited(int))); connect(&m_applicationLauncher, SIGNAL(bringToForegroundRequested(qint64)), this, SLOT(slotBringApplicationToForeground(qint64))); } QmlRunControl::~QmlRunControl() { } void QmlRunControl::start() { m_applicationLauncher.start(ApplicationLauncher::Gui, m_executable, m_commandLineArguments); emit started(); emit addToOutputWindow(this, tr("Starting %1...").arg(QDir::toNativeSeparators(m_executable))); } void QmlRunControl::stop() { m_applicationLauncher.stop(); } bool QmlRunControl::isRunning() const { return m_applicationLauncher.isRunning(); } void QmlRunControl::slotBringApplicationToForeground(qint64 pid) { if (m_debugMode) { Core::ICore *core = Core::ICore::instance(); core->modeManager()->activateMode(QLatin1String("QML_INSPECT_MODE")); } bringApplicationToForeground(pid); } void QmlRunControl::slotError(const QString &err) { emit error(this, err); emit finished(); } void QmlRunControl::slotAddToOutputWindow(const QString &line) { emit addToOutputWindowInline(this, line); } void QmlRunControl::processExited(int exitCode) { emit addToOutputWindow(this, tr("%1 exited with code %2").arg(QDir::toNativeSeparators(m_executable)).arg(exitCode)); emit finished(); } QmlRunControlFactory::QmlRunControlFactory(QObject *parent) : IRunControlFactory(parent) { } QmlRunControlFactory::~QmlRunControlFactory() { } bool QmlRunControlFactory::canRun(RunConfiguration *runConfiguration, const QString &mode) const { return (qobject_cast(runConfiguration) != 0); } RunControl *QmlRunControlFactory::create(RunConfiguration *runConfiguration, const QString &mode) { QTC_ASSERT(canRun(runConfiguration, mode), return 0); return new QmlRunControl(qobject_cast(runConfiguration), mode == ProjectExplorer::Constants::DEBUGMODE); } QString QmlRunControlFactory::displayName() const { return tr("Run"); } QWidget *QmlRunControlFactory::configurationWidget(RunConfiguration *runConfiguration) { Q_UNUSED(runConfiguration) return new QLabel("TODO add Configuration widget"); }