forked from qt-creator/qt-creator
Saves some code on the user side. Change-Id: I32cd220b6e533f5497a1865f9c34ab9db4cfda79 Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org> Reviewed-by: Eike Ziller <eike.ziller@qt.io>
397 lines
12 KiB
C++
397 lines
12 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 Nicolas Arnaud-Cormos
|
|
** 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 "macromanager.h"
|
|
|
|
#include "macrosconstants.h"
|
|
#include "macroevent.h"
|
|
#include "macro.h"
|
|
#include "imacrohandler.h"
|
|
#include "savedialog.h"
|
|
#include "actionmacrohandler.h"
|
|
#include "texteditormacrohandler.h"
|
|
#include "findmacrohandler.h"
|
|
|
|
#include <texteditor/texteditorconstants.h>
|
|
|
|
#include <coreplugin/coreconstants.h>
|
|
#include <coreplugin/actionmanager/actionmanager.h>
|
|
#include <coreplugin/actionmanager/actioncontainer.h>
|
|
#include <coreplugin/actionmanager/command.h>
|
|
#include <coreplugin/icore.h>
|
|
#include <coreplugin/icontext.h>
|
|
#include <coreplugin/editormanager/editormanager.h>
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
#include <utils/qtcassert.h>
|
|
|
|
#include <QDir>
|
|
#include <QFile>
|
|
#include <QFileInfo>
|
|
#include <QSettings>
|
|
#include <QList>
|
|
|
|
#include <QAction>
|
|
#include <QFileDialog>
|
|
#include <QMessageBox>
|
|
|
|
namespace Macros {
|
|
namespace Internal {
|
|
|
|
/*!
|
|
\namespace Macros
|
|
\brief The Macros namespace contains support for macros in \QC.
|
|
*/
|
|
|
|
/*!
|
|
|
|
\class Macro::MacroManager
|
|
\brief The MacroManager class implements a manager for macros.
|
|
|
|
The MacroManager manages all macros, loads them on startup, keeps track of the
|
|
current macro, and creates new macros.
|
|
|
|
There are two important functions in this class that can be used outside the Macros plugin:
|
|
\list
|
|
\li registerEventHandler: add a new event handler
|
|
\li registerAction: add a macro event when this action is triggered
|
|
\endlist
|
|
|
|
This class is a singleton and can be accessed using the instance function.
|
|
*/
|
|
|
|
/*!
|
|
\fn void registerAction(QAction *action, const QString &id)
|
|
|
|
Appends \a action to the list of actions registered in a macro. \a id is
|
|
the action id passed to the ActionManager.
|
|
*/
|
|
|
|
class MacroManagerPrivate
|
|
{
|
|
public:
|
|
MacroManagerPrivate(MacroManager *qq);
|
|
|
|
MacroManager *q;
|
|
QMap<QString, Macro *> macros;
|
|
QMap<QString, QAction *> actions;
|
|
Macro *currentMacro = nullptr;
|
|
bool isRecording = false;
|
|
|
|
QList<IMacroHandler*> handlers;
|
|
|
|
ActionMacroHandler *actionHandler;
|
|
TextEditorMacroHandler *textEditorHandler;
|
|
FindMacroHandler *findHandler;
|
|
|
|
void initialize();
|
|
void addMacro(Macro *macro);
|
|
void removeMacro(const QString &name);
|
|
void changeMacroDescription(Macro *macro, const QString &description);
|
|
|
|
bool executeMacro(Macro *macro);
|
|
void showSaveDialog();
|
|
};
|
|
|
|
MacroManagerPrivate::MacroManagerPrivate(MacroManager *qq):
|
|
q(qq)
|
|
{
|
|
// Load existing macros
|
|
initialize();
|
|
|
|
actionHandler = new ActionMacroHandler;
|
|
textEditorHandler = new TextEditorMacroHandler;
|
|
findHandler = new FindMacroHandler;
|
|
}
|
|
|
|
void MacroManagerPrivate::initialize()
|
|
{
|
|
macros.clear();
|
|
const QDir dir(MacroManager::macrosDirectory());
|
|
QStringList filter;
|
|
filter << QLatin1String("*.") + QLatin1String(Constants::M_EXTENSION);
|
|
QStringList files = dir.entryList(filter, QDir::Files);
|
|
|
|
foreach (const QString &name, files) {
|
|
QString fileName = dir.absolutePath() + QLatin1Char('/') + name;
|
|
auto macro = new Macro;
|
|
if (macro->loadHeader(fileName))
|
|
addMacro(macro);
|
|
else
|
|
delete macro;
|
|
}
|
|
}
|
|
|
|
static Utils::Id makeId(const QString &name)
|
|
{
|
|
return Utils::Id(Macros::Constants::PREFIX_MACRO).withSuffix(name);
|
|
}
|
|
|
|
void MacroManagerPrivate::addMacro(Macro *macro)
|
|
{
|
|
// Add sortcut
|
|
Core::Context context(TextEditor::Constants::C_TEXTEDITOR);
|
|
auto action = new QAction(macro->description(), q);
|
|
Core::Command *command = Core::ActionManager::registerAction(
|
|
action, makeId(macro->displayName()), context);
|
|
command->setAttribute(Core::Command::CA_UpdateText);
|
|
QObject::connect(action, &QAction::triggered, q, [this, macro]() {
|
|
q->executeMacro(macro->displayName());
|
|
});
|
|
|
|
// Add macro to the map
|
|
macros[macro->displayName()] = macro;
|
|
actions[macro->displayName()] = action;
|
|
}
|
|
|
|
void MacroManagerPrivate::removeMacro(const QString &name)
|
|
{
|
|
if (!macros.contains(name))
|
|
return;
|
|
// Remove shortcut
|
|
QAction *action = actions.take(name);
|
|
Core::ActionManager::unregisterAction(action, makeId(name));
|
|
delete action;
|
|
|
|
// Remove macro from the map
|
|
Macro *macro = macros.take(name);
|
|
if (macro == currentMacro)
|
|
currentMacro = nullptr;
|
|
delete macro;
|
|
}
|
|
|
|
void MacroManagerPrivate::changeMacroDescription(Macro *macro, const QString &description)
|
|
{
|
|
if (!macro->load())
|
|
return;
|
|
macro->setDescription(description);
|
|
macro->save(macro->fileName(), Core::ICore::dialogParent());
|
|
|
|
QAction *action = actions[macro->displayName()];
|
|
QTC_ASSERT(action, return);
|
|
action->setText(description);
|
|
}
|
|
|
|
bool MacroManagerPrivate::executeMacro(Macro *macro)
|
|
{
|
|
bool error = !macro->load();
|
|
foreach (const MacroEvent ¯oEvent, macro->events()) {
|
|
if (error)
|
|
break;
|
|
foreach (IMacroHandler *handler, handlers) {
|
|
if (handler->canExecuteEvent(macroEvent)) {
|
|
if (!handler->executeEvent(macroEvent))
|
|
error = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (error) {
|
|
QMessageBox::warning(
|
|
Core::ICore::dialogParent(),
|
|
MacroManager::tr("Playing Macro"),
|
|
MacroManager::tr("An error occurred while replaying the macro, execution stopped."));
|
|
}
|
|
|
|
// Set the focus back to the editor
|
|
// TODO: is it really needed??
|
|
if (Core::IEditor *current = Core::EditorManager::currentEditor())
|
|
current->widget()->setFocus(Qt::OtherFocusReason);
|
|
|
|
return !error;
|
|
}
|
|
|
|
void MacroManagerPrivate::showSaveDialog()
|
|
{
|
|
SaveDialog dialog(Core::ICore::dialogParent());
|
|
if (dialog.exec()) {
|
|
if (dialog.name().isEmpty())
|
|
return;
|
|
|
|
// Save in the resource path
|
|
const QString fileName = MacroManager::macrosDirectory() + QLatin1Char('/') + dialog.name()
|
|
+ QLatin1Char('.') + QLatin1String(Constants::M_EXTENSION);
|
|
currentMacro->setDescription(dialog.description());
|
|
currentMacro->save(fileName, Core::ICore::dialogParent());
|
|
addMacro(currentMacro);
|
|
}
|
|
}
|
|
|
|
|
|
// ---------- MacroManager ------------
|
|
MacroManager *m_instance = nullptr;
|
|
|
|
MacroManager::MacroManager() :
|
|
d(new MacroManagerPrivate(this))
|
|
{
|
|
m_instance = this;
|
|
registerMacroHandler(d->actionHandler);
|
|
registerMacroHandler(d->findHandler);
|
|
registerMacroHandler(d->textEditorHandler);
|
|
}
|
|
|
|
MacroManager::~MacroManager()
|
|
{
|
|
// Cleanup macro
|
|
QStringList macroList = d->macros.keys();
|
|
foreach (const QString &name, macroList)
|
|
d->removeMacro(name);
|
|
|
|
// Cleanup handlers
|
|
qDeleteAll(d->handlers);
|
|
|
|
delete d;
|
|
}
|
|
|
|
void MacroManager::startMacro()
|
|
{
|
|
d->isRecording = true;
|
|
// Delete anonymous macro
|
|
if (d->currentMacro && d->currentMacro->displayName().isEmpty())
|
|
delete d->currentMacro;
|
|
d->currentMacro = new Macro;
|
|
|
|
Core::ActionManager::command(Constants::START_MACRO)->action()->setEnabled(false);
|
|
Core::ActionManager::command(Constants::END_MACRO)->action()->setEnabled(true);
|
|
Core::ActionManager::command(Constants::EXECUTE_LAST_MACRO)->action()->setEnabled(false);
|
|
Core::ActionManager::command(Constants::SAVE_LAST_MACRO)->action()->setEnabled(false);
|
|
foreach (IMacroHandler *handler, d->handlers)
|
|
handler->startRecording(d->currentMacro);
|
|
|
|
QString endShortcut = Core::ActionManager::command(Constants::END_MACRO)->keySequence().toString();
|
|
QString executeShortcut = Core::ActionManager::command(Constants::EXECUTE_LAST_MACRO)->keySequence().toString();
|
|
QString help = tr("Macro mode. Type \"%1\" to stop recording and \"%2\" to play the macro.")
|
|
.arg(endShortcut).arg(executeShortcut);
|
|
Core::EditorManager::showEditorStatusBar(Constants::M_STATUS_BUFFER, help,
|
|
tr("Stop Recording Macro"),
|
|
this, [this] { endMacro(); });
|
|
}
|
|
|
|
void MacroManager::endMacro()
|
|
{
|
|
Core::EditorManager::hideEditorStatusBar(QLatin1String(Constants::M_STATUS_BUFFER));
|
|
|
|
Core::ActionManager::command(Constants::START_MACRO)->action()->setEnabled(true);
|
|
Core::ActionManager::command(Constants::END_MACRO)->action()->setEnabled(false);
|
|
Core::ActionManager::command(Constants::EXECUTE_LAST_MACRO)->action()->setEnabled(true);
|
|
Core::ActionManager::command(Constants::SAVE_LAST_MACRO)->action()->setEnabled(true);
|
|
foreach (IMacroHandler *handler, d->handlers)
|
|
handler->endRecordingMacro(d->currentMacro);
|
|
|
|
d->isRecording = false;
|
|
}
|
|
|
|
void MacroManager::executeLastMacro()
|
|
{
|
|
if (!d->currentMacro)
|
|
return;
|
|
|
|
// make sure the macro doesn't accidentally invoke a macro action
|
|
Core::ActionManager::command(Constants::START_MACRO)->action()->setEnabled(false);
|
|
Core::ActionManager::command(Constants::END_MACRO)->action()->setEnabled(false);
|
|
Core::ActionManager::command(Constants::EXECUTE_LAST_MACRO)->action()->setEnabled(false);
|
|
Core::ActionManager::command(Constants::SAVE_LAST_MACRO)->action()->setEnabled(false);
|
|
|
|
d->executeMacro(d->currentMacro);
|
|
|
|
Core::ActionManager::command(Constants::START_MACRO)->action()->setEnabled(true);
|
|
Core::ActionManager::command(Constants::END_MACRO)->action()->setEnabled(false);
|
|
Core::ActionManager::command(Constants::EXECUTE_LAST_MACRO)->action()->setEnabled(true);
|
|
Core::ActionManager::command(Constants::SAVE_LAST_MACRO)->action()->setEnabled(true);
|
|
}
|
|
|
|
bool MacroManager::executeMacro(const QString &name)
|
|
{
|
|
// Don't execute macro while recording
|
|
if (d->isRecording || !d->macros.contains(name))
|
|
return false;
|
|
|
|
Macro *macro = d->macros.value(name);
|
|
if (!d->executeMacro(macro))
|
|
return false;
|
|
|
|
// Delete anonymous macro
|
|
if (d->currentMacro && d->currentMacro->displayName().isEmpty())
|
|
delete d->currentMacro;
|
|
d->currentMacro = macro;
|
|
|
|
Core::ActionManager::command(Constants::SAVE_LAST_MACRO)->action()->setEnabled(true);
|
|
|
|
return true;
|
|
}
|
|
|
|
void MacroManager::deleteMacro(const QString &name)
|
|
{
|
|
Macro *macro = d->macros.value(name);
|
|
if (macro) {
|
|
QString fileName = macro->fileName();
|
|
d->removeMacro(name);
|
|
QFile::remove(fileName);
|
|
}
|
|
}
|
|
|
|
const QMap<QString,Macro*> &MacroManager::macros()
|
|
{
|
|
return m_instance->d->macros;
|
|
}
|
|
|
|
void MacroManager::registerMacroHandler(IMacroHandler *handler)
|
|
{
|
|
m_instance->d->handlers.prepend(handler);
|
|
}
|
|
|
|
MacroManager *MacroManager::instance()
|
|
{
|
|
return m_instance;
|
|
}
|
|
|
|
void MacroManager::changeMacro(const QString &name, const QString &description)
|
|
{
|
|
if (!d->macros.contains(name))
|
|
return;
|
|
Macro *macro = d->macros.value(name);
|
|
|
|
// Change description
|
|
if (macro->description() != description)
|
|
d->changeMacroDescription(macro, description);
|
|
}
|
|
|
|
void MacroManager::saveLastMacro()
|
|
{
|
|
if (!d->currentMacro->events().isEmpty())
|
|
d->showSaveDialog();
|
|
}
|
|
|
|
QString MacroManager::macrosDirectory()
|
|
{
|
|
const QString path = Core::ICore::userResourcePath("macros").toString();
|
|
if (QFile::exists(path) || QDir().mkpath(path))
|
|
return path;
|
|
return QString();
|
|
}
|
|
|
|
} // Internal
|
|
} // Macros
|