Files
qt-creator/src/plugins/fakevim/handler.cpp

721 lines
22 KiB
C++
Raw Normal View History

/***************************************************************************
**
** This file is part of Qt Creator
**
** Copyright (c) 2008 Nokia Corporation and/or its subsidiary(-ies).
**
** Contact: Qt Software Information (qt-info@nokia.com)
**
**
** Non-Open Source Usage
**
** Licensees may use this file in accordance with the Qt Beta Version
** License Agreement, Agreement version 2.2 provided with the Software or,
** alternatively, in accordance with the terms contained in a written
** agreement between you and Nokia.
**
** GNU General Public License Usage
**
** Alternatively, this file may be used under the terms of the GNU General
** Public License versions 2.0 or 3.0 as published by the Free Software
** Foundation and appearing in the file LICENSE.GPL included in the packaging
** of this file. Please review the following information to ensure GNU
** General Public Licensing requirements will be met:
**
** http://www.fsf.org/licensing/licenses/info/GPLv2.html and
** http://www.gnu.org/copyleft/gpl.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights. These rights are described in the Nokia Qt GPL Exception
** version 1.3, included in the file GPL_EXCEPTION.txt in this package.
**
***************************************************************************/
#include "handler.h"
2008-12-19 16:20:39 +01:00
#include <QtCore/QDebug>
#include <QtCore/QObject>
#include <QtCore/QRegExp>
#include <QtCore/QStack>
#include <QtGui/QKeyEvent>
#include <QtGui/QLineEdit>
#include <QtGui/QPlainTextEdit>
#include <QtGui/QScrollBar>
#include <QtGui/QTextBlock>
#include <QtGui/QTextCursor>
#include <QtGui/QTextEdit>
using namespace FakeVim::Internal;
#define StartOfLine QTextCursor::StartOfLine
#define EndOfLine QTextCursor::EndOfLine
#define MoveAnchor QTextCursor::MoveAnchor
#define KeepAnchor QTextCursor::KeepAnchor
#define Up QTextCursor::Up
#define Down QTextCursor::Down
#define Right QTextCursor::Right
#define Left QTextCursor::Left
///////////////////////////////////////////////////////////////////////
//
// FakeVimHandler
//
///////////////////////////////////////////////////////////////////////
const int ParagraphSeparator = 0x00002029;
using namespace Qt;
2008-12-19 16:20:39 +01:00
enum Mode
{
InsertMode,
CommandMode,
ExMode
};
2008-12-19 16:20:39 +01:00
enum SubMode
{
NoSubMode,
RegisterSubMode,
ChangeSubMode,
DeleteSubMode,
ZSubMode
};
2008-12-26 00:18:03 +01:00
enum SubSubMode
{
NoSubSubMode,
FtSubSubMode, // used for f, F, t, T
};
class FakeVimHandler::Private
{
public:
Private(FakeVimHandler *parent);
bool eventFilter(QObject *ob, QEvent *ev);
static int shift(int key) { return key + 32; }
static int control(int key) { return key + 256; }
void init();
void handleKey(int key, const QString &text);
void handleInsertMode(int key, const QString &text);
void handleCommandMode(int key, const QString &text);
void handleRegisterMode(int key, const QString &text);
void handleExMode(int key, const QString &text);
void finishMovement();
void updateCommandBuffer();
2008-12-26 00:18:03 +01:00
void search(const QString &needle, bool forward);
void showMessage(const QString &msg);
2008-12-25 22:41:09 +01:00
int mvCount() const { return m_mvcount.isEmpty() ? 1 : m_mvcount.toInt(); }
int opCount() const { return m_opcount.isEmpty() ? 1 : m_opcount.toInt(); }
int count() const { return mvCount() * opCount(); }
int leftDist() const { return m_tc.position() - m_tc.block().position(); }
int rightDist() const { return m_tc.block().length() - leftDist() - 1; }
bool atEol() const { return m_tc.atBlockEnd() && m_tc.block().length()>1; }
2008-12-19 16:20:39 +01:00
int lastPositionInDocument() const;
// all zero-based counting
int cursorLineOnScreen() const;
int linesOnScreen() const;
int cursorLineInDocument() const;
void scrollToLineInDocument(int line);
2008-12-19 14:35:57 +01:00
void moveToFirstNonBlankOnLine();
2008-12-26 00:18:03 +01:00
void moveToNextWord(bool simple);
void moveToWordBoundary(bool simple, bool forward);
void handleFfTt(int key);
FakeVimHandler *q;
Mode m_mode;
SubMode m_submode;
2008-12-26 00:18:03 +01:00
SubSubMode m_subsubmode;
int m_subsubdata;
QString m_input;
QPlainTextEdit *m_editor;
QTextCursor m_tc;
QHash<int, QString> m_registers;
int m_register;
2008-12-25 22:41:09 +01:00
QString m_mvcount;
QString m_opcount;
QStack<QString> m_undoStack;
QStack<QString> m_redoStack;
bool m_fakeEnd;
int m_commandCode; // ?, /, : ...
QString m_commandBuffer;
bool m_lastSearchForward;
QString m_lastSearchString;
};
FakeVimHandler::Private::Private(FakeVimHandler *parent)
{
q = parent;
m_mode = CommandMode;
2008-12-26 00:18:03 +01:00
m_submode = NoSubMode;
m_subsubmode = NoSubSubMode;
2008-12-25 23:19:08 +01:00
m_commandCode = 0;
m_fakeEnd = false;
m_lastSearchForward = true;
m_register = '"';
}
bool FakeVimHandler::Private::eventFilter(QObject *ob, QEvent *ev)
{
QKeyEvent *keyEvent = static_cast<QKeyEvent *>(ev);
int key = keyEvent->key();
if (key == Key_Shift || key == Key_Alt || key == Key_Control
|| key == Key_Alt || key == Key_AltGr || key == Key_Meta)
return false;
//qDebug() << "KEY: " << key << Qt::ShiftModifier;
// Fake "End of line"
m_editor = qobject_cast<QPlainTextEdit *>(ob);
if (!m_editor)
return false;
m_tc = m_editor->textCursor();
if (m_fakeEnd) {
//m_fakeEnd = false;
m_tc.movePosition(Right, MoveAnchor, 1);
qDebug() << "Unfake EOL";
}
if (key >= Key_A && key <= Key_Z
&& (keyEvent->modifiers() & Qt::ShiftModifier) == 0)
key += 32;
if ((keyEvent->modifiers() & Qt::ControlModifier) != 0)
key += 256;
handleKey(key, keyEvent->text());
// We fake vi-style end-of-line behaviour
m_fakeEnd = atEol() && m_mode == CommandMode;
//qDebug() << "POS: " << m_tc.position()
// << " BLOCK LEN: " << m_tc.block().length()
// << " LEFT: " << leftDist() << " RIGHT: " << rightDist();
if (m_fakeEnd) {
m_tc.movePosition(Left, MoveAnchor, 1);
qDebug() << "Fake EOL";
}
//qDebug() << "POS: " << m_tc.position()
// << " BLOCK LEN: " << m_tc.block().length()
// << " LEFT: " << leftDist() << " RIGHT: " << rightDist();
m_editor->setTextCursor(m_tc);
m_editor->ensureCursorVisible();
return true;
}
void FakeVimHandler::Private::handleKey(int key, const QString &text)
{
if (m_mode == InsertMode)
handleInsertMode(key, text);
else if (m_mode == CommandMode)
handleCommandMode(key, text);
else if (m_mode == ExMode)
handleExMode(key, text);
}
void FakeVimHandler::Private::finishMovement()
{
if (m_submode == ChangeSubMode) {
m_registers[m_register] = m_tc.selectedText();
m_tc.removeSelectedText();
m_mode = InsertMode;
m_submode = NoSubMode;
} else if (m_submode == DeleteSubMode) {
m_registers[m_register] = m_tc.selectedText();
m_tc.removeSelectedText();
m_submode = NoSubMode;
if (atEol())
m_tc.movePosition(Left, MoveAnchor, 1);
}
2008-12-25 22:41:09 +01:00
m_mvcount.clear();
m_opcount.clear();
m_register = '"';
m_tc.clearSelection();
updateCommandBuffer();
}
void FakeVimHandler::Private::updateCommandBuffer()
{
QString msg = QChar(m_commandCode ? m_commandCode : ' ') + m_commandBuffer;
emit q->commandBufferChanged(msg);
}
void FakeVimHandler::Private::showMessage(const QString &msg)
{
emit q->commandBufferChanged(msg);
}
void FakeVimHandler::Private::handleCommandMode(int key, const QString &text)
{
Q_UNUSED(text)
//qDebug() << "-> MODE: " << m_mode << " KEY: " << key;
if (m_submode == RegisterSubMode) {
m_register = key;
m_submode = NoSubMode;
} else if (m_submode == ChangeSubMode && key == 'c') {
m_tc.movePosition(StartOfLine, MoveAnchor);
m_tc.movePosition(Down, KeepAnchor, count());
m_registers[m_register] = m_tc.selectedText();
finishMovement();
} else if (m_submode == DeleteSubMode && key == 'd') {
m_tc.movePosition(StartOfLine, MoveAnchor);
m_tc.movePosition(Down, KeepAnchor, count());
m_registers[m_register] = m_tc.selectedText();
finishMovement();
2008-12-19 16:20:39 +01:00
} else if (m_submode == ZSubMode) {
if (key == Key_Return) {
2008-12-25 13:20:09 +01:00
// cursor line to top of window, cursor on first non-blank
scrollToLineInDocument(cursorLineInDocument());
2008-12-19 16:20:39 +01:00
moveToFirstNonBlankOnLine();
finishMovement();
2008-12-25 13:20:09 +01:00
} else {
2008-12-19 16:20:39 +01:00
qDebug() << "Ignored z + " << key << text;
}
m_submode = NoSubMode;
2008-12-26 00:18:03 +01:00
} else if (m_subsubmode == FtSubSubMode) {
handleFfTt(key);
m_subsubmode = NoSubSubMode;
finishMovement();
} else if (key >= '0' && key <= '9') {
2008-12-25 22:41:09 +01:00
if (key == '0' && m_mvcount.isEmpty()) {
m_tc.movePosition(StartOfLine, KeepAnchor);
finishMovement();
} else {
2008-12-25 22:41:09 +01:00
m_mvcount.append(QChar(key));
}
} else if (key == ':' || key == '/' || key == '?') {
m_commandCode = key;
m_mode = ExMode;
updateCommandBuffer();
} else if (key == '|') {
m_tc.movePosition(StartOfLine, KeepAnchor);
m_tc.movePosition(Right, KeepAnchor, qMin(count(), rightDist()) - 1);
finishMovement();
} else if (key == '"') {
m_submode = RegisterSubMode;
} else if (key == Key_Return) {
m_tc.movePosition(StartOfLine);
m_tc.movePosition(Down);
} else if (key == Key_Home) {
m_tc.movePosition(StartOfLine, KeepAnchor);
finishMovement();
} else if (key == '$' || key == Key_End) {
m_tc.movePosition(EndOfLine, KeepAnchor);
finishMovement();
} else if (key == 'A') {
m_tc.movePosition(EndOfLine, MoveAnchor);
m_mode = InsertMode;
2008-12-25 19:50:14 +01:00
} else if (key == 'b') {
2008-12-26 00:18:03 +01:00
moveToWordBoundary(false, false);
2008-12-25 19:50:14 +01:00
finishMovement();
} else if (key == 'B') {
2008-12-26 00:18:03 +01:00
moveToWordBoundary(true, false);
2008-12-25 19:50:14 +01:00
finishMovement();
} else if (key == 'c') {
m_submode = ChangeSubMode;
} else if (key == 'C') {
m_submode = ChangeSubMode;
m_tc.movePosition(EndOfLine, KeepAnchor);
finishMovement();
} else if (key == 'd') {
if (atEol())
m_tc.movePosition(Left, MoveAnchor, 1);
2008-12-25 22:41:09 +01:00
m_opcount = m_mvcount;
m_mvcount.clear();
m_submode = DeleteSubMode;
} else if (key == 'D') {
m_submode = DeleteSubMode;
2008-12-26 00:45:40 +01:00
m_tc.movePosition(Down, KeepAnchor, qMax(count() - 1, 0));
m_tc.movePosition(Right, KeepAnchor, rightDist());
finishMovement();
2008-12-25 19:50:14 +01:00
} else if (key == 'e') {
2008-12-26 00:18:03 +01:00
moveToWordBoundary(false, true);
2008-12-25 19:50:14 +01:00
finishMovement();
} else if (key == 'E') {
2008-12-26 00:18:03 +01:00
moveToWordBoundary(true, true);
2008-12-25 19:50:14 +01:00
finishMovement();
2008-12-26 00:18:03 +01:00
} else if (key == 'f' || key == 'F') {
m_subsubmode = FtSubSubMode;
m_subsubdata = key;
} else if (key == 'h' || key == Key_Left) {
int n = qMin(count(), leftDist());
if (m_fakeEnd && m_tc.block().length() > 1)
++n;
m_tc.movePosition(Left, KeepAnchor, n);
finishMovement();
2008-12-19 14:35:57 +01:00
} else if (key == 'H') {
2008-12-19 15:00:06 +01:00
m_tc = m_editor->cursorForPosition(QPoint(0, 0));
2008-12-19 14:35:57 +01:00
m_tc.movePosition(Down, KeepAnchor, qMax(count() - 1, 0));
moveToFirstNonBlankOnLine();
finishMovement();
} else if (key == 'i') {
m_mode = InsertMode;
} else if (key == 'j' || key == Key_Down) {
m_tc.movePosition(Down, KeepAnchor, count());
finishMovement();
} else if (key == 'k' || key == Key_Up) {
m_tc.movePosition(Up, KeepAnchor, count());
finishMovement();
} else if (key == 'l' || key == Key_Right) {
m_tc.movePosition(Right, KeepAnchor, qMin(count(), rightDist()));
finishMovement();
2008-12-19 14:43:14 +01:00
} else if (key == 'L') {
2008-12-19 15:00:06 +01:00
int heigth = m_editor->height();
m_tc = m_editor->cursorForPosition(QPoint(0, heigth));
2008-12-19 14:43:14 +01:00
m_tc.movePosition(Up, KeepAnchor, qMax(count(), 1));
moveToFirstNonBlankOnLine();
finishMovement();
2008-12-19 15:00:06 +01:00
} else if (key == 'M') {
int heigth = m_editor->height();
m_tc = m_editor->cursorForPosition(QPoint(0, heigth / 2));
moveToFirstNonBlankOnLine();
finishMovement();
} else if (key == 'n') {
search(m_lastSearchString, m_lastSearchForward);
} else if (key == 'N') {
search(m_lastSearchString, !m_lastSearchForward);
} else if (key == 'p') {
QString text = m_registers[m_register];
int n = text.count(QChar(ParagraphSeparator));
if (n > 0) {
m_tc.movePosition(Down);
m_tc.movePosition(StartOfLine);
m_tc.insertText(text);
m_tc.movePosition(Up, MoveAnchor, n);
} else {
m_tc.movePosition(Right);
m_tc.insertText(text);
m_tc.movePosition(Left);
}
} else if (key == control('r')) {
m_editor->redo();
2008-12-26 00:18:03 +01:00
} else if (key == 't' || key == 'T') {
m_subsubmode = FtSubSubMode;
m_subsubdata = key;
} else if (key == 'u') {
m_editor->undo();
} else if (key == 'w') {
2008-12-26 00:18:03 +01:00
moveToNextWord(false);
finishMovement();
} else if (key == 'W') {
2008-12-26 00:18:03 +01:00
moveToNextWord(true);
finishMovement();
} else if (key == 'x') { // = "dl"
if (atEol())
m_tc.movePosition(Left, MoveAnchor, 1);
m_submode = DeleteSubMode;
m_tc.movePosition(Right, KeepAnchor, qMin(count(), rightDist()));
finishMovement();
} else if (key == 'X') {
if (leftDist() > 0) {
m_tc.movePosition(Left, KeepAnchor, qMin(count(), leftDist()));
m_tc.deleteChar();
}
finishMovement();
2008-12-19 16:20:39 +01:00
} else if (key == 'z') {
m_submode = ZSubMode;
2008-12-25 22:29:22 +01:00
} else if (key == '~' && !atEol()) {
2008-12-25 22:22:41 +01:00
m_tc.movePosition(Right, KeepAnchor, qMin(count(), rightDist()));
QString str = m_tc.selectedText();
for (int i = str.size(); --i >= 0; ) {
QChar c = str.at(i);
str[i] = c.isUpper() ? c.toLower() : c.toUpper();
}
m_tc.deleteChar();
m_tc.insertText(str);
} else if (key == Key_PageDown || key == control('f')) {
m_tc.movePosition(Down, KeepAnchor, count() * (linesOnScreen() - 2));
finishMovement();
} else if (key == Key_PageUp || key == control('b')) {
m_tc.movePosition(Up, KeepAnchor, count() * (linesOnScreen() - 2));
finishMovement();
} else if (key == Key_Backspace) {
m_tc.deletePreviousChar();
} else if (key == Key_Delete) {
m_tc.deleteChar();
2008-12-25 22:57:21 +01:00
} else if (key == Key_Escape) {
// harmless
} else {
2008-12-19 16:20:39 +01:00
qDebug() << "Ignored" << key << text;
}
}
void FakeVimHandler::Private::handleInsertMode(int key, const QString &text)
{
if (key == Key_Escape) {
m_mode = CommandMode;
m_tc.movePosition(Left, KeepAnchor, qMin(1, leftDist()));
} else if (key == Key_Left) {
m_tc.movePosition(Left, MoveAnchor, 1);
} else if (key == Key_Down) {
m_tc.movePosition(Down, MoveAnchor, 1);
} else if (key == Key_Up) {
m_tc.movePosition(Up, MoveAnchor, 1);
} else if (key == Key_Right) {
m_tc.movePosition(Right, MoveAnchor, 1);
} else if (key == Key_Return) {
m_tc.insertBlock();
} else if (key == Key_Backspace) {
m_tc.deletePreviousChar();
} else if (key == Key_Delete) {
m_tc.deleteChar();
} else if (key == Key_PageDown || key == control('f')) {
m_tc.movePosition(Down, KeepAnchor, count() * (linesOnScreen() - 2));
finishMovement();
} else if (key == Key_PageUp || key == control('b')) {
m_tc.movePosition(Up, KeepAnchor, count() * (linesOnScreen() - 2));
finishMovement();
} else {
m_tc.insertText(text);
}
}
void FakeVimHandler::Private::handleExMode(int key, const QString &text)
{
Q_UNUSED(text)
if (key == Key_Escape) {
m_commandBuffer.clear();
m_commandCode = 0;
m_mode = CommandMode;
} else if (key == Key_Backspace) {
if (m_commandBuffer.isEmpty())
m_mode = CommandMode;
else
m_commandBuffer.chop(1);
} else if (key == Key_Return && m_commandCode == ':') {
static QRegExp reGoto("^(\\d+)$");
if (reGoto.indexIn(m_commandBuffer) != -1) {
int n = reGoto.cap(1).toInt();
m_tc.setPosition(m_tc.block().document()
->findBlockByNumber(n - 1).position());
} else if (m_commandBuffer == "q!" || m_commandBuffer == "q") {
q->quitRequested(m_editor);
}
m_commandBuffer.clear();
m_mode = CommandMode;
} else if (key == Key_Return
&& (m_commandCode == '/' || m_commandCode == '?')) {
m_lastSearchForward = (m_commandCode == '/');
m_lastSearchString = m_commandBuffer;
search(m_lastSearchString, m_lastSearchForward);
m_commandBuffer.clear();
m_commandCode = 0;
m_mode = CommandMode;
} else {
m_commandBuffer += QChar(key);
}
updateCommandBuffer();
}
2008-12-26 00:18:03 +01:00
void FakeVimHandler::Private::search(const QString &needle, bool forward)
{
//qDebug() << "NEEDLE " << needle << "BACKWARDS" << backwards;
2008-12-25 13:20:09 +01:00
QTextCursor orig = m_tc;
QTextDocument::FindFlags flags;
2008-12-26 00:18:03 +01:00
if (!forward)
flags = QTextDocument::FindBackward;
2008-12-26 00:18:03 +01:00
if (forward)
2008-12-25 13:20:09 +01:00
m_tc.movePosition(Right, MoveAnchor, 1);
m_editor->setTextCursor(m_tc);
if (m_editor->find(needle, flags)) {
m_tc = m_editor->textCursor();
2008-12-25 13:20:09 +01:00
m_tc.movePosition(Left, MoveAnchor, needle.size() - 1);
return;
}
2008-12-26 00:18:03 +01:00
m_tc.setPosition(forward ? 0 : lastPositionInDocument() - 1);
m_editor->setTextCursor(m_tc);
if (m_editor->find(needle, flags)) {
m_tc = m_editor->textCursor();
2008-12-25 13:20:09 +01:00
m_tc.movePosition(Left, MoveAnchor, needle.size() - 1);
2008-12-26 00:18:03 +01:00
if (forward)
showMessage("search hit BOTTOM, continuing at TOP");
2008-12-26 00:18:03 +01:00
else
showMessage("search hit TOP, continuing at BOTTOM");
return;
}
2008-12-25 13:20:09 +01:00
m_tc = orig;
}
2008-12-19 14:35:57 +01:00
void FakeVimHandler::Private::moveToFirstNonBlankOnLine()
{
QTextBlock block = m_tc.block();
QTextDocument *doc = m_tc.document();
2008-12-19 16:20:39 +01:00
m_tc.movePosition(StartOfLine);
2008-12-19 14:35:57 +01:00
int firstPos = m_tc.position();
for (int i = firstPos, n = firstPos + block.length(); i < n; ++i) {
if (!doc->characterAt(i).isSpace()) {
m_tc.setPosition(i, KeepAnchor);
return;
}
}
}
static int charClass(QChar c, bool simple)
{
if (simple)
return c.isSpace() ? 0 : 1;
if (c.isLetterOrNumber() || c.unicode() == '_')
return 2;
return c.isSpace() ? 0 : 1;
}
2008-12-26 00:18:03 +01:00
void FakeVimHandler::Private::moveToWordBoundary(bool simple, bool forward)
2008-12-25 19:50:14 +01:00
{
2008-12-26 00:18:03 +01:00
int repeat = count();
2008-12-25 19:50:14 +01:00
QTextDocument *doc = m_tc.document();
2008-12-26 00:18:03 +01:00
int n = forward ? lastPositionInDocument() - 1 : 0;
2008-12-25 19:50:14 +01:00
int lastClass = 0;
2008-12-25 20:12:17 +01:00
while (true) {
2008-12-26 00:18:03 +01:00
m_tc.movePosition(forward ? Right : Left, KeepAnchor, 1);
2008-12-25 19:50:14 +01:00
QChar c = doc->characterAt(m_tc.position());
int thisClass = charClass(c, simple);
2008-12-25 20:12:17 +01:00
if (thisClass != lastClass && lastClass != 0)
2008-12-25 19:50:14 +01:00
--repeat;
2008-12-25 20:12:17 +01:00
if (repeat == -1) {
2008-12-26 00:18:03 +01:00
m_tc.movePosition(forward ? Left : Right, KeepAnchor, 1);
2008-12-25 20:12:17 +01:00
break;
}
2008-12-25 19:50:14 +01:00
lastClass = thisClass;
2008-12-26 00:18:03 +01:00
if (m_tc.position() == n)
2008-12-25 20:12:17 +01:00
break;
2008-12-25 19:50:14 +01:00
}
}
2008-12-26 00:18:03 +01:00
void FakeVimHandler::Private::handleFfTt(int key)
2008-12-25 19:50:14 +01:00
{
2008-12-26 00:18:03 +01:00
// m_subsubmode \in { 'f', 'F', 't', 'T' }
bool forward = m_subsubdata == 'f' || m_subsubdata == 't';
int repeat = count();
2008-12-25 19:50:14 +01:00
QTextDocument *doc = m_tc.document();
2008-12-26 00:18:03 +01:00
QTextBlock block = m_tc.block();
int n = block.position();
if (forward)
n += block.length();
int pos = m_tc.position();
2008-12-25 20:12:17 +01:00
while (true) {
2008-12-26 00:18:03 +01:00
pos += forward ? 1 : -1;
if (pos == n)
break;
int uc = doc->characterAt(pos).unicode();
if (uc == ParagraphSeparator)
break;
if (uc == key)
2008-12-25 19:50:14 +01:00
--repeat;
2008-12-25 20:12:17 +01:00
if (repeat == 0) {
2008-12-26 00:18:03 +01:00
if (m_subsubdata == 't')
--pos;
2008-12-26 00:25:38 +01:00
else if (m_subsubdata == 'T')
2008-12-26 00:18:03 +01:00
++pos;
2008-12-26 00:25:38 +01:00
// FIXME: strange correction...
if (m_submode == DeleteSubMode && m_subsubdata == 'f')
++pos;
if (m_submode == DeleteSubMode && m_subsubdata == 't')
++pos;
2008-12-26 00:18:03 +01:00
if (forward)
m_tc.movePosition(Right, KeepAnchor, pos - m_tc.position());
else
m_tc.movePosition(Left, KeepAnchor, m_tc.position() - pos);
2008-12-25 20:12:17 +01:00
break;
2008-12-25 19:50:14 +01:00
}
}
}
2008-12-26 00:18:03 +01:00
void FakeVimHandler::Private::moveToNextWord(bool simple)
{
// FIXME: 'w' should stop on empty lines, too
2008-12-26 00:18:03 +01:00
int repeat = count();
QTextDocument *doc = m_tc.document();
int n = lastPositionInDocument() - 1;
2008-12-25 23:19:08 +01:00
QChar c = doc->characterAt(m_tc.position());
int lastClass = charClass(c, simple);
2008-12-25 20:12:17 +01:00
while (true) {
2008-12-25 23:19:08 +01:00
c = doc->characterAt(m_tc.position());
int thisClass = charClass(c, simple);
if (thisClass != lastClass && thisClass != 0)
--repeat;
2008-12-25 22:42:50 +01:00
if (repeat == 0)
break;
lastClass = thisClass;
m_tc.movePosition(Right, KeepAnchor, 1);
2008-12-25 20:12:17 +01:00
if (m_tc.position() == n)
break;
}
}
int FakeVimHandler::Private::cursorLineOnScreen() const
{
QRect rect = m_editor->cursorRect();
return rect.y() / rect.height();
}
int FakeVimHandler::Private::linesOnScreen() const
{
QRect rect = m_editor->cursorRect();
//qDebug() << m_editor->height() / rect.height();
return m_editor->height() / rect.height();
}
int FakeVimHandler::Private::cursorLineInDocument() const
{
//qDebug() << "CURSOR LINE IN DOCUMENT " << m_tc.block().blockNumber();
return m_tc.block().blockNumber();
}
void FakeVimHandler::Private::scrollToLineInDocument(int line)
{
// FIXME: works only for QPlainTextEdit
QScrollBar *scrollBar = m_editor->verticalScrollBar();
scrollBar->setValue(line);
}
int FakeVimHandler::Private::lastPositionInDocument() const
{
QTextBlock block = m_tc.block().document()->lastBlock();
return block.position() + block.length();
}
///////////////////////////////////////////////////////////////////////
//
// FakeVimHandler
//
///////////////////////////////////////////////////////////////////////
FakeVimHandler::FakeVimHandler(QObject *parent)
: QObject(parent), d(new Private(this))
{}
FakeVimHandler::~FakeVimHandler()
{
delete d;
}
bool FakeVimHandler::eventFilter(QObject *ob, QEvent *ev)
{
if (ev->type() != QEvent::KeyPress)
return QObject::eventFilter(ob, ev);
return d->eventFilter(ob, ev);
}