2008-12-19 12:20:04 +01:00
|
|
|
/***************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** 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>
|
2008-12-27 16:42:07 +01:00
|
|
|
#include <QtCore/QFile>
|
2008-12-19 16:20:39 +01:00
|
|
|
#include <QtCore/QObject>
|
|
|
|
|
#include <QtCore/QRegExp>
|
2008-12-27 16:42:07 +01:00
|
|
|
#include <QtCore/QTextStream>
|
2008-12-19 16:20:39 +01:00
|
|
|
#include <QtCore/QStack>
|
|
|
|
|
|
|
|
|
|
#include <QtGui/QKeyEvent>
|
|
|
|
|
#include <QtGui/QLineEdit>
|
|
|
|
|
#include <QtGui/QPlainTextEdit>
|
|
|
|
|
#include <QtGui/QScrollBar>
|
|
|
|
|
#include <QtGui/QTextBlock>
|
|
|
|
|
#include <QtGui/QTextCursor>
|
2009-01-06 11:33:07 +01:00
|
|
|
#include <QtGui/QTextDocumentFragment>
|
2008-12-19 16:20:39 +01:00
|
|
|
#include <QtGui/QTextEdit>
|
2008-12-19 12:20:04 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
using namespace FakeVim::Internal;
|
|
|
|
|
|
|
|
|
|
#define StartOfLine QTextCursor::StartOfLine
|
2008-12-26 18:29:38 +01:00
|
|
|
#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
|
2008-12-19 12:20:04 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// FakeVimHandler
|
|
|
|
|
//
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
2008-12-25 16:27:47 +01:00
|
|
|
|
2008-12-26 18:29:38 +01:00
|
|
|
#define EDITOR(s) (m_textedit ? m_textedit->s : m_plaintextedit->s)
|
2008-12-28 02:32:44 +01:00
|
|
|
#define THE_EDITOR (m_textedit ? (QWidget*)m_textedit : (QWidget*)m_plaintextedit)
|
2008-12-26 18:29:38 +01:00
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
const int ParagraphSeparator = 0x00002029;
|
|
|
|
|
|
|
|
|
|
using namespace Qt;
|
|
|
|
|
|
2008-12-19 16:20:39 +01:00
|
|
|
enum Mode
|
|
|
|
|
{
|
|
|
|
|
InsertMode,
|
|
|
|
|
CommandMode,
|
|
|
|
|
ExMode
|
|
|
|
|
};
|
2008-12-19 12:20:04 +01:00
|
|
|
|
2008-12-19 16:20:39 +01:00
|
|
|
enum SubMode
|
|
|
|
|
{
|
|
|
|
|
NoSubMode,
|
|
|
|
|
RegisterSubMode,
|
|
|
|
|
ChangeSubMode,
|
|
|
|
|
DeleteSubMode,
|
|
|
|
|
ZSubMode
|
|
|
|
|
};
|
2008-12-19 12:20:04 +01:00
|
|
|
|
2008-12-26 00:18:03 +01:00
|
|
|
enum SubSubMode
|
|
|
|
|
{
|
|
|
|
|
NoSubSubMode,
|
2008-12-28 03:07:52 +01:00
|
|
|
FtSubSubMode, // used for f, F, t, T
|
|
|
|
|
MarkSubSubMode, // used for m
|
|
|
|
|
BackTickSubSubMode, // used for `
|
|
|
|
|
TickSubSubMode // used for '
|
2008-12-26 00:18:03 +01:00
|
|
|
};
|
|
|
|
|
|
2008-12-27 21:28:22 +01:00
|
|
|
static const QString ConfigStartOfLine = "startofline";
|
|
|
|
|
static const QString ConfigOn = "on";
|
|
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
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();
|
2008-12-19 13:55:03 +01:00
|
|
|
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);
|
2008-12-19 12:20:04 +01:00
|
|
|
void finishMovement();
|
2008-12-27 21:01:05 +01:00
|
|
|
void updateMiniBuffer();
|
2008-12-26 00:18:03 +01:00
|
|
|
void search(const QString &needle, bool forward);
|
2008-12-19 12:20:04 +01:00
|
|
|
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(); }
|
2008-12-19 12:20:04 +01:00
|
|
|
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
|
|
|
|
2008-12-25 19:11:21 +01:00
|
|
|
int lastPositionInDocument() const;
|
2009-01-06 11:42:44 +01:00
|
|
|
int positionForLine(int line) const; // 1 based line, 0 based pos
|
|
|
|
|
int lineForPosition(int pos) const; // 1 based line, 0 based pos
|
2008-12-25 19:11:21 +01:00
|
|
|
|
2009-01-06 11:11:31 +01:00
|
|
|
// all zero-based counting
|
|
|
|
|
int cursorLineOnScreen() const;
|
|
|
|
|
int linesOnScreen() const;
|
|
|
|
|
int columnsOnScreen() const;
|
|
|
|
|
int cursorLineInDocument() const;
|
|
|
|
|
int cursorColumnInDocument() const;
|
2008-12-28 02:15:26 +01:00
|
|
|
int linesInDocument() const;
|
2009-01-06 11:11:31 +01:00
|
|
|
void scrollToLineInDocument(int line);
|
2008-12-25 16:27:47 +01:00
|
|
|
|
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);
|
2008-12-27 16:42:07 +01:00
|
|
|
void handleCommand(const QString &cmd);
|
2008-12-19 12:20:04 +01:00
|
|
|
|
2008-12-28 02:15:26 +01:00
|
|
|
// helper function for handleCommand. return 1 based line index.
|
|
|
|
|
int readLineCode(QString &cmd);
|
2009-01-06 11:33:07 +01:00
|
|
|
QTextCursor selectRange(int beginLine, int endLine);
|
2008-12-28 02:15:26 +01:00
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
FakeVimHandler *q;
|
|
|
|
|
Mode m_mode;
|
|
|
|
|
SubMode m_submode;
|
2008-12-26 00:18:03 +01:00
|
|
|
SubSubMode m_subsubmode;
|
|
|
|
|
int m_subsubdata;
|
2008-12-19 12:20:04 +01:00
|
|
|
QString m_input;
|
2008-12-26 18:29:38 +01:00
|
|
|
QTextEdit *m_textedit;
|
|
|
|
|
QPlainTextEdit *m_plaintextedit;
|
2008-12-19 12:20:04 +01:00
|
|
|
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;
|
2008-12-19 12:20:04 +01:00
|
|
|
|
|
|
|
|
QStack<QString> m_undoStack;
|
|
|
|
|
QStack<QString> m_redoStack;
|
|
|
|
|
|
|
|
|
|
bool m_fakeEnd;
|
|
|
|
|
|
2008-12-26 17:01:21 +01:00
|
|
|
bool isSearchCommand() const
|
|
|
|
|
{ return m_commandCode == '?' || m_commandCode == '/'; }
|
2008-12-19 12:20:04 +01:00
|
|
|
int m_commandCode; // ?, /, : ...
|
2008-12-27 13:39:34 +01:00
|
|
|
int m_gflag; // whether current command started with 'g'
|
2008-12-26 17:01:21 +01:00
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
QString m_commandBuffer;
|
2008-12-28 01:02:54 +01:00
|
|
|
QString m_currentFileName;
|
|
|
|
|
QString m_currentMessage;
|
2008-12-19 12:20:04 +01:00
|
|
|
|
2008-12-26 00:29:51 +01:00
|
|
|
bool m_lastSearchForward;
|
2008-12-27 12:24:50 +01:00
|
|
|
QString m_lastInsertion;
|
2008-12-26 17:01:21 +01:00
|
|
|
|
2009-01-06 11:42:44 +01:00
|
|
|
// history for '/'
|
2008-12-26 17:01:21 +01:00
|
|
|
QString lastSearchString() const;
|
|
|
|
|
QStringList m_searchHistory;
|
|
|
|
|
int m_searchHistoryIndex;
|
|
|
|
|
|
2009-01-06 11:42:44 +01:00
|
|
|
// history for ':'
|
2008-12-26 17:01:21 +01:00
|
|
|
QStringList m_commandHistory;
|
|
|
|
|
int m_commandHistoryIndex;
|
2008-12-27 21:28:22 +01:00
|
|
|
|
2009-01-06 11:42:44 +01:00
|
|
|
// visual line mode
|
|
|
|
|
void enterVisualLineMode();
|
|
|
|
|
void leaveVisualLineMode();
|
|
|
|
|
bool m_visualLineMode;
|
|
|
|
|
|
2008-12-28 03:07:52 +01:00
|
|
|
// marks as lines
|
|
|
|
|
QHash<int, int> m_marks;
|
|
|
|
|
|
2008-12-27 21:28:22 +01:00
|
|
|
// vi style configuration
|
|
|
|
|
QHash<QString, QString> m_config;
|
2008-12-19 12:20:04 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
FakeVimHandler::Private::Private(FakeVimHandler *parent)
|
|
|
|
|
{
|
|
|
|
|
q = parent;
|
2008-12-27 22:50:58 +01:00
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
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;
|
2008-12-19 12:20:04 +01:00
|
|
|
m_fakeEnd = false;
|
2008-12-26 00:29:51 +01:00
|
|
|
m_lastSearchForward = true;
|
2008-12-19 12:20:04 +01:00
|
|
|
m_register = '"';
|
2008-12-27 13:39:34 +01:00
|
|
|
m_gflag = false;
|
2008-12-27 16:42:07 +01:00
|
|
|
m_textedit = 0;
|
|
|
|
|
m_plaintextedit = 0;
|
2009-01-06 11:42:44 +01:00
|
|
|
m_visualLineMode = false;
|
2008-12-27 21:28:22 +01:00
|
|
|
|
|
|
|
|
m_config[ConfigStartOfLine] = ConfigOn;
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
// Fake "End of line"
|
2008-12-26 18:29:38 +01:00
|
|
|
m_textedit = qobject_cast<QTextEdit *>(ob);
|
|
|
|
|
m_plaintextedit = qobject_cast<QPlainTextEdit *>(ob);
|
|
|
|
|
if (!m_textedit && !m_plaintextedit)
|
2008-12-19 12:20:04 +01:00
|
|
|
return false;
|
|
|
|
|
|
2008-12-26 18:29:38 +01:00
|
|
|
m_tc = EDITOR(textCursor());
|
|
|
|
|
|
2008-12-27 22:50:58 +01:00
|
|
|
if (m_fakeEnd)
|
2008-12-19 12:20:04 +01:00
|
|
|
m_tc.movePosition(Right, MoveAnchor, 1);
|
|
|
|
|
|
|
|
|
|
if (key >= Key_A && key <= Key_Z
|
|
|
|
|
&& (keyEvent->modifiers() & Qt::ShiftModifier) == 0)
|
|
|
|
|
key += 32;
|
|
|
|
|
if ((keyEvent->modifiers() & Qt::ControlModifier) != 0)
|
|
|
|
|
key += 256;
|
2008-12-19 13:55:03 +01:00
|
|
|
handleKey(key, keyEvent->text());
|
2008-12-19 12:20:04 +01:00
|
|
|
|
|
|
|
|
// We fake vi-style end-of-line behaviour
|
2008-12-27 22:50:58 +01:00
|
|
|
m_fakeEnd = (atEol() && m_mode == CommandMode);
|
2008-12-19 12:20:04 +01:00
|
|
|
|
2008-12-27 22:50:58 +01:00
|
|
|
if (m_fakeEnd)
|
2008-12-19 12:20:04 +01:00
|
|
|
m_tc.movePosition(Left, MoveAnchor, 1);
|
|
|
|
|
|
2008-12-26 18:29:38 +01:00
|
|
|
EDITOR(setTextCursor(m_tc));
|
|
|
|
|
EDITOR(ensureCursorVisible());
|
2008-12-19 12:20:04 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-19 13:55:03 +01:00
|
|
|
void FakeVimHandler::Private::handleKey(int key, const QString &text)
|
2008-12-19 12:20:04 +01:00
|
|
|
{
|
|
|
|
|
if (m_mode == InsertMode)
|
2008-12-19 13:55:03 +01:00
|
|
|
handleInsertMode(key, text);
|
2008-12-19 12:20:04 +01:00
|
|
|
else if (m_mode == CommandMode)
|
2008-12-19 13:55:03 +01:00
|
|
|
handleCommandMode(key, text);
|
2008-12-19 12:20:04 +01:00
|
|
|
else if (m_mode == ExMode)
|
2008-12-19 13:55:03 +01:00
|
|
|
handleExMode(key, text);
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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();
|
2008-12-27 13:39:34 +01:00
|
|
|
m_gflag = false;
|
2008-12-19 12:20:04 +01:00
|
|
|
m_register = '"';
|
|
|
|
|
m_tc.clearSelection();
|
2008-12-27 21:01:05 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-27 21:01:05 +01:00
|
|
|
void FakeVimHandler::Private::updateMiniBuffer()
|
2008-12-19 12:20:04 +01:00
|
|
|
{
|
2008-12-28 01:02:54 +01:00
|
|
|
if (m_tc.isNull())
|
|
|
|
|
return;
|
2008-12-27 21:01:05 +01:00
|
|
|
QString msg;
|
2009-01-06 11:42:44 +01:00
|
|
|
if (!m_currentMessage.isEmpty()) {
|
|
|
|
|
msg = m_currentMessage;
|
|
|
|
|
m_currentMessage.clear();
|
|
|
|
|
} else if (m_visualLineMode) {
|
|
|
|
|
msg = "-- VISUAL LINE--";
|
|
|
|
|
} else {
|
2008-12-28 01:02:54 +01:00
|
|
|
msg = QChar(m_commandCode ? m_commandCode : ' ');
|
|
|
|
|
for (int i = 0; i != m_commandBuffer.size(); ++i) {
|
|
|
|
|
QChar c = m_commandBuffer.at(i);
|
|
|
|
|
if (c.unicode() < 32) {
|
|
|
|
|
msg += '^';
|
|
|
|
|
msg += QChar(c.unicode() + 64);
|
|
|
|
|
} else {
|
|
|
|
|
msg += c;
|
|
|
|
|
}
|
2008-12-27 22:41:47 +01:00
|
|
|
}
|
|
|
|
|
}
|
2008-12-27 21:01:05 +01:00
|
|
|
int l = cursorLineInDocument();
|
|
|
|
|
int w = columnsOnScreen();
|
|
|
|
|
msg += QString(w, ' ');
|
|
|
|
|
msg = msg.left(w - 20);
|
|
|
|
|
QString pos = tr("%1,%2").arg(l + 1).arg(cursorColumnInDocument() + 1);
|
|
|
|
|
msg += tr("%1").arg(pos, -12);
|
|
|
|
|
// FIXME: physical "-" logical
|
|
|
|
|
msg += tr("%1").arg(l * 100 / (m_tc.document()->blockCount() - 1), 4);
|
|
|
|
|
msg += '%';
|
2008-12-19 12:20:04 +01:00
|
|
|
emit q->commandBufferChanged(msg);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FakeVimHandler::Private::showMessage(const QString &msg)
|
|
|
|
|
{
|
2008-12-28 01:02:54 +01:00
|
|
|
//qDebug() << "MSG: " << msg;
|
|
|
|
|
m_currentMessage = msg;
|
2008-12-27 21:01:05 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-19 13:55:03 +01:00
|
|
|
void FakeVimHandler::Private::handleCommandMode(int key, const QString &text)
|
2008-12-19 12:20:04 +01:00
|
|
|
{
|
2008-12-19 13:55:03 +01:00
|
|
|
Q_UNUSED(text)
|
|
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
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
|
2009-01-06 11:11:31 +01:00
|
|
|
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();
|
2008-12-28 03:07:52 +01:00
|
|
|
} else if (m_subsubmode == MarkSubSubMode) {
|
|
|
|
|
m_marks[key] = m_tc.position();
|
|
|
|
|
m_subsubmode = NoSubSubMode;
|
|
|
|
|
} else if (m_subsubmode == BackTickSubSubMode
|
|
|
|
|
|| m_subsubmode == TickSubSubMode) {
|
|
|
|
|
if (m_marks.contains(key)) {
|
2009-01-06 11:33:07 +01:00
|
|
|
m_tc.setPosition(m_marks[key], MoveAnchor);
|
2008-12-28 03:07:52 +01:00
|
|
|
if (m_subsubmode == TickSubSubMode)
|
|
|
|
|
moveToFirstNonBlankOnLine();
|
|
|
|
|
finishMovement();
|
|
|
|
|
} else {
|
|
|
|
|
showMessage(tr("E20: Mark '%1' not set").arg(text));
|
|
|
|
|
}
|
|
|
|
|
m_subsubmode = NoSubSubMode;
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key >= '0' && key <= '9') {
|
2008-12-25 22:41:09 +01:00
|
|
|
if (key == '0' && m_mvcount.isEmpty()) {
|
2008-12-28 02:49:14 +01:00
|
|
|
moveToFirstNonBlankOnLine();
|
2008-12-19 12:20:04 +01:00
|
|
|
finishMovement();
|
|
|
|
|
} else {
|
2008-12-25 22:41:09 +01:00
|
|
|
m_mvcount.append(QChar(key));
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
} else if (key == ':' || key == '/' || key == '?') {
|
|
|
|
|
m_commandCode = key;
|
|
|
|
|
m_mode = ExMode;
|
2008-12-26 17:01:21 +01:00
|
|
|
if (isSearchCommand()) {
|
|
|
|
|
m_searchHistory.append(QString());
|
|
|
|
|
m_searchHistoryIndex = m_searchHistory.size() - 1;
|
|
|
|
|
} else {
|
|
|
|
|
m_commandHistory.append(QString());
|
|
|
|
|
m_commandHistoryIndex = m_commandHistory.size() - 1;
|
|
|
|
|
}
|
2008-12-27 21:01:05 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-28 03:07:52 +01:00
|
|
|
} else if (key == '`') {
|
|
|
|
|
m_subsubmode = BackTickSubSubMode;
|
|
|
|
|
} else if (key == '\'') {
|
|
|
|
|
m_subsubmode = TickSubSubMode;
|
2008-12-19 12:20:04 +01:00
|
|
|
} 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();
|
2008-12-27 21:51:06 +01:00
|
|
|
} else if (key == 'a') {
|
2008-12-27 22:22:16 +01:00
|
|
|
m_mode = InsertMode;
|
2008-12-27 21:51:06 +01:00
|
|
|
m_lastInsertion.clear();
|
2008-12-27 22:22:16 +01:00
|
|
|
m_tc.beginEditBlock();
|
2008-12-27 21:51:06 +01:00
|
|
|
m_tc.movePosition(Right, MoveAnchor, 1);
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'A') {
|
2008-12-27 22:22:16 +01:00
|
|
|
m_mode = InsertMode;
|
|
|
|
|
m_tc.beginEditBlock();
|
2008-12-19 12:20:04 +01:00
|
|
|
m_tc.movePosition(EndOfLine, MoveAnchor);
|
2008-12-27 21:51:06 +01:00
|
|
|
m_lastInsertion.clear();
|
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();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'c') {
|
|
|
|
|
m_submode = ChangeSubMode;
|
2008-12-27 22:22:16 +01:00
|
|
|
m_tc.beginEditBlock();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'C') {
|
|
|
|
|
m_submode = ChangeSubMode;
|
2008-12-27 22:22:16 +01:00
|
|
|
m_tc.beginEditBlock();
|
2008-12-19 12:20:04 +01:00
|
|
|
m_tc.movePosition(EndOfLine, KeepAnchor);
|
|
|
|
|
finishMovement();
|
|
|
|
|
} else if (key == 'd') {
|
2009-01-06 11:42:44 +01:00
|
|
|
if (m_visualLineMode) {
|
|
|
|
|
leaveVisualLineMode();
|
|
|
|
|
int beginLine = lineForPosition(m_marks['<']);
|
|
|
|
|
int endLine = lineForPosition(m_marks['>']);
|
|
|
|
|
m_tc = selectRange(beginLine, endLine);
|
|
|
|
|
m_tc.removeSelectedText();
|
|
|
|
|
} else {
|
|
|
|
|
if (atEol())
|
|
|
|
|
m_tc.movePosition(Left, MoveAnchor, 1);
|
|
|
|
|
m_opcount = m_mvcount;
|
|
|
|
|
m_mvcount.clear();
|
|
|
|
|
m_submode = DeleteSubMode;
|
|
|
|
|
}
|
2008-12-19 12:20:04 +01:00
|
|
|
} 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());
|
2008-12-19 12:20:04 +01:00
|
|
|
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;
|
2008-12-27 13:39:34 +01:00
|
|
|
} else if (key == 'g') {
|
|
|
|
|
m_gflag = true;
|
2008-12-27 21:28:22 +01:00
|
|
|
} else if (key == 'G') {
|
|
|
|
|
int n = m_mvcount.isEmpty() ? linesInDocument() : count();
|
2009-01-06 11:33:07 +01:00
|
|
|
m_tc.setPosition(positionForLine(n), MoveAnchor);
|
2008-12-27 21:28:22 +01:00
|
|
|
if (m_config.contains(ConfigStartOfLine))
|
|
|
|
|
moveToFirstNonBlankOnLine();
|
|
|
|
|
finishMovement();
|
2008-12-19 12:20:04 +01:00
|
|
|
} 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-26 18:29:38 +01:00
|
|
|
m_tc = EDITOR(cursorForPosition(QPoint(0, 0)));
|
2008-12-19 14:35:57 +01:00
|
|
|
m_tc.movePosition(Down, KeepAnchor, qMax(count() - 1, 0));
|
|
|
|
|
moveToFirstNonBlankOnLine();
|
|
|
|
|
finishMovement();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'i') {
|
|
|
|
|
m_mode = InsertMode;
|
2008-12-27 22:22:16 +01:00
|
|
|
m_tc.beginEditBlock();
|
|
|
|
|
m_lastInsertion.clear();
|
2008-12-28 02:44:43 +01:00
|
|
|
} else if (key == 'I') {
|
|
|
|
|
m_mode = InsertMode;
|
2008-12-28 02:49:14 +01:00
|
|
|
if (m_gflag)
|
|
|
|
|
m_tc.movePosition(StartOfLine, KeepAnchor);
|
|
|
|
|
else
|
|
|
|
|
moveToFirstNonBlankOnLine();
|
2008-12-28 02:44:43 +01:00
|
|
|
m_tc.clearSelection();
|
|
|
|
|
m_tc.beginEditBlock();
|
|
|
|
|
m_lastInsertion.clear();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'j' || key == Key_Down) {
|
|
|
|
|
m_tc.movePosition(Down, KeepAnchor, count());
|
|
|
|
|
finishMovement();
|
2008-12-27 13:39:34 +01:00
|
|
|
} else if (key == 'J') {
|
2008-12-27 21:51:06 +01:00
|
|
|
m_tc.beginEditBlock();
|
2008-12-27 13:39:34 +01:00
|
|
|
if (m_submode == NoSubMode) {
|
|
|
|
|
for (int i = qMax(count(), 2) - 1; --i >= 0; ) {
|
|
|
|
|
m_tc.movePosition(EndOfLine);
|
|
|
|
|
m_tc.deleteChar();
|
2008-12-27 13:50:52 +01:00
|
|
|
if (!m_gflag)
|
|
|
|
|
m_tc.insertText(" ");
|
2008-12-27 13:39:34 +01:00
|
|
|
}
|
2008-12-27 13:50:52 +01:00
|
|
|
if (!m_gflag)
|
|
|
|
|
m_tc.movePosition(Left, MoveAnchor, 1);
|
2008-12-27 13:39:34 +01:00
|
|
|
}
|
2008-12-27 21:51:06 +01:00
|
|
|
m_tc.endEditBlock();
|
2008-12-19 12:20:04 +01:00
|
|
|
} 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-27 22:50:58 +01:00
|
|
|
m_tc = EDITOR(cursorForPosition(QPoint(0, EDITOR(height()))));
|
2008-12-19 14:43:14 +01:00
|
|
|
m_tc.movePosition(Up, KeepAnchor, qMax(count(), 1));
|
|
|
|
|
moveToFirstNonBlankOnLine();
|
|
|
|
|
finishMovement();
|
2008-12-28 03:07:52 +01:00
|
|
|
} else if (key == 'm') {
|
|
|
|
|
m_subsubmode = MarkSubSubMode;
|
2008-12-19 15:00:06 +01:00
|
|
|
} else if (key == 'M') {
|
2008-12-27 22:50:58 +01:00
|
|
|
m_tc = EDITOR(cursorForPosition(QPoint(0, EDITOR(height()) / 2)));
|
2008-12-19 15:00:06 +01:00
|
|
|
moveToFirstNonBlankOnLine();
|
|
|
|
|
finishMovement();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'n') {
|
2008-12-26 17:01:21 +01:00
|
|
|
search(lastSearchString(), m_lastSearchForward);
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'N') {
|
2008-12-26 17:01:21 +01:00
|
|
|
search(lastSearchString(), !m_lastSearchForward);
|
2008-12-28 02:32:44 +01:00
|
|
|
} else if (key == 'o') {
|
|
|
|
|
m_mode = InsertMode;
|
|
|
|
|
m_lastInsertion.clear();
|
|
|
|
|
//m_tc.beginEditBlock(); // FIXME: unusable due to drawing errors
|
|
|
|
|
m_tc.movePosition(EndOfLine, MoveAnchor);
|
|
|
|
|
m_tc.insertText("\n");
|
2008-12-28 02:42:10 +01:00
|
|
|
} else if (key == 'O') {
|
|
|
|
|
m_mode = InsertMode;
|
|
|
|
|
m_lastInsertion.clear();
|
|
|
|
|
//m_tc.beginEditBlock(); // FIXME: unusable due to drawing errors
|
|
|
|
|
m_tc.movePosition(StartOfLine, MoveAnchor);
|
|
|
|
|
m_tc.movePosition(Left, MoveAnchor, 1);
|
|
|
|
|
m_tc.insertText("\n");
|
2008-12-19 12:20:04 +01:00
|
|
|
} 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')) {
|
2008-12-26 18:29:38 +01:00
|
|
|
EDITOR(redo());
|
2008-12-27 22:22:16 +01:00
|
|
|
} else if (key == 's') {
|
|
|
|
|
m_submode = ChangeSubMode;
|
|
|
|
|
m_tc.beginEditBlock();
|
|
|
|
|
m_tc.movePosition(Right, KeepAnchor, qMin(count(), rightDist()));
|
|
|
|
|
finishMovement();
|
2008-12-26 00:18:03 +01:00
|
|
|
} else if (key == 't' || key == 'T') {
|
|
|
|
|
m_subsubmode = FtSubSubMode;
|
|
|
|
|
m_subsubdata = key;
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == 'u') {
|
2008-12-26 18:29:38 +01:00
|
|
|
EDITOR(undo());
|
2009-01-06 11:42:44 +01:00
|
|
|
} else if (key == 'V') {
|
|
|
|
|
enterVisualLineMode();
|
2008-12-25 19:11:21 +01:00
|
|
|
} else if (key == 'w') {
|
2008-12-26 00:18:03 +01:00
|
|
|
moveToNextWord(false);
|
2008-12-25 19:11:21 +01:00
|
|
|
finishMovement();
|
|
|
|
|
} else if (key == 'W') {
|
2008-12-26 00:18:03 +01:00
|
|
|
moveToNextWord(true);
|
2008-12-25 19:11:21 +01:00
|
|
|
finishMovement();
|
2008-12-19 12:20:04 +01:00
|
|
|
} 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);
|
2008-12-25 16:27:47 +01:00
|
|
|
} 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();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Backspace) {
|
|
|
|
|
m_tc.deletePreviousChar();
|
2008-12-19 13:59:45 +01:00
|
|
|
} else if (key == Key_Delete) {
|
|
|
|
|
m_tc.deleteChar();
|
2008-12-25 22:57:21 +01:00
|
|
|
} else if (key == Key_Escape) {
|
2009-01-06 11:42:44 +01:00
|
|
|
if (m_visualLineMode)
|
|
|
|
|
leaveVisualLineMode();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else {
|
2008-12-19 16:20:39 +01:00
|
|
|
qDebug() << "Ignored" << key << text;
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-19 13:55:03 +01:00
|
|
|
void FakeVimHandler::Private::handleInsertMode(int key, const QString &text)
|
2008-12-19 12:20:04 +01:00
|
|
|
{
|
|
|
|
|
if (key == Key_Escape) {
|
2008-12-27 12:24:50 +01:00
|
|
|
for (int i = 1; i < count(); ++i)
|
|
|
|
|
m_tc.insertText(m_lastInsertion);
|
|
|
|
|
m_tc.movePosition(Left, MoveAnchor, qMin(1, leftDist()));
|
2008-12-27 22:22:16 +01:00
|
|
|
m_mode = CommandMode;
|
|
|
|
|
m_tc.endEditBlock();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Left) {
|
|
|
|
|
m_tc.movePosition(Left, MoveAnchor, 1);
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Down) {
|
|
|
|
|
m_tc.movePosition(Down, MoveAnchor, 1);
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Up) {
|
|
|
|
|
m_tc.movePosition(Up, MoveAnchor, 1);
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Right) {
|
|
|
|
|
m_tc.movePosition(Right, MoveAnchor, 1);
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Return) {
|
|
|
|
|
m_tc.insertBlock();
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Backspace) {
|
|
|
|
|
m_tc.deletePreviousChar();
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion = m_lastInsertion.left(m_lastInsertion.size() - 1);
|
2008-12-19 13:59:45 +01:00
|
|
|
} else if (key == Key_Delete) {
|
|
|
|
|
m_tc.deleteChar();
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
2008-12-25 16:27:47 +01:00
|
|
|
} else if (key == Key_PageDown || key == control('f')) {
|
|
|
|
|
m_tc.movePosition(Down, KeepAnchor, count() * (linesOnScreen() - 2));
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
|
|
|
|
} else if (key == Key_Backspace) {
|
2008-12-25 16:27:47 +01:00
|
|
|
finishMovement();
|
|
|
|
|
} else if (key == Key_PageUp || key == control('b')) {
|
|
|
|
|
m_tc.movePosition(Up, KeepAnchor, count() * (linesOnScreen() - 2));
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.clear();
|
|
|
|
|
} else if (key == Key_Backspace) {
|
2008-12-25 16:27:47 +01:00
|
|
|
finishMovement();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else {
|
2008-12-27 12:24:50 +01:00
|
|
|
m_lastInsertion.append(text);
|
2008-12-19 13:55:03 +01:00
|
|
|
m_tc.insertText(text);
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-19 13:55:03 +01:00
|
|
|
void FakeVimHandler::Private::handleExMode(int key, const QString &text)
|
2008-12-19 12:20:04 +01:00
|
|
|
{
|
2008-12-19 13:55:03 +01:00
|
|
|
Q_UNUSED(text)
|
|
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
if (key == Key_Escape) {
|
|
|
|
|
m_commandBuffer.clear();
|
|
|
|
|
m_commandCode = 0;
|
|
|
|
|
m_mode = CommandMode;
|
2008-12-27 21:01:05 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Backspace) {
|
|
|
|
|
if (m_commandBuffer.isEmpty())
|
|
|
|
|
m_mode = CommandMode;
|
|
|
|
|
else
|
|
|
|
|
m_commandBuffer.chop(1);
|
2008-12-27 21:01:05 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else if (key == Key_Return && m_commandCode == ':') {
|
2008-12-28 00:32:07 +01:00
|
|
|
if (!m_commandBuffer.isEmpty()) {
|
|
|
|
|
m_commandHistory.takeLast();
|
|
|
|
|
m_commandHistory.append(m_commandBuffer);
|
|
|
|
|
handleCommand(m_commandBuffer);
|
|
|
|
|
m_commandBuffer.clear();
|
|
|
|
|
m_commandCode = 0;
|
|
|
|
|
}
|
2008-12-19 12:20:04 +01:00
|
|
|
m_mode = CommandMode;
|
2008-12-26 17:01:21 +01:00
|
|
|
} else if (key == Key_Return && isSearchCommand()) {
|
2008-12-28 00:32:07 +01:00
|
|
|
if (!m_commandBuffer.isEmpty()) {
|
|
|
|
|
m_searchHistory.takeLast();
|
|
|
|
|
m_searchHistory.append(m_commandBuffer);
|
|
|
|
|
m_lastSearchForward = (m_commandCode == '/');
|
|
|
|
|
search(lastSearchString(), m_lastSearchForward);
|
|
|
|
|
m_commandBuffer.clear();
|
|
|
|
|
m_commandCode = 0;
|
|
|
|
|
}
|
2008-12-19 12:20:04 +01:00
|
|
|
m_mode = CommandMode;
|
2008-12-27 21:01:05 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-28 01:02:54 +01:00
|
|
|
} else if (key == Key_Up && isSearchCommand()) {
|
2008-12-28 03:07:52 +01:00
|
|
|
// FIXME: This and the three cases below are wrong as vim
|
|
|
|
|
// takes only matching entires in the history into account.
|
2008-12-28 01:02:54 +01:00
|
|
|
if (m_searchHistoryIndex > 0) {
|
2008-12-26 17:01:21 +01:00
|
|
|
--m_searchHistoryIndex;
|
|
|
|
|
m_commandBuffer = m_searchHistory.at(m_searchHistoryIndex);
|
2008-12-28 01:02:54 +01:00
|
|
|
updateMiniBuffer();
|
|
|
|
|
}
|
|
|
|
|
} else if (key == Key_Up && m_commandCode == ':') {
|
|
|
|
|
if (m_commandHistoryIndex > 0) {
|
2008-12-28 00:32:07 +01:00
|
|
|
--m_commandHistoryIndex;
|
|
|
|
|
m_commandBuffer = m_commandHistory.at(m_commandHistoryIndex);
|
2008-12-28 01:02:54 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-26 17:01:21 +01:00
|
|
|
}
|
2008-12-28 01:02:54 +01:00
|
|
|
} else if (key == Key_Down && isSearchCommand()) {
|
|
|
|
|
if (m_searchHistoryIndex < m_searchHistory.size() - 1) {
|
2008-12-26 17:01:21 +01:00
|
|
|
++m_searchHistoryIndex;
|
|
|
|
|
m_commandBuffer = m_searchHistory.at(m_searchHistoryIndex);
|
2008-12-28 01:02:54 +01:00
|
|
|
updateMiniBuffer();
|
|
|
|
|
}
|
|
|
|
|
} else if (key == Key_Down && m_commandCode == ':') {
|
|
|
|
|
if (m_commandHistoryIndex < m_commandHistory.size() - 1) {
|
2008-12-28 00:32:07 +01:00
|
|
|
++m_commandHistoryIndex;
|
|
|
|
|
m_commandBuffer = m_commandHistory.at(m_commandHistoryIndex);
|
2008-12-28 01:02:54 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-26 17:01:21 +01:00
|
|
|
}
|
2008-12-27 22:41:47 +01:00
|
|
|
} else if (key == Key_Tab) {
|
|
|
|
|
m_commandBuffer += QChar(9);
|
|
|
|
|
updateMiniBuffer();
|
2008-12-19 12:20:04 +01:00
|
|
|
} else {
|
|
|
|
|
m_commandBuffer += QChar(key);
|
2008-12-27 21:01:05 +01:00
|
|
|
updateMiniBuffer();
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-28 02:15:26 +01:00
|
|
|
// 1 based.
|
|
|
|
|
int FakeVimHandler::Private::readLineCode(QString &cmd)
|
|
|
|
|
{
|
|
|
|
|
//qDebug() << "CMD: " << cmd;
|
|
|
|
|
if (cmd.isEmpty())
|
|
|
|
|
return -1;
|
|
|
|
|
QChar c = cmd.at(0);
|
|
|
|
|
cmd = cmd.mid(1);
|
|
|
|
|
if (c == '.')
|
|
|
|
|
return cursorLineInDocument() + 1;
|
|
|
|
|
if (c == '$')
|
|
|
|
|
return linesInDocument();
|
|
|
|
|
if (c == '-') {
|
|
|
|
|
int n = readLineCode(cmd);
|
|
|
|
|
return cursorLineInDocument() + 1 - (n == -1 ? 1 : n);
|
|
|
|
|
}
|
|
|
|
|
if (c == '+') {
|
|
|
|
|
int n = readLineCode(cmd);
|
|
|
|
|
return cursorLineInDocument() + 1 + (n == -1 ? 1 : n);
|
|
|
|
|
}
|
2009-01-06 11:42:44 +01:00
|
|
|
if (c == '\'' && !cmd.isEmpty()) {
|
|
|
|
|
int pos = m_marks.value(cmd.at(0).unicode());
|
|
|
|
|
if (!pos) {
|
|
|
|
|
showMessage(tr("E20: Mark '%1' not set").arg(cmd.at(0)));
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
cmd = cmd.mid(1);
|
|
|
|
|
return lineForPosition(pos);
|
|
|
|
|
}
|
2008-12-28 02:15:26 +01:00
|
|
|
if (c.isDigit()) {
|
|
|
|
|
int n = c.unicode() - '0';
|
|
|
|
|
while (!cmd.isEmpty()) {
|
|
|
|
|
c = cmd.at(0);
|
|
|
|
|
if (!c.isDigit())
|
|
|
|
|
break;
|
|
|
|
|
cmd = cmd.mid(1);
|
|
|
|
|
n = n * 10 + (c.unicode() - '0');
|
|
|
|
|
}
|
|
|
|
|
//qDebug() << "N: " << n;
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
// not parsed
|
|
|
|
|
cmd = c + cmd;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2009-01-06 11:33:07 +01:00
|
|
|
QTextCursor FakeVimHandler::Private::selectRange(int beginLine, int endLine)
|
|
|
|
|
{
|
|
|
|
|
QTextCursor tc = m_tc;
|
|
|
|
|
tc.setPosition(positionForLine(beginLine), MoveAnchor);
|
|
|
|
|
tc.setPosition(positionForLine(endLine + 1), KeepAnchor);
|
|
|
|
|
return tc;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-28 02:15:26 +01:00
|
|
|
void FakeVimHandler::Private::handleCommand(const QString &cmd0)
|
2008-12-27 16:42:07 +01:00
|
|
|
{
|
2008-12-28 02:15:26 +01:00
|
|
|
QString cmd = cmd0;
|
|
|
|
|
if (cmd.startsWith("%"))
|
|
|
|
|
cmd = "1,$" + cmd.mid(1);
|
|
|
|
|
|
2009-01-06 11:37:24 +01:00
|
|
|
int beginLine = -1;
|
|
|
|
|
int endLine = -1;
|
2008-12-28 02:15:26 +01:00
|
|
|
|
|
|
|
|
int line = readLineCode(cmd);
|
|
|
|
|
if (line != -1)
|
|
|
|
|
beginLine = line;
|
|
|
|
|
|
|
|
|
|
if (cmd.startsWith(',')) {
|
|
|
|
|
cmd = cmd.mid(1);
|
|
|
|
|
line = readLineCode(cmd);
|
|
|
|
|
if (line != -1)
|
|
|
|
|
endLine = line;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//qDebug() << "RANGE: " << beginLine << endLine << cmd << cmd0;
|
|
|
|
|
|
2008-12-28 01:02:54 +01:00
|
|
|
static QRegExp reWrite("^w!?( (.*))?$");
|
2009-01-06 11:33:07 +01:00
|
|
|
static QRegExp reDelete("^d( (.*))?$");
|
2008-12-28 02:15:26 +01:00
|
|
|
|
|
|
|
|
if (cmd.isEmpty()) {
|
2009-01-06 11:33:07 +01:00
|
|
|
m_tc.setPosition(positionForLine(beginLine));
|
2008-12-28 01:02:54 +01:00
|
|
|
showMessage(QString());
|
2009-01-06 11:33:07 +01:00
|
|
|
} else if (cmd == "q!" || cmd == "q") { // :q
|
2008-12-28 02:32:44 +01:00
|
|
|
q->quitRequested(THE_EDITOR);
|
2008-12-28 01:02:54 +01:00
|
|
|
showMessage(QString());
|
2009-01-06 11:33:07 +01:00
|
|
|
} else if (reDelete.indexIn(cmd) != -1) { // :d
|
|
|
|
|
if (beginLine == -1)
|
|
|
|
|
beginLine = cursorLineInDocument();
|
|
|
|
|
if (endLine == -1)
|
|
|
|
|
endLine = cursorLineInDocument();
|
|
|
|
|
QTextCursor tc = selectRange(beginLine, endLine);
|
|
|
|
|
QString reg = reDelete.cap(2);
|
|
|
|
|
if (!reg.isEmpty())
|
|
|
|
|
m_registers[reg.at(0).unicode()] = tc.selection().toPlainText();
|
|
|
|
|
tc.removeSelectedText();
|
|
|
|
|
} else if (reWrite.indexIn(cmd) != -1) { // :w
|
|
|
|
|
if (beginLine == -1)
|
|
|
|
|
beginLine = 0;
|
|
|
|
|
if (endLine == -1)
|
|
|
|
|
endLine = linesInDocument();
|
2008-12-28 01:02:54 +01:00
|
|
|
bool forced = cmd.startsWith("w!");
|
|
|
|
|
QString fileName = reWrite.cap(2);
|
|
|
|
|
if (fileName.isEmpty())
|
|
|
|
|
fileName = m_currentFileName;
|
2008-12-27 19:47:29 +01:00
|
|
|
QFile file(fileName);
|
|
|
|
|
bool exists = file.exists();
|
|
|
|
|
if (exists && !forced) {
|
|
|
|
|
showMessage("E13: File exists (add ! to override)");
|
|
|
|
|
} else {
|
|
|
|
|
file.open(QIODevice::ReadWrite);
|
2009-01-06 11:33:07 +01:00
|
|
|
QTextCursor tc = selectRange(beginLine, endLine);
|
|
|
|
|
QString text = tc.selection().toPlainText();
|
2008-12-27 19:47:29 +01:00
|
|
|
QTextStream ts(&file);
|
2009-01-06 11:33:07 +01:00
|
|
|
ts << text;
|
2008-12-27 19:47:29 +01:00
|
|
|
file.close();
|
|
|
|
|
file.open(QIODevice::ReadWrite);
|
|
|
|
|
QByteArray ba = file.readAll();
|
|
|
|
|
showMessage(tr("\"%1\" %2 %3L, %4C written")
|
|
|
|
|
.arg(fileName).arg(exists ? " " : " [New] ")
|
|
|
|
|
.arg(ba.count('\n')).arg(ba.size()));
|
|
|
|
|
}
|
2008-12-27 16:42:07 +01:00
|
|
|
} else if (cmd.startsWith("r ")) {
|
2008-12-28 01:02:54 +01:00
|
|
|
m_currentFileName = cmd.mid(2);
|
|
|
|
|
QFile file(m_currentFileName);
|
2008-12-27 16:42:07 +01:00
|
|
|
file.open(QIODevice::ReadOnly);
|
|
|
|
|
QTextStream ts(&file);
|
2008-12-27 19:47:29 +01:00
|
|
|
EDITOR(setPlainText(ts.readAll()));
|
2008-12-28 01:02:54 +01:00
|
|
|
showMessage(QString());
|
2008-12-28 02:15:26 +01:00
|
|
|
} else {
|
|
|
|
|
showMessage("E492: Not an editor command: " + cmd0);
|
2008-12-27 16:42:07 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-26 00:18:03 +01:00
|
|
|
void FakeVimHandler::Private::search(const QString &needle, bool forward)
|
2008-12-19 12:20:04 +01:00
|
|
|
{
|
|
|
|
|
//qDebug() << "NEEDLE " << needle << "BACKWARDS" << backwards;
|
2008-12-25 13:20:09 +01:00
|
|
|
QTextCursor orig = m_tc;
|
2008-12-26 17:01:21 +01:00
|
|
|
QTextDocument::FindFlags flags = QTextDocument::FindCaseSensitively;
|
2008-12-26 00:18:03 +01:00
|
|
|
if (!forward)
|
2008-12-19 12:20:04 +01:00
|
|
|
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);
|
|
|
|
|
|
2008-12-26 18:29:38 +01:00
|
|
|
EDITOR(setTextCursor(m_tc));
|
|
|
|
|
if (EDITOR(find(needle, flags))) {
|
|
|
|
|
m_tc = EDITOR(textCursor());
|
2008-12-26 17:01:21 +01:00
|
|
|
// the qMax seems to be needed for QPlainTextEdit only
|
|
|
|
|
m_tc.movePosition(Left, MoveAnchor, qMax(1, needle.size() - 1));
|
2008-12-19 12:20:04 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-26 00:18:03 +01:00
|
|
|
m_tc.setPosition(forward ? 0 : lastPositionInDocument() - 1);
|
2008-12-26 18:29:38 +01:00
|
|
|
EDITOR(setTextCursor(m_tc));
|
|
|
|
|
if (EDITOR(find(needle, flags))) {
|
|
|
|
|
m_tc = EDITOR(textCursor());
|
2008-12-26 17:01:21 +01:00
|
|
|
// the qMax seems to be needed for QPlainTextEdit only
|
|
|
|
|
m_tc.movePosition(Left, MoveAnchor, qMax(1, needle.size() - 1));
|
2008-12-26 00:18:03 +01:00
|
|
|
if (forward)
|
2008-12-19 12:20:04 +01:00
|
|
|
showMessage("search hit BOTTOM, continuing at TOP");
|
2008-12-26 00:18:03 +01:00
|
|
|
else
|
|
|
|
|
showMessage("search hit TOP, continuing at BOTTOM");
|
2008-12-19 12:20:04 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-25 13:20:09 +01:00
|
|
|
m_tc = orig;
|
2008-12-19 12:20:04 +01:00
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-12-19 12:20:04 +01:00
|
|
|
|
2008-12-25 19:11:21 +01:00
|
|
|
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)
|
2008-12-25 19:11:21 +01:00
|
|
|
{
|
|
|
|
|
// FIXME: 'w' should stop on empty lines, too
|
2008-12-26 00:18:03 +01:00
|
|
|
int repeat = count();
|
2008-12-25 19:11:21 +01:00
|
|
|
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());
|
2008-12-25 19:11:21 +01:00
|
|
|
int thisClass = charClass(c, simple);
|
|
|
|
|
if (thisClass != lastClass && thisClass != 0)
|
|
|
|
|
--repeat;
|
2008-12-25 22:42:50 +01:00
|
|
|
if (repeat == 0)
|
2008-12-25 19:11:21 +01:00
|
|
|
break;
|
|
|
|
|
lastClass = thisClass;
|
|
|
|
|
m_tc.movePosition(Right, KeepAnchor, 1);
|
2008-12-25 20:12:17 +01:00
|
|
|
if (m_tc.position() == n)
|
|
|
|
|
break;
|
2008-12-25 19:11:21 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-25 16:27:47 +01:00
|
|
|
int FakeVimHandler::Private::cursorLineOnScreen() const
|
|
|
|
|
{
|
2009-01-06 11:11:31 +01:00
|
|
|
QRect rect = EDITOR(cursorRect());
|
|
|
|
|
return rect.y() / rect.height();
|
2008-12-25 16:27:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int FakeVimHandler::Private::linesOnScreen() const
|
|
|
|
|
{
|
2009-01-06 11:11:31 +01:00
|
|
|
QRect rect = EDITOR(cursorRect());
|
|
|
|
|
return EDITOR(height()) / rect.height();
|
2008-12-25 16:27:47 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-27 21:01:05 +01:00
|
|
|
int FakeVimHandler::Private::columnsOnScreen() const
|
|
|
|
|
{
|
2009-01-06 11:11:31 +01:00
|
|
|
QRect rect = EDITOR(cursorRect());
|
|
|
|
|
return EDITOR(width()) / rect.width();
|
2008-12-27 21:01:05 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-25 16:27:47 +01:00
|
|
|
int FakeVimHandler::Private::cursorLineInDocument() const
|
|
|
|
|
{
|
2009-01-06 11:11:31 +01:00
|
|
|
return m_tc.block().blockNumber();
|
2008-12-25 16:27:47 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-27 21:01:05 +01:00
|
|
|
int FakeVimHandler::Private::cursorColumnInDocument() const
|
|
|
|
|
{
|
2009-01-06 11:11:31 +01:00
|
|
|
return m_tc.position() - m_tc.block().position();
|
2008-12-27 21:01:05 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-28 02:15:26 +01:00
|
|
|
int FakeVimHandler::Private::linesInDocument() const
|
|
|
|
|
{
|
|
|
|
|
return m_tc.isNull() ? 0 : m_tc.document()->blockCount();
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-25 16:27:47 +01:00
|
|
|
void FakeVimHandler::Private::scrollToLineInDocument(int line)
|
|
|
|
|
{
|
2009-01-06 11:11:31 +01:00
|
|
|
// FIXME: works only for QPlainTextEdit
|
|
|
|
|
QScrollBar *scrollBar = EDITOR(verticalScrollBar());
|
|
|
|
|
scrollBar->setValue(line);
|
2008-12-25 16:27:47 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-25 19:11:21 +01:00
|
|
|
int FakeVimHandler::Private::lastPositionInDocument() const
|
|
|
|
|
{
|
|
|
|
|
QTextBlock block = m_tc.block().document()->lastBlock();
|
|
|
|
|
return block.position() + block.length();
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-26 17:01:21 +01:00
|
|
|
QString FakeVimHandler::Private::lastSearchString() const
|
|
|
|
|
{
|
|
|
|
|
return m_searchHistory.empty() ? QString() : m_searchHistory.back();
|
|
|
|
|
}
|
|
|
|
|
|
2009-01-06 11:33:07 +01:00
|
|
|
int FakeVimHandler::Private::positionForLine(int line) const
|
|
|
|
|
{
|
|
|
|
|
return m_tc.block().document()->findBlockByNumber(line - 1).position();
|
|
|
|
|
}
|
|
|
|
|
|
2009-01-06 11:42:44 +01:00
|
|
|
int FakeVimHandler::Private::lineForPosition(int pos) const
|
|
|
|
|
{
|
|
|
|
|
QTextCursor tc = m_tc;
|
|
|
|
|
tc.setPosition(pos);
|
|
|
|
|
return tc.block().blockNumber() + 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FakeVimHandler::Private::enterVisualLineMode()
|
|
|
|
|
{
|
|
|
|
|
m_visualLineMode = true;
|
|
|
|
|
m_marks['<'] = m_tc.position();
|
|
|
|
|
updateMiniBuffer();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FakeVimHandler::Private::leaveVisualLineMode()
|
|
|
|
|
{
|
|
|
|
|
m_visualLineMode = false;
|
|
|
|
|
m_marks['>'] = m_tc.position();
|
|
|
|
|
updateMiniBuffer();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-12-19 12:20:04 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// 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);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-27 16:42:07 +01:00
|
|
|
void FakeVimHandler::addWidget(QWidget *widget)
|
|
|
|
|
{
|
|
|
|
|
widget->installEventFilter(this);
|
|
|
|
|
QFont font = widget->font();
|
2008-12-27 21:01:05 +01:00
|
|
|
//: -misc-fixed-medium-r-semicondensed--13-120-75-75-c-60-iso8859-1
|
|
|
|
|
//font.setFamily("Misc");
|
2008-12-27 16:42:07 +01:00
|
|
|
font.setFamily("Monospace");
|
2008-12-27 21:01:05 +01:00
|
|
|
font.setStretch(QFont::SemiCondensed);
|
2008-12-27 16:42:07 +01:00
|
|
|
widget->setFont(font);
|
|
|
|
|
if (QTextEdit *ed = qobject_cast<QTextEdit *>(widget)) {
|
|
|
|
|
ed->setCursorWidth(QFontMetrics(ed->font()).width(QChar('x')));
|
|
|
|
|
ed->setLineWrapMode(QTextEdit::NoWrap);
|
|
|
|
|
} else if (QPlainTextEdit *ed = qobject_cast<QPlainTextEdit *>(widget)) {
|
|
|
|
|
ed->setCursorWidth(QFontMetrics(ed->font()).width(QChar('x')));
|
|
|
|
|
ed->setLineWrapMode(QPlainTextEdit::NoWrap);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FakeVimHandler::removeWidget(QWidget *widget)
|
|
|
|
|
{
|
|
|
|
|
widget->removeEventFilter(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FakeVimHandler::handleCommand(QWidget *widget, const QString &cmd)
|
|
|
|
|
{
|
|
|
|
|
d->m_textedit = qobject_cast<QTextEdit *>(widget);
|
|
|
|
|
d->m_plaintextedit = qobject_cast<QPlainTextEdit *>(widget);
|
|
|
|
|
d->handleCommand(cmd);
|
|
|
|
|
}
|
|
|
|
|
|