| /*************************************************************************** |
| * __________ __ ___. |
| * Open \______ \ ____ ____ | | _\_ |__ _______ ___ |
| * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / |
| * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < |
| * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ |
| * \/ \/ \/ \/ \/ |
| * $Id$ |
| * |
| * This file has been copied from Nokia's Qt Examples, with minor modifications |
| * made available under the LGPL version 2.1, as the original file was licensed |
| * |
| **************************************************************************** |
| **************************************************************************** |
| ** |
| ** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). |
| ** All rights reserved. |
| ** Contact: Nokia Corporation (qt-info@nokia.com) |
| ** |
| ** This file is part of the examples of the Qt Toolkit. |
| ** |
| ** GNU Lesser General Public License Usage |
| ** Alternatively, this file may be used under the terms of the GNU Lesser |
| ** General Public License version 2.1 as published by the Free Software |
| ** Foundation and appearing in the file LICENSE.LGPL included in the |
| ** packaging of this file. Please review the following information to |
| ** ensure the GNU Lesser General Public License version 2.1 requirements |
| ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. |
| ** |
| ** In addition, as a special exception, Nokia gives you certain additional |
| ** rights. These rights are described in the Nokia Qt LGPL Exception |
| ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. |
| ** |
| ****************************************************************************/ |
| |
| #include <QtGui> |
| #include <QApplication> |
| |
| #include "codeeditor.h" |
| |
| CodeEditor::CodeEditor(QWidget *parent) |
| : QPlainTextEdit(parent), completer(this) |
| { |
| lineNumberArea = new LineNumberArea(this); |
| |
| connect(this, SIGNAL(blockCountChanged(int)), |
| this, SLOT(updateLineNumberAreaWidth(int))); |
| connect(this, SIGNAL(updateRequest(QRect,int)), |
| this, SLOT(updateLineNumberArea(QRect,int))); |
| |
| updateLineNumberAreaWidth(0); |
| |
| QObject::connect(this, SIGNAL(cursorPositionChanged()), |
| this, SLOT(cursorMoved())); |
| completer.hide(); |
| settings.beginGroup("CodeEditor"); |
| } |
| |
| int CodeEditor::lineNumberAreaWidth() |
| { |
| int digits = 1; |
| int max = qMax(1, blockCount()); |
| while (max >= 10) { |
| max /= 10; |
| ++digits; |
| } |
| |
| int space = 3 + fontMetrics().width(QLatin1Char('9')) * digits; |
| |
| return space; |
| } |
| |
| |
| void CodeEditor::updateLineNumberAreaWidth(int /* newBlockCount */) |
| { |
| setViewportMargins(lineNumberAreaWidth(), 0, 0, 0); |
| } |
| |
| void CodeEditor::updateLineNumberArea(const QRect &rect, int dy) |
| { |
| if (dy) |
| lineNumberArea->scroll(0, dy); |
| else |
| lineNumberArea->update(0, rect.y(), lineNumberArea->width(), rect.height()); |
| |
| if (rect.contains(viewport()->rect())) |
| updateLineNumberAreaWidth(0); |
| } |
| |
| void CodeEditor::cursorMoved() |
| { |
| /* Closing the completer if the cursor has moved out of its bounds */ |
| if(completer.isVisible()) |
| { |
| if(document()->toPlainText().length() > docLength) |
| tagEnd++; |
| else if(document()->toPlainText().length() < docLength) |
| tagEnd--; |
| |
| if(textCursor().position() < tagBegin |
| || textCursor().position() > tagEnd) |
| { |
| completer.hide(); |
| } |
| } |
| } |
| |
| void CodeEditor::insertTag() |
| { |
| /* Clearing the typed tag and inserting one from the completer */ |
| QTextCursor at(document()); |
| at.setPosition(tagBegin, QTextCursor::MoveAnchor); |
| while(document()->characterAt(at.position()) == QChar('%') |
| || document()->characterAt(at.position()) == '?') |
| at.movePosition(QTextCursor::NextCharacter, QTextCursor::MoveAnchor, 1); |
| |
| at.movePosition(QTextCursor::NextCharacter, QTextCursor::KeepAnchor, |
| tagEnd - at.position()); |
| at.removeSelectedText(); |
| |
| at.insertText(completer.currentItem()->text(0)); |
| |
| completer.hide(); |
| } |
| |
| void CodeEditor::resizeEvent(QResizeEvent *e) |
| { |
| QPlainTextEdit::resizeEvent(e); |
| |
| QRect cr = contentsRect(); |
| lineNumberArea->setGeometry(QRect(cr.left(), cr.top(), |
| lineNumberAreaWidth(), cr.height())); |
| } |
| |
| void CodeEditor::keyPressEvent(QKeyEvent *event) |
| { |
| |
| if(!settings.value("completeSyntax", true).toBool()) |
| { |
| QPlainTextEdit::keyPressEvent(event); |
| return; |
| } |
| |
| if(completer.isVisible()) |
| { |
| /* Handling the completer */ |
| if(event->key() == Qt::Key_Up) |
| { |
| /* Up/down arrow presses get sent right along to the completer |
| * to navigate through the list |
| */ |
| if(completer.currentIndex().row() > 0) |
| QApplication::sendEvent(&completer, event); |
| } |
| else if(event->key() == Qt::Key_Down) |
| { |
| if(completer.currentIndex().row() |
| < completer.topLevelItemCount() - 1) |
| QApplication::sendEvent(&completer, event); |
| } |
| else if(event->key() == Qt::Key_Backspace |
| || event->key() == Qt::Key_Delete) |
| { |
| docLength = document()->toPlainText().length(); |
| QPlainTextEdit::keyPressEvent(event); |
| |
| QString filterText; |
| |
| for(int i = tagBegin; i < tagEnd; i++) |
| { |
| QChar c = document()->characterAt(i); |
| if(c != '%' && c != '?') |
| filterText.append(c); |
| } |
| |
| completer.filter(filterText); |
| } |
| else if(event->key() == Qt::Key_Escape) |
| { |
| /* Escape hides the completer */ |
| completer.hide(); |
| QPlainTextEdit::keyPressEvent(event); |
| } |
| else if(event->key() == Qt::Key_Return) |
| { |
| /* The enter key inserts the currently selected tag */ |
| insertTag(); |
| } |
| else if(event->key() == Qt::Key_Question) |
| { |
| /* The question mark doesn't filter the list */ |
| docLength = document()->toPlainText().length(); |
| QPlainTextEdit::keyPressEvent(event); |
| } |
| else if(event->key() == Qt::Key_Left |
| || event->key() == Qt::Key_Right) |
| { |
| /* Left and right keys shouldn't affect tagEnd */ |
| QPlainTextEdit::keyPressEvent(event); |
| } |
| else |
| { |
| /* Otherwise, we have to filter the list */ |
| docLength = document()->toPlainText().length(); |
| QPlainTextEdit::keyPressEvent(event); |
| |
| QString filterText; |
| |
| for(int i = tagBegin; i < tagEnd; i++) |
| { |
| QChar c = document()->characterAt(i); |
| if(c != '%' && c != '?') |
| filterText.append(c); |
| } |
| |
| completer.filter(filterText); |
| } |
| } |
| else |
| { |
| /* Deciding whether to show the completer */ |
| QPlainTextEdit::keyPressEvent(event); |
| if(event->key() == Qt::Key_Percent) |
| { |
| tagBegin = textCursor().position(); |
| tagEnd = textCursor().position(); |
| completer.filter(""); |
| completer.move(cursorRect().left(), cursorRect().bottom()); |
| if(completer.frameGeometry().right() > width()) |
| completer.move(width() - completer.width(), completer.y()); |
| if(completer.frameGeometry().bottom() > height()) |
| completer.move(completer.x(), |
| cursorRect().top() - completer.height()); |
| completer.show(); |
| } |
| } |
| |
| } |
| |
| void CodeEditor::lineNumberAreaPaintEvent(QPaintEvent *event) |
| { |
| QPainter painter(lineNumberArea); |
| painter.fillRect(event->rect(), Qt::lightGray); |
| |
| QTextBlock block = firstVisibleBlock(); |
| int blockNumber = block.blockNumber(); |
| int top = (int) blockBoundingGeometry(block).translated(contentOffset()).top(); |
| int bottom = top + (int) blockBoundingRect(block).height(); |
| |
| while (block.isValid() && top <= event->rect().bottom()) { |
| if (block.isVisible() && bottom >= event->rect().top()) { |
| QString number = QString::number(blockNumber + 1); |
| /* Drawing an error circle if necessary */ |
| if(errors.contains(blockNumber + 1)) |
| { |
| painter.fillRect(QRect(0, top, lineNumberArea->width(), |
| fontMetrics().height()), errorColor); |
| } |
| painter.setPen(Qt::black); |
| painter.drawText(0, top, lineNumberArea->width(), |
| fontMetrics().height(), Qt::AlignRight, number); |
| } |
| |
| block = block.next(); |
| top = bottom; |
| bottom = top + (int) blockBoundingRect(block).height(); |
| ++blockNumber; |
| } |
| } |
| |