Debugger: Improve the helper build mechanism on the settings page.

- Give the controls a tooltip listing file details (date)
- Make building a QtConcurrent task
- Make log window scroll to bottom and pop up on error
- Make the build code pass on error messages about copying the
  source files to the log file
- Clean up the building code string-wise, use QLatin1String and
  translate messages, cache the icons
This commit is contained in:
Friedemann Kleint
2009-08-07 14:10:36 +02:00
parent 50fb8bfb03
commit debb3961c2
6 changed files with 346 additions and 204 deletions

View File

@@ -28,13 +28,15 @@
**************************************************************************/ **************************************************************************/
#include "debugginghelper.h" #include "debugginghelper.h"
#include <coreplugin/icore.h> #include <coreplugin/icore.h>
#include <QtCore/QFileInfo> #include <QtCore/QFileInfo>
#include <QtCore/QCoreApplication>
#include <QtCore/QHash> #include <QtCore/QHash>
#include <QtCore/QProcess> #include <QtCore/QProcess>
#include <QtCore/QDir> #include <QtCore/QDir>
#include <QtCore/QDateTime> #include <QtCore/QDateTime>
#include <QtGui/QApplication>
#include <QtGui/QDesktopServices> #include <QtGui/QDesktopServices>
using namespace ProjectExplorer; using namespace ProjectExplorer;
@@ -44,7 +46,7 @@ QString DebuggingHelperLibrary::findSystemQt(const Environment &env)
QStringList paths = env.path(); QStringList paths = env.path();
foreach (const QString &path, paths) { foreach (const QString &path, paths) {
foreach (const QString &possibleCommand, possibleQMakeCommands()) { foreach (const QString &possibleCommand, possibleQMakeCommands()) {
QFileInfo qmake(path + "/" + possibleCommand); const QFileInfo qmake(path + QLatin1Char('/') + possibleCommand);
if (qmake.exists()) { if (qmake.exists()) {
if (!qtVersionForQMake(qmake.absoluteFilePath()).isNull()) { if (!qtVersionForQMake(qmake.absoluteFilePath()).isNull()) {
return qmake.absoluteFilePath(); return qmake.absoluteFilePath();
@@ -62,12 +64,13 @@ bool DebuggingHelperLibrary::hasDebuggingHelperLibrary(const QString &qmakePath)
QStringList DebuggingHelperLibrary::debuggingHelperLibraryDirectories(const QString &qtInstallData, const QString &qtpath) QStringList DebuggingHelperLibrary::debuggingHelperLibraryDirectories(const QString &qtInstallData, const QString &qtpath)
{ {
uint hash = qHash(qtpath); const QChar slash = QLatin1Char('/');
const uint hash = qHash(qtpath);
QStringList directories; QStringList directories;
directories directories
<< (qtInstallData + "/qtc-debugging-helper/") << (qtInstallData + QLatin1String("/qtc-debugging-helper/"))
<< QDir::cleanPath((QApplication::applicationDirPath() + "/../qtc-debugging-helper/" + QString::number(hash))) + "/" << QDir::cleanPath((QCoreApplication::applicationDirPath() + QLatin1String("/../qtc-debugging-helper/") + QString::number(hash))) + slash
<< (QDesktopServices::storageLocation(QDesktopServices::DataLocation) + "/qtc-debugging-helper/" + QString::number(hash)) + "/"; << (QDesktopServices::storageLocation(QDesktopServices::DataLocation) + QLatin1String("/qtc-debugging-helper/") + QString::number(hash)) + slash;
return directories; return directories;
} }
@@ -84,7 +87,7 @@ QString DebuggingHelperLibrary::debuggingHelperLibrary(const QString &qmakePath)
QString DebuggingHelperLibrary::qtInstallDataDir(const QString &qmakePath) QString DebuggingHelperLibrary::qtInstallDataDir(const QString &qmakePath)
{ {
QProcess proc; QProcess proc;
proc.start(qmakePath, QStringList() << "-query"<< "QT_INSTALL_DATA"); proc.start(qmakePath, QStringList() << QLatin1String("-query") << QLatin1String("QT_INSTALL_DATA"));
if (proc.waitForFinished()) if (proc.waitForFinished())
return QString(proc.readAll().trimmed()); return QString(proc.readAll().trimmed());
return QString::null; return QString::null;
@@ -99,119 +102,142 @@ QString DebuggingHelperLibrary::qtDir(const QString &qmakePath)
// Debugging Helper Library // Debugging Helper Library
static inline QString helperFilePath(const QString &directory)
{
#if defined(Q_OS_WIN)
return directory + QLatin1String("debug/gdbmacros.dll");
#elif defined(Q_OS_MAC)
return directory + QLatin1String("libgdbmacros.dylib");
#else // generic UNIX
return directory + QLatin1String("libgdbmacros.so");
#endif
}
QStringList DebuggingHelperLibrary::debuggingHelperLibraryLocations(const QString &qtInstallData, const QString &qtpath) QStringList DebuggingHelperLibrary::debuggingHelperLibraryLocations(const QString &qtInstallData, const QString &qtpath)
{ {
QStringList result; QStringList result;
foreach(const QString &directory, debuggingHelperLibraryDirectories(qtInstallData, qtpath)) { foreach(const QString &directory, debuggingHelperLibraryDirectories(qtInstallData, qtpath))
#if defined(Q_OS_WIN) result << QFileInfo(helperFilePath(directory)).filePath();
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
result << fi.filePath();
}
return result; return result;
} }
QString DebuggingHelperLibrary::debuggingHelperLibrary(const QString &qtInstallData, const QString &qtpath) QString DebuggingHelperLibrary::debuggingHelperLibrary(const QString &qtInstallData, const QString &qtpath)
{ {
foreach(const QString &directory, debuggingHelperLibraryDirectories(qtInstallData, qtpath)) { foreach(const QString &directory, debuggingHelperLibraryDirectories(qtInstallData, qtpath)) {
#if defined(Q_OS_WIN) const QFileInfo fi(helperFilePath(directory));
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
if (fi.exists()) if (fi.exists())
return fi.filePath(); return fi.filePath();
} }
return QString(); return QString();
} }
QString DebuggingHelperLibrary::buildDebuggingHelperLibrary(const QString &qmakePath, const QString &make, const Environment &env) 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()) if (directory.isEmpty())
return QString::null; return errorMessage;
return buildDebuggingHelperLibrary(directory, make, qmakePath, QString::null, env); 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: // Locations to try:
// $QTDIR/qtc-debugging-helper // $QTDIR/qtc-debugging-helper
// $APPLICATION-DIR/qtc-debugging-helper/$hash // $APPLICATION-DIR/qtc-debugging-helper/$hash
// $USERDIR/qtc-debugging-helper/$hash // $USERDIR/qtc-debugging-helper/$hash
QStringList directories = DebuggingHelperLibrary::debuggingHelperLibraryDirectories(qtInstallData, qtdir); const QStringList directories = DebuggingHelperLibrary::debuggingHelperLibraryDirectories(qtInstallData, qtdir);
QStringList files; QStringList files;
files << "gdbmacros.cpp" << "gdbmacros.pro" files << QLatin1String("gdbmacros.cpp") << QLatin1String("gdbmacros.pro")
<< "LICENSE.LGPL" << "LGPL_EXCEPTION.TXT"; << QLatin1String("LICENSE.LGPL") << QLatin1String("LGPL_EXCEPTION.TXT");
foreach(const QString &directory, directories) { // Try to find a writeable directory.
QString dumperPath = Core::ICore::instance()->resourcePath() + "/gdbmacros/"; foreach(const QString &directory, directories)
bool success = true; if (copyDebuggingHelperFiles(files, directory, errorMessage)) {
QDir().mkpath(directory); errorMessage->clear();
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)
return directory; 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 DebuggingHelperLibrary::buildDebuggingHelperLibrary(const QString &directory, const QString &makeCommand, const QString &qmakeCommand, const QString &mkspec, const Environment &env)
{ {
QString output; QString output;
const QChar newline = QLatin1Char('\n');
// Setup process // Setup process
QProcess proc; QProcess proc;
proc.setEnvironment(env.toStringList()); proc.setEnvironment(env.toStringList());
proc.setWorkingDirectory(directory); proc.setWorkingDirectory(directory);
proc.setProcessChannelMode(QProcess::MergedChannels); proc.setProcessChannelMode(QProcess::MergedChannels);
output += QString("Building debugging helper library in %1\n").arg(directory); output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "Building debugging helper library in %1\n").arg(directory);
output += "\n"; output += newline;
QString makeFullPath = env.searchInPath(makeCommand); const QString makeFullPath = env.searchInPath(makeCommand);
if (QFileInfo(directory + "/Makefile").exists()) { if (QFileInfo(directory + QLatin1String("/Makefile")).exists()) {
if (!makeFullPath.isEmpty()) { if (!makeFullPath.isEmpty()) {
output += QString("Running %1 clean...\n").arg(makeFullPath); const QString cleanTarget = QLatin1String("distclean");
proc.start(makeFullPath, QStringList() << "clean"); output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "Running %1 %2...\n").arg(makeFullPath, cleanTarget);
proc.start(makeFullPath, QStringList(cleanTarget));
proc.waitForFinished(); proc.waitForFinished();
output += proc.readAll(); output += QString::fromLocal8Bit(proc.readAll());
} else { } 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; return output;
} }
} }
output += newline;
output += QCoreApplication::translate("ProjectExplorer::DebuggingHelperLibrary", "Running %1 ...\n").arg(qmakeCommand);
output += QString("\nRunning %1 ...\n").arg(qmakeCommand); QStringList makeArgs;
makeArgs << QLatin1String("-spec")<< (mkspec.isEmpty() ? QString(QLatin1String("default")) : mkspec) << QLatin1String("gdbmacros.pro");
proc.start(qmakeCommand, QStringList()<<"-spec"<< (mkspec.isEmpty() ? "default" : mkspec) <<"gdbmacros.pro"); proc.start(qmakeCommand, makeArgs);
proc.waitForFinished(); proc.waitForFinished();
output += proc.readAll(); output += proc.readAll();
output += "\n"; output += newline;;
if (!makeFullPath.isEmpty()) { 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.start(makeFullPath, QStringList());
proc.waitForFinished(); proc.waitForFinished();
output += proc.readAll(); output += proc.readAll();
} else { } 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; return output;
} }
@@ -219,14 +245,14 @@ QString DebuggingHelperLibrary::buildDebuggingHelperLibrary(const QString &direc
QString DebuggingHelperLibrary::qtVersionForQMake(const QString &qmakePath) QString DebuggingHelperLibrary::qtVersionForQMake(const QString &qmakePath)
{ {
QProcess qmake; QProcess qmake;
qmake.start(qmakePath, QStringList()<<"--version"); qmake.start(qmakePath, QStringList(QLatin1String("--version")));
if (!qmake.waitForFinished()) if (!qmake.waitForFinished())
return false; return false;
QString output = qmake.readAllStandardOutput(); 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); regexp.indexIn(output);
if (regexp.cap(2).startsWith("2.")) { if (regexp.cap(2).startsWith(QLatin1String("2."))) {
QRegExp regexp2("Using Qt version[\\s]*([\\d\\.]*)", Qt::CaseInsensitive); QRegExp regexp2(QLatin1String("Using Qt version[\\s]*([\\d\\.]*)"), Qt::CaseInsensitive);
regexp2.indexIn(output); regexp2.indexIn(output);
return regexp2.cap(1); return regexp2.cap(1);
} }
@@ -237,11 +263,11 @@ QStringList DebuggingHelperLibrary::possibleQMakeCommands()
{ {
// On windows noone has renamed qmake, right? // On windows noone has renamed qmake, right?
#ifdef Q_OS_WIN #ifdef Q_OS_WIN
return QStringList() << "qmake.exe"; return QStringList(QLatin1String("qmake.exe"));
#else #else
// On unix some distributions renamed qmake to avoid clashes // On unix some distributions renamed qmake to avoid clashes
QStringList result; QStringList result;
result << "qmake-qt4" << "qmake4" << "qmake"; result << QLatin1String("qmake-qt4") << QLatin1String("qmake4") << QLatin1String("qmake");
return result; return result;
#endif #endif
} }

View File

@@ -56,10 +56,12 @@ public:
static QString buildDebuggingHelperLibrary(const QString &qmakePath, const QString &make, const Environment &env); 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); 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 &qmakePath);
static QStringList debuggingHelperLibraryLocations(const QString &qtInstallData, const QString &qtpath); 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: private:
static QStringList debuggingHelperLibraryDirectories(const QString &qtInstallData, const QString &qtpath); static QStringList debuggingHelperLibraryDirectories(const QString &qtInstallData, const QString &qtpath);

View File

@@ -3,14 +3,43 @@
#include "ui_qtversionmanager.h" #include "ui_qtversionmanager.h"
#include "qt4projectmanagerconstants.h" #include "qt4projectmanagerconstants.h"
#include "qtversionmanager.h" #include "qtversionmanager.h"
#include <coreplugin/coreconstants.h>
#include <utils/treewidgetcolumnstretcher.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/QDebug>
#include <QtCore/QDir> #include <QtCore/QDir>
#include <QtCore/QDateTime>
using namespace Qt4ProjectManager; using namespace Qt4ProjectManager;
using namespace Qt4ProjectManager::Internal; 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 // QtOptionsPage
/// ///
@@ -65,6 +94,11 @@ void QtOptionsPage::apply()
QtOptionsPageWidget::QtOptionsPageWidget(QWidget *parent, QList<QtVersion *> versions, QtVersion *defaultVersion) QtOptionsPageWidget::QtOptionsPageWidget(QWidget *parent, QList<QtVersion *> versions, QtVersion *defaultVersion)
: QWidget(parent) : 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_defaultVersion(versions.indexOf(defaultVersion))
, m_specifyNameString(tr("<specify a name>")) , m_specifyNameString(tr("<specify a name>"))
, m_specifyPathString(tr("<specify a path>")) , m_specifyPathString(tr("<specify a path>"))
@@ -106,10 +140,7 @@ QtOptionsPageWidget::QtOptionsPageWidget(QWidget *parent, QList<QtVersion *> ver
item->setData(0, Qt::UserRole, version->uniqueId()); item->setData(0, Qt::UserRole, version->uniqueId());
if (version->isValid()) { if (version->isValid()) {
if (version->hasDebuggingHelper()) item->setData(2, Qt::DecorationRole, version->hasDebuggingHelper() ? m_debuggingHelperOkIcon : m_debuggingHelperErrorIcon);
item->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/ok.png"));
else
item->setData(2, Qt::DecorationRole, QIcon(":/extensionsystem/images/error.png"));
} else { } else {
item->setData(2, Qt::DecorationRole, QIcon()); item->setData(2, Qt::DecorationRole, QIcon());
} }
@@ -160,27 +191,62 @@ QtOptionsPageWidget::QtOptionsPageWidget(QWidget *parent, QList<QtVersion *> ver
updateState(); updateState();
} }
QtVersion *QtOptionsPageWidget::currentVersion() const
{
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;
}
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;
}
// 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();
}
}
void QtOptionsPageWidget::buildDebuggingHelper() void QtOptionsPageWidget::buildDebuggingHelper()
{ {
// Find the qt version for this button.. if (QtVersion *version = currentVersion()) {
QTreeWidgetItem *currentItem = m_ui->qtdirList->currentItem(); m_ui->showLogButton->setEnabled(false);
int currentItemIndex = indexForTreeItem(currentItem); // Run a debugging helper build task in the background.
if (currentItemIndex < 0) DebuggingHelperBuildTask *buildTask = new DebuggingHelperBuildTask(*version);
return; connect(buildTask, SIGNAL(finished(QString,QString)), this, SLOT(debuggingHelperBuildFinished(QString,QString)),
Qt::QueuedConnection);
QtVersion *version = m_versions[currentItemIndex]; QFuture<void> task = QtConcurrent::run(buildTask, &DebuggingHelperBuildTask::run);
const QString taskName = tr("Building helpers");
QString result = m_versions.at(currentItemIndex)->buildDebuggingHelperLibrary(); Core::ICore::instance()->progressManager()->addTask(task, taskName,
currentItem->setData(2, Qt::UserRole, result); QLatin1String("Qt4ProjectManager::BuildHelpers"),
Core::ProgressManager::CloseOnSuccess);
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"));
} }
m_ui->showLogButton->setEnabled(true);
} }
void QtOptionsPageWidget::showDebuggingBuildLog() void QtOptionsPageWidget::showDebuggingBuildLog()
@@ -190,10 +256,13 @@ void QtOptionsPageWidget::showDebuggingBuildLog()
int currentItemIndex = indexForTreeItem(currentItem); int currentItemIndex = indexForTreeItem(currentItem);
if (currentItemIndex < 0) if (currentItemIndex < 0)
return; return;
// Show text and scroll to bottom
QDialog dlg; QDialog dlg;
Ui_ShowBuildLog ui; Ui_ShowBuildLog ui;
ui.setupUi(&dlg); ui.setupUi(&dlg);
ui.log->setPlainText(currentItem->data(2, Qt::UserRole).toString()); ui.log->setPlainText(currentItem->data(2, Qt::UserRole).toString());
ui.log->moveCursor(QTextCursor::End);
ui.log->ensureCursorVisible();
dlg.exec(); dlg.exec();
} }
@@ -243,12 +312,41 @@ void QtOptionsPageWidget::removeQtDir()
updateState(); 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&nbsp;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() void QtOptionsPageWidget::updateState()
{ {
int currentIndex = indexForTreeItem(m_ui->qtdirList->currentItem()); const QtVersion *version = currentVersion();
bool enabled = (currentIndex >= 0); const bool enabled = version != 0;
bool isAutodetected = (enabled const bool isAutodetected = enabled && version->isAutodetected();
&& m_versions.at(currentIndex)->isAutodetected());
m_ui->delButton->setEnabled(enabled && !isAutodetected); m_ui->delButton->setEnabled(enabled && !isAutodetected);
m_ui->nameEdit->setEnabled(enabled && !isAutodetected); m_ui->nameEdit->setEnabled(enabled && !isAutodetected);
m_ui->qtPath->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(); bool hasLog = enabled && !m_ui->qtdirList->currentItem()->data(2, Qt::UserRole).toString().isEmpty();
m_ui->showLogButton->setEnabled(hasLog); m_ui->showLogButton->setEnabled(hasLog);
QtVersion *version = 0;
if (enabled)
version = m_versions.at(currentIndex);
if (version) { if (version) {
m_ui->rebuildButton->setEnabled(version->isValid()); m_ui->rebuildButton->setEnabled(version->isValid());
if (version->hasDebuggingHelper()) updateDebuggingHelperStateLabel(version);
m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/ok.png"));
else
m_ui->debuggingHelperStateLabel->setPixmap(QPixmap(":/extensionsystem/images/error.png"));
} else { } else {
m_ui->rebuildButton->setEnabled(false); m_ui->rebuildButton->setEnabled(false);
m_ui->debuggingHelperStateLabel->setPixmap(QPixmap()); 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()) if (!item || !item->parent())
return -1; 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) { for (int index = 0; index < m_versions.size(); ++index) {
if (m_versions.at(index)->uniqueId() == uniqueId) if (m_versions.at(index)->uniqueId() == uniqueId)
return index; return index;
@@ -507,16 +599,10 @@ void QtOptionsPageWidget::updateCurrentQtPath()
showEnvironmentPage(currentItem); showEnvironmentPage(currentItem);
if (m_versions[currentItemIndex]->isValid()) { const QtVersion *version = m_versions.at(currentItemIndex);
bool hasLog = !currentItem->data(2, Qt::UserRole).toString().isEmpty(); if (version->isValid()) {
bool hasHelper = m_versions[currentItemIndex]->hasDebuggingHelper(); const bool hasLog = !currentItem->data(2, Qt::UserRole).toString().isEmpty();
if (hasHelper) { currentItem->setData(2, Qt::DecorationRole, version->hasDebuggingHelper() ? m_debuggingHelperOkIcon : m_debuggingHelperErrorIcon);
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"));
}
m_ui->showLogButton->setEnabled(hasLog); m_ui->showLogButton->setEnabled(hasLog);
m_ui->rebuildButton->setEnabled(true); m_ui->rebuildButton->setEnabled(true);
} else { } else {
@@ -524,6 +610,7 @@ void QtOptionsPageWidget::updateCurrentQtPath()
m_ui->debuggingHelperStateLabel->setPixmap(QPixmap()); m_ui->debuggingHelperStateLabel->setPixmap(QPixmap());
m_ui->rebuildButton->setEnabled(true); m_ui->rebuildButton->setEnabled(true);
} }
updateDebuggingHelperStateLabel(version);
} }
void QtOptionsPageWidget::updateCurrentMingwDirectory() void QtOptionsPageWidget::updateCurrentMingwDirectory()

View File

@@ -32,6 +32,8 @@
#include <coreplugin/dialogs/ioptionspage.h> #include <coreplugin/dialogs/ioptionspage.h>
#include <QtGui/QWidget> #include <QtGui/QWidget>
#include <QtGui/QPixmap>
#include <QtGui/QIcon>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QTreeWidgetItem; class QTreeWidgetItem;
@@ -46,6 +48,24 @@ namespace Ui {
class QtVersionManager; 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 class QtOptionsPageWidget : public QWidget
{ {
Q_OBJECT Q_OBJECT
@@ -57,11 +77,17 @@ public:
void finish(); void finish();
private: private:
const QPixmap m_debuggingHelperOkPixmap;
const QPixmap m_debuggingHelperErrorPixmap;
const QIcon m_debuggingHelperOkIcon;
const QIcon m_debuggingHelperErrorIcon;
void showEnvironmentPage(QTreeWidgetItem * item); void showEnvironmentPage(QTreeWidgetItem * item);
void fixQtVersionName(int index); void fixQtVersionName(int index);
int indexForWidget(QWidget *debuggingHelperWidget) const; int indexForTreeItem(const QTreeWidgetItem *item) const;
int indexForTreeItem(QTreeWidgetItem *item) const;
QTreeWidgetItem *treeItemForIndex(int index) const; QTreeWidgetItem *treeItemForIndex(int index) const;
QtVersion *currentVersion() const;
void updateDebuggingHelperStateLabel(const QtVersion *version = 0);
Internal::Ui::QtVersionManager *m_ui; Internal::Ui::QtVersionManager *m_ui;
QList<QtVersion *> m_versions; QList<QtVersion *> m_versions;
@@ -89,6 +115,7 @@ private slots:
void msvcVersionChanged(); void msvcVersionChanged();
void buildDebuggingHelper(); void buildDebuggingHelper();
void showDebuggingBuildLog(); void showDebuggingBuildLog();
void debuggingHelperBuildFinished(const QString &name, const QString &output);
}; };
class QtOptionsPage : public Core::IOptionsPage class QtOptionsPage : public Core::IOptionsPage

View File

@@ -1266,8 +1266,10 @@ QString QtVersion::buildDebuggingHelperLibrary()
// TODO: the debugging helper doesn't comply to actual tool chain yet // TODO: the debugging helper doesn't comply to actual tool chain yet
ProjectExplorer::ToolChain *tc = createToolChain(defaultToolchainType()); ProjectExplorer::ToolChain *tc = createToolChain(defaultToolchainType());
tc->addToEnvironment(env); tc->addToEnvironment(env);
QString directory = DebuggingHelperLibrary::copyDebuggingHelperLibrary(qtInstallData, path()); QString output;
QString output = DebuggingHelperLibrary::buildDebuggingHelperLibrary(directory, tc->makeCommand(), qmakeCommand(), mkspec(), env); QString directory = DebuggingHelperLibrary::copyDebuggingHelperLibrary(qtInstallData, path(), &output);
if (!directory.isEmpty())
output += DebuggingHelperLibrary::buildDebuggingHelperLibrary(directory, tc->makeCommand(), qmakeCommand(), mkspec(), env);
m_hasDebuggingHelper = !debuggingHelperLibrary().isEmpty(); m_hasDebuggingHelper = !debuggingHelperLibrary().isEmpty();
delete tc; delete tc;
return output; return output;

View File

@@ -17,31 +17,6 @@
<string>Qt versions</string> <string>Qt versions</string>
</property> </property>
<layout class="QGridLayout" name="gridLayout"> <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"> <item row="0" column="2">
<layout class="QVBoxLayout"> <layout class="QVBoxLayout">
<property name="spacing"> <property name="spacing">
@@ -91,43 +66,6 @@
</item> </item>
</layout> </layout>
</item> </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"> <item row="4" column="1">
<layout class="QHBoxLayout" name="horizontalLayout"> <layout class="QHBoxLayout" name="horizontalLayout">
<property name="spacing"> <property name="spacing">
@@ -162,23 +100,6 @@ p, li { white-space: pre-wrap; }
</item> </item>
</layout> </layout>
</item> </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"> <item row="6" column="1">
<layout class="QHBoxLayout" name="horizontalLayout_2"> <layout class="QHBoxLayout" name="horizontalLayout_2">
<item> <item>
@@ -210,6 +131,75 @@ p, li { white-space: pre-wrap; }
</item> </item>
</layout> </layout>
</item> </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"> <item row="7" column="1">
<widget class="QLabel" name="errorLabel"> <widget class="QLabel" name="errorLabel">
<property name="text"> <property name="text">
@@ -217,6 +207,16 @@ p, li { white-space: pre-wrap; }
</property> </property>
</widget> </widget>
</item> </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> </layout>
<zorder>qtdirList</zorder> <zorder>qtdirList</zorder>
<zorder>versionNameLabel</zorder> <zorder>versionNameLabel</zorder>
@@ -226,12 +226,10 @@ p, li { white-space: pre-wrap; }
<zorder>mingwLabel</zorder> <zorder>mingwLabel</zorder>
<zorder>mingwPath</zorder> <zorder>mingwPath</zorder>
<zorder>msvcLabel</zorder> <zorder>msvcLabel</zorder>
<zorder>label</zorder> <zorder>debuggingHelperLabel</zorder>
<zorder>errorLabel</zorder> <zorder>errorLabel</zorder>
<zorder>mwcLabel</zorder> <zorder>mwcLabel</zorder>
<zorder>mwcPath</zorder> <zorder>mwcPath</zorder>
<zorder>msvcComboBox</zorder>
<zorder>msvcNotFoundLabel</zorder>
</widget> </widget>
</item> </item>
<item> <item>