Files
qt-creator/tests/manual/fakevim/main.cpp
Marcus Tillmanns 015d12ccf3 FakeVim: Accept suggestion with Tab Key
* Changed signals to callbacks as only one receiver was ever added
* Added "tabPressedInInsertMode" callback to allow accepting
  a suggestion with the Tab Key

Fixes: QTCREATORBUG-28830
Change-Id: Ie70ba595b8802b6100fff495164d8e0471b1354c
Reviewed-by: hjk <hjk@qt.io>
2023-05-02 06:53:54 +00:00

233 lines
6.6 KiB
C++

// Copyright (C) 2016 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include "fakevimhandler.h"
#include <QApplication>
#include <QFontMetrics>
#include <QMainWindow>
#include <QMessageBox>
#include <QPainter>
#include <QPlainTextEdit>
#include <QRegularExpression>
#include <QStatusBar>
#include <QTextEdit>
using namespace FakeVim::Internal;
/**
* Simple editor widget.
* @tparam TextEdit QTextEdit or QPlainTextEdit as base class
*/
template <typename TextEdit>
class Editor : public TextEdit
{
public:
Editor()
{
TextEdit::setCursorWidth(0);
TextEdit::setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
}
void paintEvent(QPaintEvent *e)
{
TextEdit::paintEvent(e);
// Draw text cursor.
QRect rect = TextEdit::cursorRect();
if ( e->rect().contains(rect) ) {
QPainter painter(TextEdit::viewport());
if ( TextEdit::overwriteMode() ) {
QFontMetrics fm(TextEdit::font());
rect.setWidth(fm.horizontalAdvance('m'));
painter.setPen(Qt::NoPen);
painter.setBrush(TextEdit::palette().color(QPalette::Base));
painter.setCompositionMode(QPainter::CompositionMode_Difference);
} else {
rect.setWidth(TextEdit::cursorWidth());
painter.setPen(TextEdit::palette().color(QPalette::Text));
}
painter.drawRect(rect);
}
}
};
static void highlightMatches(QWidget *widget, const QString &pattern)
{
auto ed = qobject_cast<QTextEdit *>(widget);
if (!ed)
return;
// Clear previous highlights.
ed->selectAll();
QTextCursor cur = ed->textCursor();
QTextCharFormat fmt = cur.charFormat();
fmt.setBackground(Qt::transparent);
cur.setCharFormat(fmt);
// Highlight matches.
QTextDocument *doc = ed->document();
const QRegularExpression re(pattern);
cur = doc->find(re);
int a = cur.position();
while ( !cur.isNull() ) {
if ( cur.hasSelection() ) {
fmt.setBackground(Qt::yellow);
cur.setCharFormat(fmt);
} else {
cur.movePosition(QTextCursor::NextCharacter);
}
cur = doc->find(re, cur);
int b = cur.position();
if (a == b) {
cur.movePosition(QTextCursor::NextCharacter);
cur = doc->find(re, cur);
b = cur.position();
if (a == b) break;
}
a = b;
}
}
class StatusData
{
public:
void setStatusMessage(const QString &msg, int pos)
{
m_statusMessage = pos == -1 ? msg : msg.left(pos) + QChar(10073) + msg.mid(pos);
}
void setStatusInfo(const QString &info)
{
m_statusData = info;
}
QString currentStatusLine() const
{
const int slack = 80 - m_statusMessage.size() - m_statusData.size();
return m_statusMessage + QString(slack, ' ') + m_statusData;
}
private:
QString m_statusMessage;
QString m_statusData;
};
static QWidget *createEditorWidget(bool usePlainTextEdit)
{
QWidget *editor = 0;
if (usePlainTextEdit)
editor = new Editor<QPlainTextEdit>;
else
editor = new Editor<QTextEdit>;
editor->setObjectName("Editor");
editor->setFocus();
return editor;
}
static void initHandler(FakeVimHandler &handler)
{
// Set some Vim options.
handler.handleCommand("set expandtab");
handler.handleCommand("set shiftwidth=8");
handler.handleCommand("set tabstop=16");
handler.handleCommand("set autoindent");
// Try to source file "fakevimrc" from current directory.
handler.handleCommand("source fakevimrc");
handler.installEventFilter();
handler.setupWidget();
}
static void initMainWindow(QMainWindow &mainWindow, QWidget *centralWidget, const QString &title)
{
mainWindow.setWindowTitle(QString("FakeVim (%1)").arg(title));
mainWindow.setCentralWidget(centralWidget);
mainWindow.resize(600, 650);
mainWindow.move(0, 0);
mainWindow.show();
// Set monospace font for editor and status bar.
QFont font = QApplication::font();
font.setFamily("Monospace");
centralWidget->setFont(font);
mainWindow.statusBar()->setFont(font);
}
void readFile(FakeVimHandler &handler, const QString &editFileName)
{
handler.handleCommand("r " + editFileName);
}
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
QStringList args = app.arguments();
// If first argument is present use QPlainTextEdit instead on QTextEdit;
bool usePlainTextEdit = args.size() > 1;
// Second argument is path to file to edit.
const QString editFileName = args.value(2, "/usr/share/vim/vim73/tutor/tutor");
// Create editor widget.
QWidget *editor = createEditorWidget(usePlainTextEdit);
// Create main window.
QMainWindow mainWindow;
initMainWindow(mainWindow, editor, usePlainTextEdit ? "QPlainTextEdit" : "QTextEdit");
// Keep track of status line related data.
StatusData statusData;
// Create FakeVimHandler instance which will emulate Vim behavior in editor widget.
FakeVimHandler handler(editor, nullptr);
handler.commandBufferChanged.set([&](const QString &msg, int cursorPos, int, int) {
statusData.setStatusMessage(msg, cursorPos);
mainWindow.statusBar()->showMessage(statusData.currentStatusLine());
});
handler.selectionChanged.set([&handler](const QList<QTextEdit::ExtraSelection> &s) {
QWidget *widget = handler.widget();
if (auto ed = qobject_cast<QPlainTextEdit *>(widget))
ed->setExtraSelections(s);
else if (auto ed = qobject_cast<QTextEdit *>(widget))
ed->setExtraSelections(s);
});
handler.extraInformationChanged.set([&](const QString &info) {
statusData.setStatusInfo(info);
mainWindow.statusBar()->showMessage(statusData.currentStatusLine());
});
handler.statusDataChanged.set([&](const QString &info) {
statusData.setStatusInfo(info);
mainWindow.statusBar()->showMessage(statusData.currentStatusLine());
});
handler.highlightMatches.set(
[&](const QString &needle) { highlightMatches(handler.widget(), needle); });
handler.handleExCommandRequested.set([](bool *handled, const ExCommand &cmd) {
if (cmd.matches("q", "quit") || cmd.matches("qa", "qall")) {
QApplication::quit();
*handled = true;
} else {
*handled = false;
}
});
// Initialize FakeVimHandler.
initHandler(handler);
// Read file content to editor.
readFile(handler, editFileName);
return app.exec();
}