forked from qt-creator/qt-creator
In AutoTest, use the toolbar and menu variants accordingly. Change-Id: I1405de1ace1efe6c3e51b17356a9e4dfb3b11dcd Reviewed-by: Christian Stenger <christian.stenger@qt.io>
464 lines
17 KiB
C++
464 lines
17 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "autotestplugin.h"
|
|
|
|
#include "autotestconstants.h"
|
|
#include "autotesticons.h"
|
|
#include "projectsettingswidget.h"
|
|
#include "testcodeparser.h"
|
|
#include "testframeworkmanager.h"
|
|
#include "testnavigationwidget.h"
|
|
#include "testprojectsettings.h"
|
|
#include "testresultspane.h"
|
|
#include "testrunner.h"
|
|
#include "testsettings.h"
|
|
#include "testsettingspage.h"
|
|
#include "testtreeitem.h"
|
|
#include "testtreemodel.h"
|
|
#include "testtreeview.h"
|
|
|
|
#include "boost/boosttestframework.h"
|
|
#include "catch/catchframework.h"
|
|
#include "gtest/gtestframework.h"
|
|
#include "qtest/qttestframework.h"
|
|
#include "quick/quicktestframework.h"
|
|
|
|
#include <coreplugin/actionmanager/actioncontainer.h>
|
|
#include <coreplugin/actionmanager/actionmanager.h>
|
|
#include <coreplugin/actionmanager/command.h>
|
|
#include <coreplugin/coreconstants.h>
|
|
#include <coreplugin/icontext.h>
|
|
#include <coreplugin/icore.h>
|
|
#include <coreplugin/messagemanager.h>
|
|
#include <cppeditor/cppeditorconstants.h>
|
|
#include <extensionsystem/pluginmanager.h>
|
|
#include <projectexplorer/buildmanager.h>
|
|
#include <projectexplorer/project.h>
|
|
#include <projectexplorer/projectexplorer.h>
|
|
#include <projectexplorer/projectexplorericons.h>
|
|
#include <projectexplorer/projectpanelfactory.h>
|
|
#include <projectexplorer/runcontrol.h>
|
|
#include <projectexplorer/session.h>
|
|
#include <projectexplorer/target.h>
|
|
#include <texteditor/textdocument.h>
|
|
#include <texteditor/texteditor.h>
|
|
#include <utils/textutils.h>
|
|
#include <utils/utilsicons.h>
|
|
|
|
#include <QAction>
|
|
#include <QList>
|
|
#include <QMainWindow>
|
|
#include <QMap>
|
|
#include <QMenu>
|
|
#include <QMessageBox>
|
|
#include <QTextCursor>
|
|
|
|
#ifdef WITH_TESTS
|
|
#include "autotestunittests.h"
|
|
#endif
|
|
|
|
using namespace Core;
|
|
|
|
namespace Autotest {
|
|
namespace Internal {
|
|
|
|
class AutotestPluginPrivate : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
AutotestPluginPrivate();
|
|
~AutotestPluginPrivate() override;
|
|
|
|
TestNavigationWidgetFactory *m_navigationWidgetFactory = nullptr;
|
|
TestResultsPane *m_resultsPane = nullptr;
|
|
QMap<QString, ChoicePair> m_runconfigCache;
|
|
|
|
void initializeMenuEntries();
|
|
void onRunAllTriggered();
|
|
void onRunSelectedTriggered();
|
|
void onRunFailedTriggered();
|
|
void onRunFileTriggered();
|
|
void onRunUnderCursorTriggered(TestRunMode mode);
|
|
|
|
TestSettings m_settings;
|
|
TestSettingsPage m_testSettingPage{&m_settings};
|
|
|
|
TestCodeParser m_testCodeParser;
|
|
TestTreeModel m_testTreeModel{&m_testCodeParser};
|
|
TestRunner m_testRunner;
|
|
TestFrameworkManager m_frameworkManager;
|
|
};
|
|
|
|
static AutotestPluginPrivate *dd = nullptr;
|
|
static QHash<ProjectExplorer::Project *, TestProjectSettings *> s_projectSettings;
|
|
|
|
AutotestPlugin::AutotestPlugin()
|
|
{
|
|
// needed to be used in QueuedConnection connects
|
|
qRegisterMetaType<TestResult>();
|
|
qRegisterMetaType<TestTreeItem *>();
|
|
qRegisterMetaType<TestCodeLocationAndType>();
|
|
// warm up meta type system to be able to read Qt::CheckState with persistent settings
|
|
qRegisterMetaType<Qt::CheckState>();
|
|
}
|
|
|
|
AutotestPlugin::~AutotestPlugin()
|
|
{
|
|
delete dd;
|
|
dd = nullptr;
|
|
}
|
|
|
|
AutotestPluginPrivate::AutotestPluginPrivate()
|
|
{
|
|
dd = this; // Needed as the code below access it via the static plugin interface
|
|
initializeMenuEntries();
|
|
m_frameworkManager.registerTestFramework(new QtTestFramework);
|
|
m_frameworkManager.registerTestFramework(new QuickTestFramework);
|
|
m_frameworkManager.registerTestFramework(new GTestFramework);
|
|
m_frameworkManager.registerTestFramework(new BoostTestFramework);
|
|
m_frameworkManager.registerTestFramework(new CatchFramework);
|
|
|
|
m_frameworkManager.synchronizeSettings(ICore::settings());
|
|
m_navigationWidgetFactory = new TestNavigationWidgetFactory;
|
|
m_resultsPane = TestResultsPane::instance();
|
|
|
|
auto panelFactory = new ProjectExplorer::ProjectPanelFactory();
|
|
panelFactory->setPriority(666);
|
|
// panelFactory->setIcon(); // TODO ?
|
|
panelFactory->setDisplayName(tr("Testing"));
|
|
panelFactory->setCreateWidgetFunction([](ProjectExplorer::Project *project) {
|
|
return new ProjectTestSettingsWidget(project);
|
|
});
|
|
ProjectExplorer::ProjectPanelFactory::registerFactory(panelFactory);
|
|
|
|
TestFrameworkManager::activateFrameworksAndToolsFromSettings(&m_settings);
|
|
m_testTreeModel.synchronizeTestFrameworks();
|
|
m_testTreeModel.synchronizeTestTools();
|
|
|
|
auto sessionManager = ProjectExplorer::SessionManager::instance();
|
|
connect(sessionManager, &ProjectExplorer::SessionManager::startupProjectChanged,
|
|
this, [this] { m_runconfigCache.clear(); });
|
|
|
|
connect(sessionManager, &ProjectExplorer::SessionManager::aboutToRemoveProject,
|
|
this, [] (ProjectExplorer::Project *project) {
|
|
auto it = s_projectSettings.find(project);
|
|
if (it != s_projectSettings.end()) {
|
|
delete it.value();
|
|
s_projectSettings.erase(it);
|
|
}
|
|
});
|
|
}
|
|
|
|
AutotestPluginPrivate::~AutotestPluginPrivate()
|
|
{
|
|
if (!s_projectSettings.isEmpty()) {
|
|
qDeleteAll(s_projectSettings);
|
|
s_projectSettings.clear();
|
|
}
|
|
|
|
delete m_navigationWidgetFactory;
|
|
delete m_resultsPane;
|
|
}
|
|
|
|
TestSettings *AutotestPlugin::settings()
|
|
{
|
|
return &dd->m_settings;
|
|
}
|
|
|
|
TestProjectSettings *AutotestPlugin::projectSettings(ProjectExplorer::Project *project)
|
|
{
|
|
auto &settings = s_projectSettings[project];
|
|
if (!settings)
|
|
settings = new TestProjectSettings(project);
|
|
|
|
return settings;
|
|
}
|
|
|
|
void AutotestPluginPrivate::initializeMenuEntries()
|
|
{
|
|
ActionContainer *menu = ActionManager::createMenu(Constants::MENU_ID);
|
|
menu->menu()->setTitle(tr("&Tests"));
|
|
menu->setOnAllDisabledBehavior(ActionContainer::Show);
|
|
|
|
QAction *action = new QAction(tr("Run &All Tests"), this);
|
|
action->setIcon(Utils::Icons::RUN_SMALL_TOOLBAR.icon());
|
|
action->setToolTip(tr("Run All Tests"));
|
|
Command *command = ActionManager::registerAction(action, Constants::ACTION_RUN_ALL_ID);
|
|
command->setDefaultKeySequence(
|
|
QKeySequence(useMacShortcuts ? tr("Ctrl+Meta+T, Ctrl+Meta+A") : tr("Alt+Shift+T,Alt+A")));
|
|
connect(action, &QAction::triggered, this, &AutotestPluginPrivate::onRunAllTriggered);
|
|
action->setEnabled(false);
|
|
menu->addAction(command);
|
|
|
|
action = new QAction(tr("&Run Selected Tests"), this);
|
|
action->setIcon(Utils::Icons::RUN_SELECTED.icon());
|
|
action->setToolTip(tr("Run Selected Tests"));
|
|
command = ActionManager::registerAction(action, Constants::ACTION_RUN_SELECTED_ID);
|
|
command->setDefaultKeySequence(
|
|
QKeySequence(useMacShortcuts ? tr("Ctrl+Meta+T, Ctrl+Meta+R") : tr("Alt+Shift+T,Alt+R")));
|
|
connect(action, &QAction::triggered, this, &AutotestPluginPrivate::onRunSelectedTriggered);
|
|
action->setEnabled(false);
|
|
menu->addAction(command);
|
|
|
|
action = new QAction(tr("Run &Failed Tests"), this);
|
|
Utils::Icon runFailedIcon = Utils::Icons::RUN_SMALL_TOOLBAR;
|
|
for (const Utils::IconMaskAndColor &maskAndColor: Icons::RUN_FAILED_OVERLAY)
|
|
runFailedIcon.append(maskAndColor);
|
|
action->setIcon(runFailedIcon.icon());
|
|
action->setToolTip(tr("Run Failed Tests"));
|
|
command = ActionManager::registerAction(action, Constants::ACTION_RUN_FAILED_ID);
|
|
command->setDefaultKeySequence(
|
|
useMacShortcuts ? tr("Ctrl+Meta+T, Ctrl+Meta+F") : tr("Alt+Shift+T,Alt+F"));
|
|
connect(action, &QAction::triggered, this, &AutotestPluginPrivate::onRunFailedTriggered);
|
|
action->setEnabled(false);
|
|
menu->addAction(command);
|
|
|
|
action = new QAction(tr("Run Tests for &Current File"), this);
|
|
action->setIcon(Utils::Icons::RUN_FILE.icon());
|
|
action->setToolTip(tr("Run Tests for Current File"));
|
|
command = ActionManager::registerAction(action, Constants::ACTION_RUN_FILE_ID);
|
|
command->setDefaultKeySequence(
|
|
QKeySequence(useMacShortcuts ? tr("Ctrl+Meta+T, Ctrl+Meta+C") : tr("Alt+Shift+T,Alt+C")));
|
|
connect(action, &QAction::triggered, this, &AutotestPluginPrivate::onRunFileTriggered);
|
|
action->setEnabled(false);
|
|
menu->addAction(command);
|
|
|
|
action = new QAction(tr("Re&scan Tests"), this);
|
|
command = ActionManager::registerAction(action, Constants::ACTION_SCAN_ID);
|
|
command->setDefaultKeySequence(
|
|
QKeySequence(useMacShortcuts ? tr("Ctrl+Meta+T, Ctrl+Meta+S") : tr("Alt+Shift+T,Alt+S")));
|
|
|
|
connect(action, &QAction::triggered, this, [] { dd->m_testCodeParser.updateTestTree(); });
|
|
menu->addAction(command);
|
|
|
|
ActionContainer *toolsMenu = ActionManager::actionContainer(Core::Constants::M_TOOLS);
|
|
toolsMenu->addMenu(menu);
|
|
using namespace ProjectExplorer;
|
|
connect(BuildManager::instance(), &BuildManager::buildStateChanged,
|
|
this, &AutotestPlugin::updateMenuItemsEnabledState);
|
|
connect(BuildManager::instance(), &BuildManager::buildQueueFinished,
|
|
this, &AutotestPlugin::updateMenuItemsEnabledState);
|
|
connect(ProjectExplorerPlugin::instance(), &ProjectExplorerPlugin::runActionsUpdated,
|
|
this, &AutotestPlugin::updateMenuItemsEnabledState);
|
|
connect(&dd->m_testTreeModel, &TestTreeModel::testTreeModelChanged,
|
|
this, &AutotestPlugin::updateMenuItemsEnabledState);
|
|
}
|
|
|
|
bool AutotestPlugin::initialize(const QStringList &arguments, QString *errorString)
|
|
{
|
|
Q_UNUSED(arguments)
|
|
Q_UNUSED(errorString)
|
|
|
|
dd = new AutotestPluginPrivate;
|
|
return true;
|
|
}
|
|
|
|
void AutotestPlugin::extensionsInitialized()
|
|
{
|
|
ActionContainer *contextMenu = ActionManager::actionContainer(CppEditor::Constants::M_CONTEXT);
|
|
if (!contextMenu) // if QC is started without CppEditor plugin
|
|
return;
|
|
|
|
QAction *action = new QAction(tr("&Run Test Under Cursor"), this);
|
|
action->setEnabled(false);
|
|
action->setIcon(Utils::Icons::RUN_SMALL.icon());
|
|
|
|
Command *command = ActionManager::registerAction(action, Constants::ACTION_RUN_UCURSOR);
|
|
connect(action, &QAction::triggered,
|
|
std::bind(&AutotestPluginPrivate::onRunUnderCursorTriggered, dd, TestRunMode::Run));
|
|
contextMenu->addSeparator();
|
|
contextMenu->addAction(command);
|
|
|
|
action = new QAction(tr("&Debug Test Under Cursor"), this);
|
|
action->setEnabled(false);
|
|
action->setIcon(ProjectExplorer::Icons::DEBUG_START_SMALL.icon());
|
|
|
|
command = ActionManager::registerAction(action, Constants::ACTION_RUN_DBG_UCURSOR);
|
|
connect(action, &QAction::triggered,
|
|
std::bind(&AutotestPluginPrivate::onRunUnderCursorTriggered, dd, TestRunMode::Debug));
|
|
contextMenu->addAction(command);
|
|
contextMenu->addSeparator();
|
|
}
|
|
|
|
ExtensionSystem::IPlugin::ShutdownFlag AutotestPlugin::aboutToShutdown()
|
|
{
|
|
dd->m_testCodeParser.aboutToShutdown();
|
|
dd->m_testTreeModel.disconnect();
|
|
return SynchronousShutdown;
|
|
}
|
|
|
|
void AutotestPluginPrivate::onRunAllTriggered()
|
|
{
|
|
m_testRunner.setSelectedTests(m_testTreeModel.getAllTestCases());
|
|
m_testRunner.prepareToRunTests(TestRunMode::Run);
|
|
}
|
|
|
|
void AutotestPluginPrivate::onRunSelectedTriggered()
|
|
{
|
|
m_testRunner.setSelectedTests(m_testTreeModel.getSelectedTests());
|
|
m_testRunner.prepareToRunTests(TestRunMode::Run);
|
|
}
|
|
|
|
void AutotestPluginPrivate::onRunFailedTriggered()
|
|
{
|
|
const QList<ITestConfiguration *> failed = m_testTreeModel.getFailedTests();
|
|
if (failed.isEmpty()) // the framework might not be able to provide them
|
|
return;
|
|
m_testRunner.setSelectedTests(failed);
|
|
m_testRunner.prepareToRunTests(TestRunMode::Run);
|
|
}
|
|
|
|
void AutotestPluginPrivate::onRunFileTriggered()
|
|
{
|
|
const IDocument *document = EditorManager::currentDocument();
|
|
if (!document)
|
|
return;
|
|
|
|
const Utils::FilePath &fileName = document->filePath();
|
|
if (fileName.isEmpty())
|
|
return;
|
|
|
|
const QList<ITestConfiguration *> tests = m_testTreeModel.getTestsForFile(fileName);
|
|
if (tests.isEmpty())
|
|
return;
|
|
|
|
m_testRunner.setSelectedTests(tests);
|
|
m_testRunner.prepareToRunTests(TestRunMode::Run);
|
|
}
|
|
|
|
static QList<ITestConfiguration *> testItemsToTestConfigurations(const QList<TestTreeItem *> &items,
|
|
TestRunMode mode)
|
|
{
|
|
QList<ITestConfiguration *> configs;
|
|
for (const TestTreeItem * item : items) {
|
|
if (ITestConfiguration *currentConfig = item->asConfiguration(mode))
|
|
configs << currentConfig;
|
|
}
|
|
return configs;
|
|
}
|
|
|
|
void AutotestPluginPrivate::onRunUnderCursorTriggered(TestRunMode mode)
|
|
{
|
|
TextEditor::BaseTextEditor *currentEditor = TextEditor::BaseTextEditor::currentTextEditor();
|
|
QTextCursor cursor = currentEditor->editorWidget()->textCursor();
|
|
cursor.select(QTextCursor::WordUnderCursor);
|
|
const QString text = cursor.selectedText();
|
|
if (text.isEmpty())
|
|
return; // Do not trigger when no name under cursor
|
|
|
|
const QList<TestTreeItem *> testsItems = m_testTreeModel.testItemsByName(text);
|
|
if (testsItems.isEmpty())
|
|
return; // Wrong location triggered
|
|
|
|
// check whether we have been triggered on a test function definition
|
|
const int line = currentEditor->currentLine();
|
|
const QString &filePath = currentEditor->textDocument()->filePath().toString();
|
|
const QList<TestTreeItem *> filteredItems = Utils::filtered(testsItems, [&](TestTreeItem *it){
|
|
return it->line() == line && it->filePath() == filePath;
|
|
});
|
|
|
|
const QList<ITestConfiguration *> testsToRun = testItemsToTestConfigurations(
|
|
filteredItems.size() == 1 ? filteredItems : testsItems, mode);
|
|
|
|
if (testsToRun.isEmpty()) {
|
|
MessageManager::write(tr("Selected test was not found (%1).").arg(text), MessageManager::Flash);
|
|
return;
|
|
}
|
|
|
|
m_testRunner.setSelectedTests(testsToRun);
|
|
m_testRunner.prepareToRunTests(mode);
|
|
}
|
|
|
|
void AutotestPlugin::updateMenuItemsEnabledState()
|
|
{
|
|
const ProjectExplorer::Project *project = ProjectExplorer::SessionManager::startupProject();
|
|
const ProjectExplorer::Target *target = project ? project->activeTarget() : nullptr;
|
|
const bool canScan = !dd->m_testRunner.isTestRunning()
|
|
&& dd->m_testCodeParser.state() == TestCodeParser::Idle;
|
|
const bool hasTests = dd->m_testTreeModel.hasTests();
|
|
// avoid expensive call to PE::canRunStartupProject() - limit to minimum necessary checks
|
|
const bool canRun = hasTests && canScan
|
|
&& project && !project->needsConfiguration()
|
|
&& target && target->activeRunConfiguration()
|
|
&& !ProjectExplorer::BuildManager::isBuilding();
|
|
const bool canRunFailed = canRun && dd->m_testTreeModel.hasFailedTests();
|
|
|
|
ActionManager::command(Constants::ACTION_RUN_ALL_ID)->action()->setEnabled(canRun);
|
|
ActionManager::command(Constants::ACTION_RUN_SELECTED_ID)->action()->setEnabled(canRun);
|
|
ActionManager::command(Constants::ACTION_RUN_FAILED_ID)->action()->setEnabled(canRunFailed);
|
|
ActionManager::command(Constants::ACTION_RUN_FILE_ID)->action()->setEnabled(canRun);
|
|
ActionManager::command(Constants::ACTION_SCAN_ID)->action()->setEnabled(canScan);
|
|
|
|
ActionContainer *contextMenu = ActionManager::actionContainer(CppEditor::Constants::M_CONTEXT);
|
|
if (!contextMenu)
|
|
return; // When no context menu, actions do not exists
|
|
|
|
ActionManager::command(Constants::ACTION_RUN_UCURSOR)->action()->setEnabled(canRun);
|
|
ActionManager::command(Constants::ACTION_RUN_DBG_UCURSOR)->action()->setEnabled(canRun);
|
|
}
|
|
|
|
void AutotestPlugin::cacheRunConfigChoice(const QString &buildTargetKey, const ChoicePair &choice)
|
|
{
|
|
if (dd)
|
|
dd->m_runconfigCache.insert(buildTargetKey, choice);
|
|
}
|
|
|
|
ChoicePair AutotestPlugin::cachedChoiceFor(const QString &buildTargetKey)
|
|
{
|
|
return dd ? dd->m_runconfigCache.value(buildTargetKey) : ChoicePair();
|
|
}
|
|
|
|
void AutotestPlugin::clearChoiceCache()
|
|
{
|
|
if (dd)
|
|
dd->m_runconfigCache.clear();
|
|
}
|
|
|
|
void AutotestPlugin::popupResultsPane()
|
|
{
|
|
if (dd)
|
|
dd->m_resultsPane->popup(Core::IOutputPane::NoModeSwitch);
|
|
}
|
|
|
|
QVector<QObject *> AutotestPlugin::createTestObjects() const
|
|
{
|
|
QVector<QObject *> tests;
|
|
#ifdef WITH_TESTS
|
|
tests << new AutoTestUnitTests(&dd->m_testTreeModel);
|
|
#endif
|
|
return tests;
|
|
}
|
|
|
|
bool ChoicePair::matches(const ProjectExplorer::RunConfiguration *rc) const
|
|
{
|
|
return rc && rc->displayName() == displayName && rc->runnable().executable.toString() == executable;
|
|
}
|
|
|
|
} // Internal
|
|
} // Autotest
|
|
|
|
#include "autotestplugin.moc"
|