diff --git a/src/plugins/projectexplorer/debugginghelper.cpp b/src/plugins/projectexplorer/debugginghelper.cpp index 8af5ba3045612b045dd6f540ef2d5bc0bdee424c..eac74ef2c96df2ed4fb1b4b9d20b5d4800d03aa3 100644 --- a/src/plugins/projectexplorer/debugginghelper.cpp +++ b/src/plugins/projectexplorer/debugginghelper.cpp @@ -28,13 +28,15 @@ **************************************************************************/ #include "debugginghelper.h" + #include <coreplugin/icore.h> #include <QtCore/QFileInfo> +#include <QtCore/QCoreApplication> #include <QtCore/QHash> #include <QtCore/QProcess> #include <QtCore/QDir> #include <QtCore/QDateTime> -#include <QtGui/QApplication> + #include <QtGui/QDesktopServices> using namespace ProjectExplorer; @@ -44,7 +46,7 @@ QString DebuggingHelperLibrary::findSystemQt(const Environment &env) QStringList paths = env.path(); foreach (const QString &path, paths) { foreach (const QString &possibleCommand, possibleQMakeCommands()) { - QFileInfo qmake(path + "/" + possibleCommand); + const QFileInfo qmake(path + QLatin1Char('/') + possibleCommand); if (qmake.exists()) { if (!qtVersionForQMake(qmake.absoluteFilePath()).isNull()) { return qmake.absoluteFilePath(); @@ -62,12 +64,13 @@ bool DebuggingHelperLibrary::hasDebuggingHelperLibrary(const QString &qmakePath) QStringList DebuggingHelperLibrary::debuggingHelperLibraryDirectories(const QString &qtInstallData, const QString &qtpath) { - uint hash = qHash(qtpath); + const QChar slash = QLatin1Char('/'); + const uint hash = qHash(qtpath); QStringList directories; directories - << (qtInstallData + "/qtc-debugging-helper/") - << QDir::cleanPath((QApplication::applicationDirPath() + "/../qtc-debugging-helper/" + QString::number(hash))) + "/" - << (QDesktopServices::storageLocation(QDesktopServices::DataLocation) + "/qtc-debugging-helper/" + QString::number(hash)) + "/"; + << (qtInstallData + QLatin1String("/qtc-debugging-helper/")) + << QDir::cleanPath((QCoreApplication::applicationDirPath() + QLatin1String("/../qtc-debugging-helper/") + QString::number(hash))) + slash + << (QDesktopServices::storageLocation(QDesktopServices::DataLocation) + QLatin1String("/qtc-debugging-helper/") + QString::number(hash)) + slash; return directories; } @@ -84,7 +87,7 @@ QString DebuggingHelperLibrary::debuggingHelperLibrary(const QString &qmakePath) QString DebuggingHelperLibrary::qtInstallDataDir(const QString &qmakePath) { QProcess proc; - proc.start(qmakePath, QStringList() << "-query"<< "QT_INSTALL_DATA"); + proc.start(qmakePath, QStringList() << QLatin1String("-query") << QLatin1String("QT_INSTALL_DATA")); if (proc.waitForFinished()) return QString(proc.readAll().trimmed()); return QString::null; @@ -99,119 +102,142 @@ QString DebuggingHelperLibrary::qtDir(const QString &qmakePath) // Debugging Helper Library -QStringList DebuggingHelperLibrary::debuggingHelperLibraryLocations(const QString &qtInstallData, const QString &qtpath) +static inline QString helperFilePath(const QString &directory) { - QStringList result; - foreach(const QString &directory, debuggingHelperLibraryDirectories(qtInstallData, qtpath)) { #if defined(Q_OS_WIN) - QFileInfo fi(directory + "debug/gdbmacros.dll"); + return directory + QLatin1String("debug/gdbmacros.dll"); #elif defined(Q_OS_MAC) - QFileInfo fi(directory + "libgdbmacros.dylib"); + return directory + QLatin1String("libgdbmacros.dylib"); #else // generic UNIX - QFileInfo fi(directory + "libgdbmacros.so"); + return directory + QLatin1String("libgdbmacros.so"); #endif - result << fi.filePath(); - } +} + +QStringList DebuggingHelperLibrary::debuggingHelperLibraryLocations(const QString &qtInstallData, const QString &qtpath) +{ + QStringList result; + foreach(const QString &directory, debuggingHelperLibraryDirectories(qtInstallData, qtpath)) + result << QFileInfo(helperFilePath(directory)).filePath(); return result; } QString DebuggingHelperLibrary::debuggingHelperLibrary(const QString &qtInstallData, const QString &qtpath) { foreach(const QString &directory, debuggingHelperLibraryDirectories(qtInstallData, qtpath)) { -#if defined(Q_OS_WIN) - QFileInfo fi(directory + "debug/gdbmacros.dll"); -#elif defined(Q_OS_MAC) - QFileInfo fi(directory + "libgdbmacros.dylib"); -#else // generic UNIX - QFileInfo fi(directory + "libgdbmacros.so"); -#endif + const QFileInfo fi(helperFilePath(directory)); if (fi.exists()) return fi.filePath(); } return QString(); } - QString DebuggingHelperLibrary::buildDebuggingHelperLibrary(const QString &qmakePath, const QString &make, const Environment &env) { - QString directory = copyDebuggingHelperLibrary(qtInstallDataDir(qmakePath), qtDir(qmakePath)); + QString errorMessage; + const QString directory = copyDebuggingHelperLibrary(qtInstallDataDir(qmakePath), qtDir(qmakePath), &errorMessage); if (directory.isEmpty()) - return QString::null; + return errorMessage; return buildDebuggingHelperLibrary(directory, make, qmakePath, QString::null, env); } -QString DebuggingHelperLibrary::copyDebuggingHelperLibrary(const QString &qtInstallData, const QString &qtdir) +// Copy helper source files to a target directory, replacing older files. +static bool copyDebuggingHelperFiles(const QStringList &files, + const QString &targetDirectory, + QString *errorMessage) +{ + const QString dumperSourcePath = Core::ICore::instance()->resourcePath() + QLatin1String("/gdbmacros/"); + if (!QDir().mkpath(targetDirectory)) { + *errorMessage = QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "The target directory %1 could not be created.").arg(targetDirectory); + return false; + } + foreach (const QString &file, files) { + const QString source = dumperSourcePath + file; + const QString dest = targetDirectory + file; + const QFileInfo destInfo(dest); + if (destInfo.exists()) { + if (destInfo.lastModified() >= QFileInfo(source).lastModified()) + continue; + if (!QFile::remove(dest)) { + *errorMessage = QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "The existing file %1 could not be removed.").arg(destInfo.absoluteFilePath()); + return false; + } + } + if (!QFile::copy(source, dest)) { + *errorMessage = QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "The file %1 could not be copied to %2.").arg(source, dest); + return false; + } + } + return true; +} + +QString DebuggingHelperLibrary::copyDebuggingHelperLibrary(const QString &qtInstallData, + const QString &qtdir, + QString *errorMessage) { // Locations to try: // $QTDIR/qtc-debugging-helper // $APPLICATION-DIR/qtc-debugging-helper/$hash // $USERDIR/qtc-debugging-helper/$hash - QStringList directories = DebuggingHelperLibrary::debuggingHelperLibraryDirectories(qtInstallData, qtdir); + const QStringList directories = DebuggingHelperLibrary::debuggingHelperLibraryDirectories(qtInstallData, qtdir); QStringList files; - files << "gdbmacros.cpp" << "gdbmacros.pro" - << "LICENSE.LGPL" << "LGPL_EXCEPTION.TXT"; - foreach(const QString &directory, directories) { - QString dumperPath = Core::ICore::instance()->resourcePath() + "/gdbmacros/"; - bool success = true; - QDir().mkpath(directory); - foreach (const QString &file, files) { - QString source = dumperPath + file; - QString dest = directory + file; - QFileInfo destInfo(dest); - if (destInfo.exists()) { - if (destInfo.lastModified() >= QFileInfo(source).lastModified()) - continue; - success &= QFile::remove(dest); - } - success &= QFile::copy(source, dest); - } - if (success) + files << QLatin1String("gdbmacros.cpp") << QLatin1String("gdbmacros.pro") + << QLatin1String("LICENSE.LGPL") << QLatin1String("LGPL_EXCEPTION.TXT"); + // Try to find a writeable directory. + foreach(const QString &directory, directories) + if (copyDebuggingHelperFiles(files, directory, errorMessage)) { + errorMessage->clear(); return directory; - } - return QString::null; + } + *errorMessage = QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "The debugger helpers could not be built in any of the directories:\n- %1\n\nReason: %2") + .arg(directories.join(QLatin1String("\n- ")), *errorMessage); + return QString(); } QString DebuggingHelperLibrary::buildDebuggingHelperLibrary(const QString &directory, const QString &makeCommand, const QString &qmakeCommand, const QString &mkspec, const Environment &env) { QString output; + const QChar newline = QLatin1Char('\n'); // Setup process QProcess proc; proc.setEnvironment(env.toStringList()); proc.setWorkingDirectory(directory); proc.setProcessChannelMode(QProcess::MergedChannels); - output += QString("Building debugging helper library in %1\n").arg(directory); - output += "\n"; + output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "Building debugging helper library in %1\n").arg(directory); + output += newline; - QString makeFullPath = env.searchInPath(makeCommand); - if (QFileInfo(directory + "/Makefile").exists()) { + const QString makeFullPath = env.searchInPath(makeCommand); + if (QFileInfo(directory + QLatin1String("/Makefile")).exists()) { if (!makeFullPath.isEmpty()) { - output += QString("Running %1 clean...\n").arg(makeFullPath); - proc.start(makeFullPath, QStringList() << "clean"); + const QString cleanTarget = QLatin1String("distclean"); + output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "Running %1 %2...\n").arg(makeFullPath, cleanTarget); + proc.start(makeFullPath, QStringList(cleanTarget)); proc.waitForFinished(); - output += proc.readAll(); + output += QString::fromLocal8Bit(proc.readAll()); } else { - output += QString("%1 not found in PATH\n").arg(makeCommand); + output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "%1 not found in PATH\n").arg(makeCommand); return output; } } + output += newline; + output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "Running %1 ...\n").arg(qmakeCommand); - output += QString("\nRunning %1 ...\n").arg(qmakeCommand); - - proc.start(qmakeCommand, QStringList()<<"-spec"<< (mkspec.isEmpty() ? "default" : mkspec) <<"gdbmacros.pro"); + QStringList makeArgs; + makeArgs << QLatin1String("-spec")<< (mkspec.isEmpty() ? QString(QLatin1String("default")) : mkspec) << QLatin1String("gdbmacros.pro"); + proc.start(qmakeCommand, makeArgs); proc.waitForFinished(); output += proc.readAll(); - output += "\n"; + output += newline;; if (!makeFullPath.isEmpty()) { - output += QString("Running %1 ...\n").arg(makeFullPath); + output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "Running %1 ...\n").arg(makeFullPath); proc.start(makeFullPath, QStringList()); proc.waitForFinished(); output += proc.readAll(); } else { - output += QString("%1 not found in PATH\n").arg(makeCommand); + output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "%1 not found in PATH\n").arg(makeCommand); } return output; } @@ -219,14 +245,14 @@ QString DebuggingHelperLibrary::buildDebuggingHelperLibrary(const QString &direc QString DebuggingHelperLibrary::qtVersionForQMake(const QString &qmakePath) { QProcess qmake; - qmake.start(qmakePath, QStringList()<<"--version"); + qmake.start(qmakePath, QStringList(QLatin1String("--version"))); if (!qmake.waitForFinished()) return false; QString output = qmake.readAllStandardOutput(); - QRegExp regexp("(QMake version|QMake version:)[\\s]*([\\d.]*)", Qt::CaseInsensitive); + QRegExp regexp(QLatin1String("(QMake version|QMake version:)[\\s]*([\\d.]*)"), Qt::CaseInsensitive); regexp.indexIn(output); - if (regexp.cap(2).startsWith("2.")) { - QRegExp regexp2("Using Qt version[\\s]*([\\d\\.]*)", Qt::CaseInsensitive); + if (regexp.cap(2).startsWith(QLatin1String("2."))) { + QRegExp regexp2(QLatin1String("Using Qt version[\\s]*([\\d\\.]*)"), Qt::CaseInsensitive); regexp2.indexIn(output); return regexp2.cap(1); } @@ -237,11 +263,11 @@ QStringList DebuggingHelperLibrary::possibleQMakeCommands() { // On windows noone has renamed qmake, right? #ifdef Q_OS_WIN - return QStringList() << "qmake.exe"; + return QStringList(QLatin1String("qmake.exe")); #else // On unix some distributions renamed qmake to avoid clashes QStringList result; - result << "qmake-qt4" << "qmake4" << "qmake"; + result << QLatin1String("qmake-qt4") << QLatin1String("qmake4") << QLatin1String("qmake"); return result; #endif } diff --git a/src/plugins/projectexplorer/debugginghelper.h b/src/plugins/projectexplorer/debugginghelper.h index a98ab2a6166d4baaa6bc4b3eb2c0380e93181e6d..47727d1c466fe70bf1a787108224374fae5cd0f3 100644 --- a/src/plugins/projectexplorer/debugginghelper.h +++ b/src/plugins/projectexplorer/debugginghelper.h @@ -56,10 +56,12 @@ public: static QString buildDebuggingHelperLibrary(const QString &qmakePath, const QString &make, const Environment &env); static QString buildDebuggingHelperLibrary(const QString &directory, const QString &makeCommand, const QString &qmakeCommand, const QString &mkspec, const Environment &env); + // Build the helpers and return the output log/errormessage. static QStringList debuggingHelperLibraryLocations(const QString &qmakePath); static QStringList debuggingHelperLibraryLocations(const QString &qtInstallData, const QString &qtpath); - static QString copyDebuggingHelperLibrary(const QString &qtInstallData, const QString &qtdir); + // Copy the source files to a target location and return the chosen target location. + static QString copyDebuggingHelperLibrary(const QString &qtInstallData, const QString &qtdir, QString *errorMessage); private: static QStringList debuggingHelperLibraryDirectories(const QString &qtInstallData, const QString &qtpath); diff --git a/src/plugins/qt4projectmanager/qtoptionspage.cpp b/src/plugins/qt4projectmanager/qtoptionspage.cpp index 6c5c4c4a9ccc7021ffbbc089e3a5c4f3e990eece..2f3721c324a403b457042b13561d6b3f97db3556 100644 --- a/src/plugins/qt4projectmanager/qtoptionspage.cpp +++ b/src/plugins/qt4projectmanager/qtoptionspage.cpp @@ -3,14 +3,43 @@ #include "ui_qtversionmanager.h" #include "qt4projectmanagerconstants.h" #include "qtversionmanager.h" + #include <coreplugin/coreconstants.h> +#include <coreplugin/icore.h> +#include <coreplugin/progressmanager/progressmanager.h> #include <utils/treewidgetcolumnstretcher.h> +#include <utils/qtcassert.h> +#include <QtCore/QFuture> +#include <QtCore/QtConcurrentRun> #include <QtCore/QDebug> #include <QtCore/QDir> +#include <QtCore/QDateTime> using namespace Qt4ProjectManager; using namespace Qt4ProjectManager::Internal; + +/// +// DebuggingHelperBuildTask +/// + +DebuggingHelperBuildTask::DebuggingHelperBuildTask(const QtVersion &version) : + m_version(new QtVersion(version)) +{ +} + +DebuggingHelperBuildTask::~DebuggingHelperBuildTask() +{ + delete m_version; +} + +void DebuggingHelperBuildTask::run() +{ + const QString output = m_version->buildDebuggingHelperLibrary(); + emit finished(m_version->name(), output); + deleteLater(); +} + /// // QtOptionsPage /// @@ -65,6 +94,11 @@ void QtOptionsPage::apply() QtOptionsPageWidget::QtOptionsPageWidget(QWidget *parent, QList<QtVersion *> versions, QtVersion *defaultVersion) : QWidget(parent) + , m_debuggingHelperOkPixmap(QLatin1String(":/extensionsystem/images/ok.png")) + , m_debuggingHelperErrorPixmap(QLatin1String(":/extensionsystem/images/error.png")) + , m_debuggingHelperOkIcon(m_debuggingHelperOkPixmap) + , m_debuggingHelperErrorIcon(m_debuggingHelperErrorPixmap) + , m_defaultVersion(versions.indexOf(defaultVersion)) , m_specifyNameString(tr("<specify a name>")) , m_specifyPathString(tr("<specify a path>")) @@ -106,10 +140,7 @@ QtOptionsPageWidget::QtOptionsPageWidget(QWidget *parent, QList<QtVersion *> ver item->setData(0, Qt::UserRole, version->uniqueId()); if (version->isValid()) { - if (version->hasDebuggingHelper()) - item->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/ok.png")); - else - item->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/error.png")); + item->setData(2, Qt::DecorationRole, version->hasDebuggingHelper() ? m_debuggingHelperOkIcon : m_debuggingHelperErrorIcon); } else { item->setData(2, Qt::DecorationRole, QIcon()); } @@ -160,27 +191,62 @@ QtOptionsPageWidget::QtOptionsPageWidget(QWidget *parent, QList<QtVersion *> ver updateState(); } -void QtOptionsPageWidget::buildDebuggingHelper() +QtVersion *QtOptionsPageWidget::currentVersion() const { - // Find the qt version for this button.. - QTreeWidgetItem *currentItem = m_ui->qtdirList->currentItem(); - int currentItemIndex = indexForTreeItem(currentItem); - if (currentItemIndex < 0) - return; + if (QTreeWidgetItem *currentItem = m_ui->qtdirList->currentItem()) { + const int currentItemIndex = indexForTreeItem(currentItem); + if (currentItemIndex >= 0 && currentItemIndex < m_versions.size()) + return m_versions.at(currentItemIndex); + } + return 0; +} - QtVersion *version = m_versions[currentItemIndex]; +static inline int findVersionByName(const QList<QtVersion *> &l, const QString &name) +{ + const int size = l.size(); + for (int i = 0; i < size; i++) + if (l.at(i)->name() == name) + return i; + return -1; +} - QString result = m_versions.at(currentItemIndex)->buildDebuggingHelperLibrary(); - currentItem->setData(2, Qt::UserRole, result); +// Update with results of terminated helper build +void QtOptionsPageWidget::debuggingHelperBuildFinished(const QString &name, const QString &output) +{ + const int index = findVersionByName(m_versions, name); + if (index == -1) + return; // Oops, somebody managed to delete the version + // Update item view + QtVersion *version = m_versions.at(index); + QTreeWidgetItem *item = treeItemForIndex(index); + QTC_ASSERT(item, return) + item->setData(2, Qt::UserRole, output); + const bool success = version->hasDebuggingHelper(); + item->setData(2, Qt::DecorationRole, success ? m_debuggingHelperOkIcon : m_debuggingHelperErrorIcon); + + // Update bottom control if the selection is still the same + if (version == currentVersion()) { + m_ui->showLogButton->setEnabled(true); + updateDebuggingHelperStateLabel(version); + if (!success) + showDebuggingBuildLog(); + } +} - if (version->hasDebuggingHelper()) { - m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/ok.png")); - currentItem->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/ok.png")); - } else { - m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/error.png")); - currentItem->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/error.png")); +void QtOptionsPageWidget::buildDebuggingHelper() +{ + if (QtVersion *version = currentVersion()) { + m_ui->showLogButton->setEnabled(false); + // Run a debugging helper build task in the background. + DebuggingHelperBuildTask *buildTask = new DebuggingHelperBuildTask(*version); + connect(buildTask, SIGNAL(finished(QString,QString)), this, SLOT(debuggingHelperBuildFinished(QString,QString)), + Qt::QueuedConnection); + QFuture<void> task = QtConcurrent::run(buildTask, &DebuggingHelperBuildTask::run); + const QString taskName = tr("Building helpers"); + Core::ICore::instance()->progressManager()->addTask(task, taskName, + QLatin1String("Qt4ProjectManager::BuildHelpers"), + Core::ProgressManager::CloseOnSuccess); } - m_ui->showLogButton->setEnabled(true); } void QtOptionsPageWidget::showDebuggingBuildLog() @@ -190,10 +256,13 @@ void QtOptionsPageWidget::showDebuggingBuildLog() int currentItemIndex = indexForTreeItem(currentItem); if (currentItemIndex < 0) return; + // Show text and scroll to bottom QDialog dlg; Ui_ShowBuildLog ui; ui.setupUi(&dlg); ui.log->setPlainText(currentItem->data(2, Qt::UserRole).toString()); + ui.log->moveCursor(QTextCursor::End); + ui.log->ensureCursorVisible(); dlg.exec(); } @@ -243,12 +312,41 @@ void QtOptionsPageWidget::removeQtDir() updateState(); } +// Format html table tooltip about helpers +static inline QString msgHtmlHelperToolTip(const QFileInfo &fi) +{ + return QtOptionsPageWidget::tr("<html><body><table><tr><td>File:</td><td><pre>%1</pre></td></tr>" + "<tr><td>Last modified:</td><td>%2</td></tr>" + "<tr><td>Size:</td><td>%3 Bytes</td></tr></table></body></html>"). + arg(fi.absoluteFilePath()). + arg(fi.lastModified().toString(Qt::SystemLocaleLongDate)). + arg(fi.size()); +} + +// Update the state label with a pixmap and set a tooltip describing +// the file on neighbouring controls. +void QtOptionsPageWidget::updateDebuggingHelperStateLabel(const QtVersion *version) +{ + QString tooltip; + if (version && version->isValid()) { + const bool hasHelper = version->hasDebuggingHelper(); + m_ui->debuggingHelperStateLabel->setPixmap(hasHelper ? m_debuggingHelperOkPixmap : m_debuggingHelperErrorPixmap); + if (hasHelper) + tooltip = msgHtmlHelperToolTip(QFileInfo(version->debuggingHelperLibrary())); + } else { + m_ui->debuggingHelperStateLabel->setPixmap(QPixmap()); + } + m_ui->debuggingHelperStateLabel->setToolTip(tooltip); + m_ui->debuggingHelperLabel->setToolTip(tooltip); + m_ui->showLogButton->setToolTip(tooltip); + m_ui->rebuildButton->setToolTip(tooltip); +} + void QtOptionsPageWidget::updateState() { - int currentIndex = indexForTreeItem(m_ui->qtdirList->currentItem()); - bool enabled = (currentIndex >= 0); - bool isAutodetected = (enabled - && m_versions.at(currentIndex)->isAutodetected()); + const QtVersion *version = currentVersion(); + const bool enabled = version != 0; + const bool isAutodetected = enabled && version->isAutodetected(); m_ui->delButton->setEnabled(enabled && !isAutodetected); m_ui->nameEdit->setEnabled(enabled && !isAutodetected); m_ui->qtPath->setEnabled(enabled && !isAutodetected); @@ -257,15 +355,9 @@ void QtOptionsPageWidget::updateState() bool hasLog = enabled && !m_ui->qtdirList->currentItem()->data(2, Qt::UserRole).toString().isEmpty(); m_ui->showLogButton->setEnabled(hasLog); - QtVersion *version = 0; - if (enabled) - version = m_versions.at(currentIndex); if (version) { m_ui->rebuildButton->setEnabled(version->isValid()); - if (version->hasDebuggingHelper()) - m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/ok.png")); - else - m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/error.png")); + updateDebuggingHelperStateLabel(version); } else { m_ui->rebuildButton->setEnabled(false); m_ui->debuggingHelperStateLabel->setPixmap(QPixmap()); @@ -358,11 +450,11 @@ void QtOptionsPageWidget::showEnvironmentPage(QTreeWidgetItem *item) } } -int QtOptionsPageWidget::indexForTreeItem(QTreeWidgetItem *item) const +int QtOptionsPageWidget::indexForTreeItem(const QTreeWidgetItem *item) const { if (!item || !item->parent()) return -1; - int uniqueId = item->data(0, Qt::UserRole).toInt(); + const int uniqueId = item->data(0, Qt::UserRole).toInt(); for (int index = 0; index < m_versions.size(); ++index) { if (m_versions.at(index)->uniqueId() == uniqueId) return index; @@ -507,16 +599,10 @@ void QtOptionsPageWidget::updateCurrentQtPath() showEnvironmentPage(currentItem); - if (m_versions[currentItemIndex]->isValid()) { - bool hasLog = !currentItem->data(2, Qt::UserRole).toString().isEmpty(); - bool hasHelper = m_versions[currentItemIndex]->hasDebuggingHelper(); - if (hasHelper) { - currentItem->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/ok.png")); - m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/ok.png")); - } else { - currentItem->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/error.png")); - m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/error.png")); - } + const QtVersion *version = m_versions.at(currentItemIndex); + if (version->isValid()) { + const bool hasLog = !currentItem->data(2, Qt::UserRole).toString().isEmpty(); + currentItem->setData(2, Qt::DecorationRole, version->hasDebuggingHelper() ? m_debuggingHelperOkIcon : m_debuggingHelperErrorIcon); m_ui->showLogButton->setEnabled(hasLog); m_ui->rebuildButton->setEnabled(true); } else { @@ -524,6 +610,7 @@ void QtOptionsPageWidget::updateCurrentQtPath() m_ui->debuggingHelperStateLabel->setPixmap(QPixmap()); m_ui->rebuildButton->setEnabled(true); } + updateDebuggingHelperStateLabel(version); } void QtOptionsPageWidget::updateCurrentMingwDirectory() diff --git a/src/plugins/qt4projectmanager/qtoptionspage.h b/src/plugins/qt4projectmanager/qtoptionspage.h index 57275795fe6ac1e9f7fe47cc4b765869d06ba592..dc2bb3a9465b122501df927cfe2af8d512a81347 100644 --- a/src/plugins/qt4projectmanager/qtoptionspage.h +++ b/src/plugins/qt4projectmanager/qtoptionspage.h @@ -32,6 +32,8 @@ #include <coreplugin/dialogs/ioptionspage.h> #include <QtGui/QWidget> +#include <QtGui/QPixmap> +#include <QtGui/QIcon> QT_BEGIN_NAMESPACE class QTreeWidgetItem; @@ -46,6 +48,24 @@ namespace Ui { class QtVersionManager; } +// A task suitable to be run by QtConcurrent to build the helpers. +// It may outlive the settings page if someone quickly cancels it, +// so, it maintains a copy of the QtVersion and emits finished() by name. +class DebuggingHelperBuildTask : public QObject { + Q_OBJECT +public: + explicit DebuggingHelperBuildTask(const QtVersion &version); + virtual ~DebuggingHelperBuildTask(); + + void run(); + +signals: + void finished(const QString &versionName, const QString &output); + +private: + QtVersion *m_version; +}; + class QtOptionsPageWidget : public QWidget { Q_OBJECT @@ -57,11 +77,17 @@ public: void finish(); private: + const QPixmap m_debuggingHelperOkPixmap; + const QPixmap m_debuggingHelperErrorPixmap; + const QIcon m_debuggingHelperOkIcon; + const QIcon m_debuggingHelperErrorIcon; + void showEnvironmentPage(QTreeWidgetItem * item); void fixQtVersionName(int index); - int indexForWidget(QWidget *debuggingHelperWidget) const; - int indexForTreeItem(QTreeWidgetItem *item) const; + int indexForTreeItem(const QTreeWidgetItem *item) const; QTreeWidgetItem *treeItemForIndex(int index) const; + QtVersion *currentVersion() const; + void updateDebuggingHelperStateLabel(const QtVersion *version = 0); Internal::Ui::QtVersionManager *m_ui; QList<QtVersion *> m_versions; @@ -89,6 +115,7 @@ private slots: void msvcVersionChanged(); void buildDebuggingHelper(); void showDebuggingBuildLog(); + void debuggingHelperBuildFinished(const QString &name, const QString &output); }; class QtOptionsPage : public Core::IOptionsPage @@ -105,7 +132,7 @@ public: void apply(); void finish() { } private: - QtOptionsPageWidget *m_widget; + QtOptionsPageWidget *m_widget; }; } //namespace Internal diff --git a/src/plugins/qt4projectmanager/qtversionmanager.cpp b/src/plugins/qt4projectmanager/qtversionmanager.cpp index a0680c4458910645473996fad5433ab65c3da73a..c5ce9097337631af530d2e1589b3d9548ffed8eb 100644 --- a/src/plugins/qt4projectmanager/qtversionmanager.cpp +++ b/src/plugins/qt4projectmanager/qtversionmanager.cpp @@ -1266,8 +1266,10 @@ QString QtVersion::buildDebuggingHelperLibrary() // TODO: the debugging helper doesn't comply to actual tool chain yet ProjectExplorer::ToolChain *tc = createToolChain(defaultToolchainType()); tc->addToEnvironment(env); - QString directory = DebuggingHelperLibrary::copyDebuggingHelperLibrary(qtInstallData, path()); - QString output = DebuggingHelperLibrary::buildDebuggingHelperLibrary(directory, tc->makeCommand(), qmakeCommand(), mkspec(), env); + QString output; + QString directory = DebuggingHelperLibrary::copyDebuggingHelperLibrary(qtInstallData, path(), &output); + if (!directory.isEmpty()) + output += DebuggingHelperLibrary::buildDebuggingHelperLibrary(directory, tc->makeCommand(), qmakeCommand(), mkspec(), env); m_hasDebuggingHelper = !debuggingHelperLibrary().isEmpty(); delete tc; return output; diff --git a/src/plugins/qt4projectmanager/qtversionmanager.ui b/src/plugins/qt4projectmanager/qtversionmanager.ui index b92932ece627c75d7c02d5733934a0084a0a22ce..bc716ab3cd8386dd18352fa58181a7cfe6e08697 100644 --- a/src/plugins/qt4projectmanager/qtversionmanager.ui +++ b/src/plugins/qt4projectmanager/qtversionmanager.ui @@ -17,31 +17,6 @@ <string>Qt versions</string> </property> <layout class="QGridLayout" name="gridLayout"> - <item row="0" column="0" colspan="2"> - <widget class="QTreeWidget" name="qtdirList"> - <property name="uniformRowHeights"> - <bool>true</bool> - </property> - <property name="columnCount"> - <number>3</number> - </property> - <column> - <property name="text"> - <string>Name</string> - </property> - </column> - <column> - <property name="text"> - <string>Path</string> - </property> - </column> - <column> - <property name="text"> - <string>Debugging Helper</string> - </property> - </column> - </widget> - </item> <item row="0" column="2"> <layout class="QVBoxLayout"> <property name="spacing"> @@ -91,43 +66,6 @@ </item> </layout> </item> - <item row="1" column="0"> - <widget class="QLabel" name="versionNameLabel"> - <property name="text"> - <string>Version Name:</string> - </property> - </widget> - </item> - <item row="1" column="1"> - <widget class="QLineEdit" name="nameEdit"/> - </item> - <item row="2" column="0"> - <widget class="QLabel" name="pathLabel"> - <property name="text"> - <string>Path:</string> - </property> - </widget> - </item> - <item row="2" column="1"> - <widget class="Core::Utils::PathChooser" name="qtPath" native="true"/> - </item> - <item row="3" column="0"> - <widget class="QLabel" name="mingwLabel"> - <property name="text"> - <string>MinGw Directory:</string> - </property> - </widget> - </item> - <item row="3" column="1"> - <widget class="Core::Utils::PathChooser" name="mingwPath" native="true"/> - </item> - <item row="4" column="0"> - <widget class="QLabel" name="msvcLabel"> - <property name="text"> - <string>MSVC Version:</string> - </property> - </widget> - </item> <item row="4" column="1"> <layout class="QHBoxLayout" name="horizontalLayout"> <property name="spacing"> @@ -162,23 +100,6 @@ p, li { white-space: pre-wrap; } </item> </layout> </item> - <item row="5" column="0"> - <widget class="QLabel" name="mwcLabel"> - <property name="text"> - <string>MWC Directory:</string> - </property> - </widget> - </item> - <item row="5" column="1"> - <widget class="Core::Utils::PathChooser" name="mwcPath" native="true"/> - </item> - <item row="6" column="0"> - <widget class="QLabel" name="label"> - <property name="text"> - <string>Debugging Helper:</string> - </property> - </widget> - </item> <item row="6" column="1"> <layout class="QHBoxLayout" name="horizontalLayout_2"> <item> @@ -210,6 +131,75 @@ p, li { white-space: pre-wrap; } </item> </layout> </item> + <item row="0" column="0" colspan="2"> + <widget class="QTreeWidget" name="qtdirList"> + <property name="uniformRowHeights"> + <bool>true</bool> + </property> + <property name="columnCount"> + <number>3</number> + </property> + <column> + <property name="text"> + <string>Name</string> + </property> + </column> + <column> + <property name="text"> + <string>Path</string> + </property> + </column> + <column> + <property name="text"> + <string>Debugging Helper</string> + </property> + </column> + </widget> + </item> + <item row="1" column="0"> + <widget class="QLabel" name="versionNameLabel"> + <property name="text"> + <string>Version Name:</string> + </property> + </widget> + </item> + <item row="1" column="1"> + <widget class="QLineEdit" name="nameEdit"/> + </item> + <item row="2" column="0"> + <widget class="QLabel" name="pathLabel"> + <property name="text"> + <string>Path:</string> + </property> + </widget> + </item> + <item row="2" column="1"> + <widget class="Core::Utils::PathChooser" name="qtPath"/> + </item> + <item row="3" column="0"> + <widget class="QLabel" name="mingwLabel"> + <property name="text"> + <string>MinGw Directory:</string> + </property> + </widget> + </item> + <item row="3" column="1"> + <widget class="Core::Utils::PathChooser" name="mingwPath"/> + </item> + <item row="4" column="0"> + <widget class="QLabel" name="msvcLabel"> + <property name="text"> + <string>MSVC Version:</string> + </property> + </widget> + </item> + <item row="6" column="0"> + <widget class="QLabel" name="debuggingHelperLabel"> + <property name="text"> + <string>Debugging Helper:</string> + </property> + </widget> + </item> <item row="7" column="1"> <widget class="QLabel" name="errorLabel"> <property name="text"> @@ -217,6 +207,16 @@ p, li { white-space: pre-wrap; } </property> </widget> </item> + <item row="5" column="0"> + <widget class="QLabel" name="mwcLabel"> + <property name="text"> + <string>MWC Directory:</string> + </property> + </widget> + </item> + <item row="5" column="1"> + <widget class="Core::Utils::PathChooser" name="mwcPath"/> + </item> </layout> <zorder>qtdirList</zorder> <zorder>versionNameLabel</zorder> @@ -226,12 +226,10 @@ p, li { white-space: pre-wrap; } <zorder>mingwLabel</zorder> <zorder>mingwPath</zorder> <zorder>msvcLabel</zorder> - <zorder>label</zorder> + <zorder>debuggingHelperLabel</zorder> <zorder>errorLabel</zorder> <zorder>mwcLabel</zorder> <zorder>mwcPath</zorder> - <zorder>msvcComboBox</zorder> - <zorder>msvcNotFoundLabel</zorder> </widget> </item> <item>