/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** Commercial Usage ** ** Licensees holding valid Qt Commercial licenses may use this file in ** accordance with the Qt Commercial License Agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Nokia. ** ** 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. ** ** If you are unsure which license is appropriate for your use, please ** contact the sales department at http://qt.nokia.com/contact. ** **************************************************************************/ #include "cppquickfix.h" #include "cppeditor.h" #include #include #include #include #include #include #include using namespace CppEditor::Internal; using namespace CPlusPlus; namespace { class ASTPath: public ASTVisitor { Document::Ptr _doc; unsigned _line; unsigned _column; QList _nodes; public: ASTPath(Document::Ptr doc) : ASTVisitor(doc->control()), _doc(doc), _line(0), _column(0) {} QList operator()(const QTextCursor &cursor) { _nodes.clear(); _line = cursor.blockNumber() + 1; _column = cursor.columnNumber() + 1; accept(_doc->translationUnit()->ast()); return _nodes; } protected: virtual bool preVisit(AST *ast) { unsigned firstToken = ast->firstToken(); unsigned lastToken = ast->lastToken(); if (firstToken > 0 && lastToken > firstToken) { unsigned startLine, startColumn; getTokenStartPosition(firstToken, &startLine, &startColumn); if (_line > startLine || (_line == startLine && _column >= startColumn)) { unsigned endLine, endColumn; getTokenEndPosition(lastToken - 1, &endLine, &endColumn); if (_line < endLine || (_line == endLine && _column < endColumn)) { _nodes.append(ast); return true; } } } return false; } }; class HelloQuickFixOp: public QuickFixOperation { public: HelloQuickFixOp(Document::Ptr doc, const Snapshot &snapshot, const QTextCursor &textCursor) : QuickFixOperation(doc, snapshot, textCursor) {} virtual QString description() const { return QLatin1String("Hello"); // ### tr? } virtual void apply(QTextCursor) { // nothing to do. } }; } // end of anonymous namespace QuickFixOperation::QuickFixOperation(CPlusPlus::Document::Ptr doc, const CPlusPlus::Snapshot &snapshot, const QTextCursor &textCursor) : _doc(doc), _snapshot(snapshot), _textCursor(textCursor) { } QuickFixOperation::~QuickFixOperation() { } QTextCursor QuickFixOperation::textCursor() const { return _textCursor; } QTextCursor QuickFixOperation::cursor(AST *ast) const { TranslationUnit *unit = document()->translationUnit(); unsigned startLine, startColumn, endLine, endColumn; unit->getTokenStartPosition(ast->firstToken(), &startLine, &startColumn); unit->getTokenEndPosition(ast->lastToken() - 1, &endLine, &endColumn); QTextDocument *textDocument = _textCursor.document(); QTextCursor tc(textDocument); tc.setPosition(textDocument->findBlockByNumber(startLine - 1).position() + startColumn - 1); tc.setPosition(textDocument->findBlockByNumber(endLine - 1).position() + endColumn - 1, QTextCursor::KeepAnchor); return tc; } const CPlusPlus::Token &QuickFixOperation::tokenAt(unsigned index) const { return _doc->translationUnit()->tokenAt(index); } void QuickFixOperation::getTokenStartPosition(unsigned index, unsigned *line, unsigned *column) const { _doc->translationUnit()->getPosition(tokenAt(index).begin(), line, column); } void QuickFixOperation::getTokenEndPosition(unsigned index, unsigned *line, unsigned *column) const { _doc->translationUnit()->getPosition(tokenAt(index).end(), line, column); } QTextCursor QuickFixOperation::cursor(unsigned index) const { const Token &tk = tokenAt(index); unsigned line, col; getTokenStartPosition(index, &line, &col); QTextCursor tc = _textCursor; tc.setPosition(tc.document()->findBlockByNumber(line - 1).position() + col - 1); tc.setPosition(tc.position() + tk.f.length, QTextCursor::KeepAnchor); return tc; } QTextCursor QuickFixOperation::moveAtStartOfToken(unsigned index) const { unsigned line, col; getTokenStartPosition(index, &line, &col); QTextCursor tc = _textCursor; tc.setPosition(tc.document()->findBlockByNumber(line - 1).position() + col - 1); return tc; } QTextCursor QuickFixOperation::moveAtEndOfToken(unsigned index) const { const Token &tk = tokenAt(index); unsigned line, col; getTokenStartPosition(index, &line, &col); QTextCursor tc = _textCursor; tc.setPosition(tc.document()->findBlockByNumber(line - 1).position() + col + tk.f.length - 1); return tc; } CPPQuickFixCollector::CPPQuickFixCollector() : _modelManager(CppTools::CppModelManagerInterface::instance()), _editor(0) { } CPPQuickFixCollector::~CPPQuickFixCollector() { } bool CPPQuickFixCollector::supportsEditor(TextEditor::ITextEditable *editor) { return qobject_cast(editor) != 0; } bool CPPQuickFixCollector::triggersCompletion(TextEditor::ITextEditable *) { return false; } int CPPQuickFixCollector::startCompletion(TextEditor::ITextEditable *editable) { Q_ASSERT(editable != 0); _editor = qobject_cast(editable->widget()); Q_ASSERT(_editor != 0); const SemanticInfo info = _editor->semanticInfo(); QTextCursor textCursor = _editor->textCursor(); if (info.revision != _editor->document()->revision()) { // outdated qWarning() << "TODO: outdated semantic info, force a reparse."; return -1; } if (info.doc) { ASTPath astPath(info.doc); const QList path = astPath(_editor->textCursor()); // ### build the list of the quick fix ops by scanning path. if (! _quickFixes.isEmpty()) return editable->position(); } return -1; } void CPPQuickFixCollector::completions(QList *quickFixItems) { for (int i = 0; i < _quickFixes.size(); ++i) { QuickFixOperationPtr op = _quickFixes.at(i); TextEditor::CompletionItem item(this); item.text = op->description(); item.data = QVariant::fromValue(i); quickFixItems->append(item); } } void CPPQuickFixCollector::complete(const TextEditor::CompletionItem &item) { const int index = item.data.toInt(); if (index < _quickFixes.size()) { QuickFixOperationPtr quickFix = _quickFixes.at(index); quickFix->apply(_editor->textCursor()); } } void CPPQuickFixCollector::cleanup() { _quickFixes.clear(); }