Files
qt-creator/src/plugins/bineditor/bineditorplugin.cpp

546 lines
18 KiB
C++
Raw Normal View History

/****************************************************************************
2008-12-02 12:01:29 +01:00
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
2008-12-02 12:01:29 +01:00
**
** This file is part of Qt Creator.
2008-12-02 12:01:29 +01:00
**
** 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.
2010-12-17 16:01:08 +01:00
**
****************************************************************************/
2008-12-02 14:09:21 +01:00
2008-12-02 12:01:29 +01:00
#include "bineditorplugin.h"
#include "bineditorwidget.h"
2008-12-02 12:01:29 +01:00
#include "bineditorconstants.h"
#include "bineditorservice.h"
2008-12-02 12:01:29 +01:00
#include <coreplugin/icore.h>
#include <QCoreApplication>
#include <QDir>
#include <QFile>
#include <QFileInfo>
#include <QRegExp>
#include <QVariant>
#include <QMenu>
#include <QAction>
#include <QMessageBox>
#include <QHBoxLayout>
#include <QLineEdit>
#include <QRegExpValidator>
#include <QToolBar>
2008-12-02 12:01:29 +01:00
#include <coreplugin/actionmanager/actionmanager.h>
2008-12-02 12:01:29 +01:00
#include <coreplugin/coreconstants.h>
#include <coreplugin/id.h>
#include <coreplugin/editormanager/editormanager.h>
#include <coreplugin/editormanager/ieditor.h>
#include <coreplugin/find/ifindsupport.h>
#include <coreplugin/idocument.h>
#include <extensionsystem/pluginmanager.h>
2008-12-02 12:01:29 +01:00
#include <utils/reloadpromptutils.h>
#include <utils/qtcassert.h>
2008-12-02 12:01:29 +01:00
using namespace Utils;
using namespace Core;
2008-12-02 12:01:29 +01:00
namespace BinEditor {
namespace Internal {
2008-12-02 12:01:29 +01:00
class BinEditorFind : public IFindSupport
2008-12-02 12:01:29 +01:00
{
Q_OBJECT
2008-12-02 12:01:29 +01:00
public:
BinEditorFind(BinEditorWidget *widget)
{
m_widget = widget;
}
2008-12-02 12:01:29 +01:00
bool supportsReplace() const override { return false; }
QString currentFindString() const override { return QString(); }
QString completedFindString() const override { return QString(); }
FindFlags supportedFindFlags() const override
{
return FindBackward | FindCaseSensitively;
}
void resetIncrementalSearch() override
{
m_incrementalStartPos = m_contPos = -1;
m_incrementalWrappedState = false;
}
void highlightAll(const QString &txt, FindFlags findFlags) override
{
m_widget->highlightSearchResults(txt.toLatin1(), textDocumentFlagsForFindFlags(findFlags));
}
void clearHighlights() override
{
m_widget->highlightSearchResults(QByteArray());
}
2008-12-02 12:01:29 +01:00
int find(const QByteArray &pattern, int pos, FindFlags findFlags, bool *wrapped)
{
if (wrapped)
*wrapped = false;
2008-12-02 12:01:29 +01:00
if (pattern.isEmpty()) {
m_widget->setCursorPosition(pos);
2008-12-02 12:01:29 +01:00
return pos;
}
int res = m_widget->find(pattern, pos, textDocumentFlagsForFindFlags(findFlags));
if (res < 0) {
pos = (findFlags & FindBackward) ? -1 : 0;
res = m_widget->find(pattern, pos, textDocumentFlagsForFindFlags(findFlags));
if (res < 0)
return res;
if (wrapped)
*wrapped = true;
}
return res;
2008-12-02 12:01:29 +01:00
}
Result findIncremental(const QString &txt, FindFlags findFlags) override
{
2008-12-02 12:01:29 +01:00
QByteArray pattern = txt.toLatin1();
if (pattern != m_lastPattern)
resetIncrementalSearch(); // Because we don't search for nibbles.
m_lastPattern = pattern;
2008-12-02 12:01:29 +01:00
if (m_incrementalStartPos < 0)
m_incrementalStartPos = m_widget->selectionStart();
if (m_contPos == -1)
m_contPos = m_incrementalStartPos;
bool wrapped;
int found = find(pattern, m_contPos, findFlags, &wrapped);
if (wrapped != m_incrementalWrappedState && (found >= 0)) {
m_incrementalWrappedState = wrapped;
showWrapIndicator(m_widget);
}
Result result;
if (found >= 0) {
result = Found;
m_widget->highlightSearchResults(pattern, textDocumentFlagsForFindFlags(findFlags));
m_contPos = -1;
} else {
if (found == -2) {
result = NotYetFound;
m_contPos +=
findFlags & FindBackward
? -BinEditorWidget::SearchStride : BinEditorWidget::SearchStride;
} else {
result = NotFound;
m_contPos = -1;
m_widget->highlightSearchResults(QByteArray(), 0);
}
}
return result;
2008-12-02 12:01:29 +01:00
}
Result findStep(const QString &txt, FindFlags findFlags) override
{
2008-12-02 12:01:29 +01:00
QByteArray pattern = txt.toLatin1();
bool wasReset = (m_incrementalStartPos < 0);
if (m_contPos == -1) {
m_contPos = m_widget->cursorPosition() + 1;
if (findFlags & FindBackward)
m_contPos = m_widget->selectionStart()-1;
}
bool wrapped;
int found = find(pattern, m_contPos, findFlags, &wrapped);
if (wrapped)
showWrapIndicator(m_widget);
Result result;
if (found >= 0) {
result = Found;
2008-12-02 12:01:29 +01:00
m_incrementalStartPos = found;
m_contPos = -1;
if (wasReset)
m_widget->highlightSearchResults(pattern, textDocumentFlagsForFindFlags(findFlags));
} else if (found == -2) {
result = NotYetFound;
m_contPos += findFlags & FindBackward
? -BinEditorWidget::SearchStride : BinEditorWidget::SearchStride;
} else {
result = NotFound;
m_contPos = -1;
}
return result;
2008-12-02 12:01:29 +01:00
}
2008-12-02 12:01:29 +01:00
private:
BinEditorWidget *m_widget;
qint64 m_incrementalStartPos = -1;
qint64 m_contPos = -1; // Only valid if last result was NotYetFound.
bool m_incrementalWrappedState = false;
QByteArray m_lastPattern;
2008-12-02 12:01:29 +01:00
};
class BinEditorDocument : public IDocument
2008-12-02 12:01:29 +01:00
{
Q_OBJECT
public:
BinEditorDocument(BinEditorWidget *parent) :
IDocument(parent)
2008-12-02 12:01:29 +01:00
{
setId(Core::Constants::K_DEFAULT_BINARY_EDITOR_ID);
setMimeType(QLatin1String(BinEditor::Constants::C_BINEDITOR_MIMETYPE));
m_widget = parent;
EditorService *es = m_widget->editorService();
es->setFetchDataHandler([this](quint64 address) { provideData(address); });
es->setNewRangeRequestHandler([this](quint64 offset) { provideNewRange(offset); });
es->setDataChangedHandler([this](quint64, const QByteArray &) { contentsChanged(); });
}
QByteArray contents() const override
{
return m_widget->contents();
2008-12-02 12:01:29 +01:00
}
bool setContents(const QByteArray &contents) override
{
m_widget->clear();
if (!contents.isEmpty()) {
m_widget->setSizes(0, contents.length(), contents.length());
m_widget->addData(0, contents);
}
return true;
}
ReloadBehavior reloadBehavior(ChangeTrigger state, ChangeType type) const override
{
return type == TypeRemoved ? BehaviorSilent : IDocument::reloadBehavior(state, type);
}
bool save(QString *errorString, const QString &fn, bool autoSave) override
{
QTC_ASSERT(!autoSave, return true); // bineditor does not support autosave - it would be a bit expensive
const FileName fileNameToUse = fn.isEmpty() ? filePath() : FileName::fromString(fn);
if (m_widget->save(errorString, filePath().toString(), fileNameToUse.toString())) {
setFilePath(fileNameToUse);
2008-12-02 12:01:29 +01:00
return true;
} else {
return false;
2008-12-02 12:01:29 +01:00
}
}
OpenResult open(QString *errorString, const QString &fileName,
const QString &realFileName) override
{
QTC_CHECK(fileName == realFileName); // The bineditor can do no autosaving
return openImpl(errorString, fileName);
}
OpenResult openImpl(QString *errorString, const QString &fileName, quint64 offset = 0)
{
2008-12-02 12:01:29 +01:00
QFile file(fileName);
if (file.open(QIODevice::ReadOnly)) {
file.close();
quint64 size = static_cast<quint64>(file.size());
if (size == 0) {
QString msg = tr("The Binary Editor cannot open empty files.");
if (errorString)
*errorString = msg;
else
QMessageBox::critical(ICore::mainWindow(), tr("File Error"), msg);
return OpenResult::CannotHandle;
}
if (size / 16 >= qint64(1) << 31) {
// The limit is 2^31 lines (due to QText* interfaces) * 16 bytes per line.
QString msg = tr("The file is too big for the Binary Editor (max. 32GB).");
if (errorString)
*errorString = msg;
else
QMessageBox::critical(ICore::mainWindow(), tr("File Error"), msg);
return OpenResult::CannotHandle;
}
if (offset >= size)
return OpenResult::CannotHandle;
setFilePath(FileName::fromString(fileName));
m_widget->setSizes(offset, file.size());
return OpenResult::Success;
2008-12-02 12:01:29 +01:00
}
QString errStr = tr("Cannot open %1: %2").arg(
QDir::toNativeSeparators(fileName), file.errorString());
if (errorString)
*errorString = errStr;
else
QMessageBox::critical(ICore::mainWindow(), tr("File Error"), errStr);
return OpenResult::ReadError;
2008-12-02 12:01:29 +01:00
}
void provideData(quint64 address)
{
const FileName fn = filePath();
if (fn.isEmpty())
return;
QFile file(fn.toString());
if (file.open(QIODevice::ReadOnly)) {
int blockSize = m_widget->dataBlockSize();
file.seek(address);
QByteArray data = file.read(blockSize);
file.close();
const int dataSize = data.size();
if (dataSize != blockSize)
data += QByteArray(blockSize - dataSize, 0);
m_widget->addData(address, data);
} else {
QMessageBox::critical(ICore::mainWindow(), tr("File Error"),
tr("Cannot open %1: %2").arg(
fn.toUserOutput(), file.errorString()));
}
}
void provideNewRange(quint64 offset)
{
if (filePath().exists())
openImpl(0, filePath().toString(), offset);
}
public:
bool isModified() const override
{
return isTemporary()/*e.g. memory view*/ ? false
: m_widget->isModified();
}
bool isFileReadOnly() const override {
const FileName fn = filePath();
if (fn.isEmpty())
2011-02-25 13:21:54 +01:00
return false;
return !fn.toFileInfo().isWritable();
2008-12-02 12:01:29 +01:00
}
bool isSaveAsAllowed() const override { return true; }
2008-12-02 12:01:29 +01:00
bool reload(QString *errorString, ReloadFlag flag, ChangeType type) override
{
if (flag == FlagIgnore)
return true;
if (type == TypePermissions) {
2008-12-02 12:01:29 +01:00
emit changed();
} else {
emit aboutToReload();
int cPos = m_widget->cursorPosition();
m_widget->clear();
const bool success = (openImpl(errorString, filePath().toString()) == OpenResult::Success);
m_widget->setCursorPosition(cPos);
emit reloadFinished(success);
return success;
2008-12-02 12:01:29 +01:00
}
return true;
2008-12-02 12:01:29 +01:00
}
private:
BinEditorWidget *m_widget;
2008-12-02 12:01:29 +01:00
};
class BinEditor : public IEditor
2008-12-02 12:01:29 +01:00
{
Q_OBJECT
public:
BinEditor(BinEditorWidget *widget)
{
setWidget(widget);
m_file = new BinEditorDocument(widget);
m_addressEdit = new QLineEdit;
auto addressValidator = new QRegExpValidator(QRegExp("[0-9a-fA-F]{1,16}"), m_addressEdit);
m_addressEdit->setValidator(addressValidator);
2008-12-02 12:01:29 +01:00
auto l = new QHBoxLayout;
auto w = new QWidget;
2008-12-02 12:01:29 +01:00
l->setMargin(0);
l->setContentsMargins(0, 0, 5, 0);
l->addStretch(1);
l->addWidget(m_addressEdit);
2008-12-02 12:01:29 +01:00
w->setLayout(l);
m_toolBar = new QToolBar;
m_toolBar->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
m_toolBar->addWidget(w);
widget->setEditor(this);
connect(widget, &BinEditorWidget::cursorPositionChanged,
this, &BinEditor::updateCursorPosition);
connect(m_addressEdit, &QLineEdit::editingFinished,
this, &BinEditor::jumpToAddress);
connect(widget, &BinEditorWidget::modificationChanged,
m_file, &IDocument::changed);
updateCursorPosition(widget->cursorPosition());
2008-12-02 12:01:29 +01:00
}
~BinEditor() override
{
delete m_widget;
}
2008-12-02 12:01:29 +01:00
IDocument *document() override { return m_file; }
2008-12-02 12:01:29 +01:00
QWidget *toolBar() override { return m_toolBar; }
2008-12-02 12:01:29 +01:00
private:
void updateCursorPosition(qint64 position) {
m_addressEdit->setText(QString::number(editorWidget()->baseAddress() + position, 16));
}
void jumpToAddress() {
editorWidget()->jumpToAddress(m_addressEdit->text().toULongLong(0, 16));
updateCursorPosition(editorWidget()->cursorPosition());
}
BinEditorWidget *editorWidget() const
{
QTC_ASSERT(qobject_cast<BinEditorWidget *>(m_widget.data()), return 0);
return static_cast<BinEditorWidget *>(m_widget.data());
2008-12-02 12:01:29 +01:00
}
private:
BinEditorDocument *m_file;
2008-12-02 12:01:29 +01:00
QToolBar *m_toolBar;
QLineEdit *m_addressEdit;
2008-12-02 12:01:29 +01:00
};
///////////////////////////////// BinEditorPluginPrivate //////////////////////////////////
class BinEditorPluginPrivate : public QObject
{
public:
BinEditorPluginPrivate();
~BinEditorPluginPrivate();
QAction *m_undoAction = nullptr;
QAction *m_redoAction = nullptr;
QAction *m_copyAction = nullptr;
QAction *m_selectAllAction = nullptr;
2008-12-02 12:01:29 +01:00
FactoryServiceImpl m_factoryService;
BinEditorFactory m_editorFactory;
};
BinEditorPluginPrivate::BinEditorPluginPrivate()
{
ExtensionSystem::PluginManager::addObject(&m_factoryService);
ExtensionSystem::PluginManager::addObject(&m_editorFactory);
m_undoAction = new QAction(tr("&Undo"), this);
m_redoAction = new QAction(tr("&Redo"), this);
m_copyAction = new QAction(this);
m_selectAllAction = new QAction(this);
Context context;
context.add(Core::Constants::K_DEFAULT_BINARY_EDITOR_ID);
context.add(Constants::C_BINEDITOR);
ActionManager::registerAction(m_undoAction, Core::Constants::UNDO, context);
ActionManager::registerAction(m_redoAction, Core::Constants::REDO, context);
ActionManager::registerAction(m_copyAction, Core::Constants::COPY, context);
ActionManager::registerAction(m_selectAllAction, Core::Constants::SELECTALL, context);
}
BinEditorPluginPrivate::~BinEditorPluginPrivate()
{
ExtensionSystem::PluginManager::removeObject(&m_editorFactory);
ExtensionSystem::PluginManager::removeObject(&m_factoryService);
}
static BinEditorPluginPrivate *dd = nullptr;
2008-12-02 12:01:29 +01:00
///////////////////////////////// BinEditorFactory //////////////////////////////////
BinEditorFactory::BinEditorFactory()
2008-12-02 12:01:29 +01:00
{
setId(Core::Constants::K_DEFAULT_BINARY_EDITOR_ID);
setDisplayName(QCoreApplication::translate("OpenWith::Editors", Constants::C_BINEDITOR_DISPLAY_NAME));
addMimeType(Constants::C_BINEDITOR_MIMETYPE);
2008-12-02 12:01:29 +01:00
}
IEditor *BinEditorFactory::createEditor()
2008-12-02 12:01:29 +01:00
{
auto widget = new BinEditorWidget();
auto editor = new BinEditor(widget);
connect(dd->m_undoAction, &QAction::triggered, widget, &BinEditorWidget::undo);
connect(dd->m_redoAction, &QAction::triggered, widget, &BinEditorWidget::redo);
connect(dd->m_copyAction, &QAction::triggered, widget, &BinEditorWidget::copy);
connect(dd->m_selectAllAction, &QAction::triggered, widget, &BinEditorWidget::selectAll);
auto updateActions = [widget] {
dd->m_selectAllAction->setEnabled(true);
dd->m_undoAction->setEnabled(widget->isUndoAvailable());
dd->m_redoAction->setEnabled(widget->isRedoAvailable());
};
connect(widget, &BinEditorWidget::undoAvailable, widget, updateActions);
connect(widget, &BinEditorWidget::redoAvailable, widget, updateActions);
auto aggregate = new Aggregation::Aggregate;
auto binEditorFind = new BinEditorFind(widget);
aggregate->add(binEditorFind);
aggregate->add(widget);
return editor;
2008-12-02 12:01:29 +01:00
}
///////////////////////////////// BinEditor Services //////////////////////////////////
EditorService *FactoryServiceImpl::createEditorService(const QString &title0, bool wantsEditor)
{
BinEditorWidget *widget = nullptr;
if (wantsEditor) {
QString title = title0;
IEditor *editor = EditorManager::openEditorWithContents(
Core::Constants::K_DEFAULT_BINARY_EDITOR_ID, &title);
if (!editor)
return 0;
widget = qobject_cast<BinEditorWidget *>(editor->widget());
widget->setEditor(editor);
} else {
widget = new BinEditorWidget;
widget->setWindowTitle(title0);
}
return widget->editorService();
}
2008-12-02 12:01:29 +01:00
///////////////////////////////// BinEditorPlugin //////////////////////////////////
BinEditorPlugin::~BinEditorPlugin()
{
delete dd;
dd = nullptr;
2008-12-02 12:01:29 +01:00
}
bool BinEditorPlugin::initialize(const QStringList &arguments, QString *errorMessage)
2008-12-02 12:01:29 +01:00
{
Q_UNUSED(arguments)
Q_UNUSED(errorMessage)
dd = new BinEditorPluginPrivate;
2008-12-02 12:01:29 +01:00
return true;
}
} // namespace Internal
} // namespace BinEditor
2008-12-02 12:01:29 +01:00
#include "bineditorplugin.moc"