forked from qt-creator/qt-creator
		
	Task-number: QTCREATORBUG-24098 Change-Id: I8649423fc69faadad324bc3bb51004633fa7cc4d Reviewed-by: hjk <hjk@qt.io>
		
			
				
	
	
		
			256 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/****************************************************************************
 | 
						|
**
 | 
						|
** Copyright (C) 2016 The Qt Company Ltd.
 | 
						|
** Contact: https://www.qt.io/licensing/
 | 
						|
**
 | 
						|
** This file is part of Qt Creator.
 | 
						|
**
 | 
						|
** Commercial License Usage
 | 
						|
** Licensees holding valid commercial Qt licenses may use this file in
 | 
						|
** accordance with the commercial license agreement provided with the
 | 
						|
** Software or, alternatively, in accordance with the terms contained in
 | 
						|
** a written agreement between you and The Qt Company. For licensing terms
 | 
						|
** and conditions see https://www.qt.io/terms-conditions. For further
 | 
						|
** information use the contact form at https://www.qt.io/contact-us.
 | 
						|
**
 | 
						|
** GNU General Public License Usage
 | 
						|
** Alternatively, this file may be used under the terms of the GNU
 | 
						|
** General Public License version 3 as published by the Free Software
 | 
						|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
 | 
						|
** included in the packaging of this file. Please review the following
 | 
						|
** information to ensure the GNU General Public License requirements will
 | 
						|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
 | 
						|
**
 | 
						|
****************************************************************************/
 | 
						|
 | 
						|
#include "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.connect([&](const QString &msg, int cursorPos, int, int) {
 | 
						|
        statusData.setStatusMessage(msg, cursorPos);
 | 
						|
        mainWindow.statusBar()->showMessage(statusData.currentStatusLine());
 | 
						|
    });
 | 
						|
 | 
						|
    handler.selectionChanged.connect([&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.connect([&](const QString &info) {
 | 
						|
        statusData.setStatusInfo(info);
 | 
						|
        mainWindow.statusBar()->showMessage(statusData.currentStatusLine());
 | 
						|
    });
 | 
						|
 | 
						|
    handler.statusDataChanged.connect([&](const QString &info) {
 | 
						|
        statusData.setStatusInfo(info);
 | 
						|
        mainWindow.statusBar()->showMessage(statusData.currentStatusLine());
 | 
						|
    });
 | 
						|
 | 
						|
    handler.highlightMatches.connect([&](const QString &needle) {
 | 
						|
        highlightMatches(handler.widget(), needle);
 | 
						|
    });
 | 
						|
 | 
						|
    handler.handleExCommandRequested.connect([](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();
 | 
						|
}
 |