/**************************************************************************** ** ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** 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, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "clangcompletion.h" #include "clangutils.h" #include "pchmanager.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static const bool DebugTiming = !qgetenv("QTC_CLANG_VERBOSE").isEmpty(); using namespace ClangCodeModel; using namespace ClangCodeModel::Internal; using namespace CPlusPlus; using namespace CppTools; using namespace TextEditor; static const char SNIPPET_ICON_PATH[] = ":/texteditor/images/snippet.png"; namespace { int activationSequenceChar(const QChar &ch, const QChar &ch2, const QChar &ch3, unsigned *kind, bool wantFunctionCall) { int referencePosition = 0; int completionKind = T_EOF_SYMBOL; switch (ch.toLatin1()) { case '.': if (ch2 != QLatin1Char('.')) { completionKind = T_DOT; referencePosition = 1; } break; case ',': completionKind = T_COMMA; referencePosition = 1; break; case '(': if (wantFunctionCall) { completionKind = T_LPAREN; referencePosition = 1; } break; case ':': if (ch3 != QLatin1Char(':') && ch2 == QLatin1Char(':')) { completionKind = T_COLON_COLON; referencePosition = 2; } break; case '>': if (ch2 == QLatin1Char('-')) { completionKind = T_ARROW; referencePosition = 2; } break; case '*': if (ch2 == QLatin1Char('.')) { completionKind = T_DOT_STAR; referencePosition = 2; } else if (ch3 == QLatin1Char('-') && ch2 == QLatin1Char('>')) { completionKind = T_ARROW_STAR; referencePosition = 3; } break; case '\\': case '@': if (ch2.isNull() || ch2.isSpace()) { completionKind = T_DOXY_COMMENT; referencePosition = 1; } break; case '<': completionKind = T_ANGLE_STRING_LITERAL; referencePosition = 1; break; case '"': completionKind = T_STRING_LITERAL; referencePosition = 1; break; case '/': completionKind = T_SLASH; referencePosition = 1; break; case '#': completionKind = T_POUND; referencePosition = 1; break; } if (kind) *kind = completionKind; return referencePosition; } static QList unfilteredCompletion(const ClangCompletionAssistInterface* interface, const QString &fileName, unsigned line, unsigned column, QByteArray modifiedInput = QByteArray(), bool isSignalSlotCompletion = false) { ClangCompleter::Ptr wrapper = interface->clangWrapper(); QMutexLocker lock(wrapper->mutex()); //### TODO: check if we're cancelled after we managed to acquire the mutex wrapper->setFileName(fileName); wrapper->setOptions(interface->options()); wrapper->setSignalSlotCompletion(isSignalSlotCompletion); UnsavedFiles unsavedFiles = interface->unsavedFiles(); if (!modifiedInput.isEmpty()) unsavedFiles.insert(fileName, modifiedInput); QTime t; if (DebugTiming) { qDebug() << "Here we go with ClangCompletionAssistProcessor...."; t.start(); } QList result = wrapper->codeCompleteAt(line, column + 1, unsavedFiles); qSort(result); if (DebugTiming) qDebug() << "... Completion done in" << t.elapsed() << "ms, with" << result.count() << "items."; return result; } } // Anonymous namespace ClangCodeModel { namespace Internal { // ----------------------------- // ClangCompletionAssistProvider // ----------------------------- ClangCompletionAssistProvider::ClangCompletionAssistProvider() : m_clangCompletionWrapper(new ClangCodeModel::ClangCompleter) { } IAssistProcessor *ClangCompletionAssistProvider::createProcessor() const { return new ClangCompletionAssistProcessor; } IAssistInterface *ClangCompletionAssistProvider::createAssistInterface( ProjectExplorer::Project *project, TextEditor::BaseTextEditor *editor, QTextDocument *document, int position, AssistReason reason) const { Q_UNUSED(project); QString fileName = editor->document()->filePath(); CppModelManagerInterface *modelManager = CppModelManagerInterface::instance(); QList parts = modelManager->projectPart(fileName); if (parts.isEmpty()) parts += modelManager->fallbackProjectPart(); QStringList includePaths, frameworkPaths, options; PchInfo::Ptr pchInfo; foreach (ProjectPart::Ptr part, parts) { if (part.isNull()) continue; options = ClangCodeModel::Utils::createClangOptions(part, fileName); pchInfo = PchManager::instance()->pchInfo(part); if (!pchInfo.isNull()) options.append(ClangCodeModel::Utils::createPCHInclusionOptions(pchInfo->fileName())); includePaths = part->includePaths; frameworkPaths = part->frameworkPaths; break; } return new ClangCodeModel::ClangCompletionAssistInterface( m_clangCompletionWrapper, document, position, fileName, reason, options, includePaths, frameworkPaths, pchInfo); } // ------------------------ // ClangAssistProposalModel // ------------------------ class ClangAssistProposalModel : public TextEditor::BasicProposalItemListModel { public: ClangAssistProposalModel() : TextEditor::BasicProposalItemListModel() , m_sortable(false) , m_completionOperator(T_EOF_SYMBOL) , m_replaceDotForArrow(false) {} virtual bool isSortable(const QString &prefix) const; bool m_sortable; unsigned m_completionOperator; bool m_replaceDotForArrow; }; // ------------------- // ClangAssistProposal // ------------------- class ClangAssistProposal : public TextEditor::GenericProposal { public: ClangAssistProposal(int cursorPos, TextEditor::IGenericProposalModel *model) : TextEditor::GenericProposal(cursorPos, model) , m_replaceDotForArrow(static_cast(model)->m_replaceDotForArrow) {} virtual bool isCorrective() const { return m_replaceDotForArrow; } virtual void makeCorrection(BaseTextEditor *editor) { editor->setCursorPosition(basePosition() - 1); editor->replace(1, QLatin1String("->")); moveBasePosition(1); } private: bool m_replaceDotForArrow; }; // ---------------------- // ClangFunctionHintModel // ---------------------- class ClangFunctionHintModel : public TextEditor::IFunctionHintProposalModel { public: ClangFunctionHintModel(const QList functionSymbols) : m_functionSymbols(functionSymbols) , m_currentArg(-1) {} virtual void reset() {} virtual int size() const { return m_functionSymbols.size(); } virtual QString text(int index) const; virtual int activeArgument(const QString &prefix) const; private: QList m_functionSymbols; mutable int m_currentArg; }; QString ClangFunctionHintModel::text(int index) const { #if 0 // TODO: add the boldening to the result Overview overview; overview.setShowReturnTypes(true); overview.setShowArgumentNames(true); overview.setMarkedArgument(m_currentArg + 1); Function *f = m_functionSymbols.at(index); const QString prettyMethod = overview(f->type(), f->name()); const int begin = overview.markedArgumentBegin(); const int end = overview.markedArgumentEnd(); QString hintText; hintText += Qt::escape(prettyMethod.left(begin)); hintText += ""; hintText += Qt::escape(prettyMethod.mid(begin, end - begin)); hintText += ""; hintText += Qt::escape(prettyMethod.mid(end)); return hintText; #endif return m_functionSymbols.at(index).hint(); } int ClangFunctionHintModel::activeArgument(const QString &prefix) const { int argnr = 0; int parcount = 0; SimpleLexer tokenize; QList tokens = tokenize(prefix); for (int i = 0; i < tokens.count(); ++i) { const CPlusPlus::Token &tk = tokens.at(i); if (tk.is(T_LPAREN)) ++parcount; else if (tk.is(T_RPAREN)) --parcount; else if (! parcount && tk.is(T_COMMA)) ++argnr; } if (parcount < 0) return -1; if (argnr != m_currentArg) m_currentArg = argnr; return argnr; } class ClangAssistProposalItem : public TextEditor::BasicProposalItem { public: ClangAssistProposalItem() {} virtual bool prematurelyApplies(const QChar &c) const; virtual void applyContextualContent(TextEditor::BaseTextEditor *editor, int basePosition) const; void keepCompletionOperator(unsigned compOp) { m_completionOperator = compOp; } bool isOverloaded() const { return !m_overloads.isEmpty(); } void addOverload(const CodeCompletionResult &ccr) { m_overloads.append(ccr); } CodeCompletionResult originalItem() const { const QVariant &v = data(); if (v.canConvert()) return v.value(); else return CodeCompletionResult(); } bool isCodeCompletionResult() const { return data().canConvert(); } private: unsigned m_completionOperator; mutable QChar m_typedChar; QList m_overloads; }; /// @return True, because clang always returns priorities for sorting bool ClangAssistProposalModel::isSortable(const QString &prefix) const { Q_UNUSED(prefix) return true; } } // namespace Internal } // namespace ClangCodeModel bool ClangAssistProposalItem::prematurelyApplies(const QChar &typedChar) const { bool ok = false; if (m_completionOperator == T_SIGNAL || m_completionOperator == T_SLOT) ok = QString::fromLatin1("(,").contains(typedChar); else if (m_completionOperator == T_STRING_LITERAL || m_completionOperator == T_ANGLE_STRING_LITERAL) ok = (typedChar == QLatin1Char('/')) && text().endsWith(QLatin1Char('/')); else if (!isCodeCompletionResult()) ok = (typedChar == QLatin1Char('(')); /* && data().canConvert()*/ //### else if (originalItem().completionKind() == CodeCompletionResult::ObjCMessageCompletionKind) ok = QString::fromLatin1(";.,").contains(typedChar); else ok = QString::fromLatin1(";.,:(").contains(typedChar); if (ok) m_typedChar = typedChar; return ok; } void ClangAssistProposalItem::applyContextualContent(TextEditor::BaseTextEditor *editor, int basePosition) const { const CodeCompletionResult ccr = originalItem(); QString toInsert = text(); QString extraChars; int extraLength = 0; int cursorOffset = 0; bool autoParenthesesEnabled = true; if (m_completionOperator == T_SIGNAL || m_completionOperator == T_SLOT) { extraChars += QLatin1Char(')'); if (m_typedChar == QLatin1Char('(')) // Eat the opening parenthesis m_typedChar = QChar(); } else if (m_completionOperator == T_STRING_LITERAL || m_completionOperator == T_ANGLE_STRING_LITERAL) { if (!toInsert.endsWith(QLatin1Char('/'))) { extraChars += QLatin1Char((m_completionOperator == T_ANGLE_STRING_LITERAL) ? '>' : '"'); } else { if (m_typedChar == QLatin1Char('/')) // Eat the slash m_typedChar = QChar(); } } else if (ccr.isValid()) { const CompletionSettings &completionSettings = TextEditorSettings::instance()->completionSettings(); const bool autoInsertBrackets = completionSettings.m_autoInsertBrackets; if (autoInsertBrackets && (ccr.completionKind() == CodeCompletionResult::FunctionCompletionKind || ccr.completionKind() == CodeCompletionResult::DestructorCompletionKind || ccr.completionKind() == CodeCompletionResult::SignalCompletionKind || ccr.completionKind() == CodeCompletionResult::SlotCompletionKind)) { // When the user typed the opening parenthesis, he'll likely also type the closing one, // in which case it would be annoying if we put the cursor after the already automatically // inserted closing parenthesis. const bool skipClosingParenthesis = m_typedChar != QLatin1Char('('); if (completionSettings.m_spaceAfterFunctionName) extraChars += QLatin1Char(' '); extraChars += QLatin1Char('('); if (m_typedChar == QLatin1Char('(')) m_typedChar = QChar(); // If the function doesn't return anything, automatically place the semicolon, // unless we're doing a scope completion (then it might be function definition). const QChar characterAtCursor = editor->textDocument()->characterAt(editor->position()); bool endWithSemicolon = m_typedChar == QLatin1Char(';')/* || (function->returnType()->isVoidType() && m_completionOperator != T_COLON_COLON)*/; //### const QChar semicolon = m_typedChar.isNull() ? QLatin1Char(';') : m_typedChar; if (endWithSemicolon && characterAtCursor == semicolon) { endWithSemicolon = false; m_typedChar = QChar(); } // If the function takes no arguments, automatically place the closing parenthesis if (!isOverloaded() && !ccr.hasParameters() && skipClosingParenthesis) { extraChars += QLatin1Char(')'); if (endWithSemicolon) { extraChars += semicolon; m_typedChar = QChar(); } } else if (autoParenthesesEnabled) { const QChar lookAhead = editor->textDocument()->characterAt(editor->position() + 1); if (MatchingText::shouldInsertMatchingText(lookAhead)) { extraChars += QLatin1Char(')'); --cursorOffset; if (endWithSemicolon) { extraChars += semicolon; --cursorOffset; m_typedChar = QChar(); } } } } #if 0 if (autoInsertBrackets && data().canConvert()) { if (m_typedChar == QLatin1Char('(')) m_typedChar = QChar(); // everything from the closing parenthesis on are extra chars, to // make sure an auto-inserted ")" gets replaced by ") const" if necessary int closingParen = toInsert.lastIndexOf(QLatin1Char(')')); extraChars = toInsert.mid(closingParen); toInsert.truncate(closingParen); } #endif } // Append an unhandled typed character, adjusting cursor offset when it had been adjusted before if (!m_typedChar.isNull()) { extraChars += m_typedChar; if (cursorOffset != 0) --cursorOffset; } // Avoid inserting characters that are already there const int endsPosition = editor->position(TextEditor::ITextEditor::EndOfLine); const QString existingText = editor->textDocument()->textAt(editor->position(), endsPosition - editor->position()); int existLength = 0; if (!existingText.isEmpty()) { // Calculate the exist length in front of the extra chars existLength = toInsert.length() - (editor->position() - basePosition); while (!existingText.startsWith(toInsert.right(existLength))) { if (--existLength == 0) break; } } for (int i = 0; i < extraChars.length(); ++i) { const QChar a = extraChars.at(i); const QChar b = editor->textDocument()->characterAt(editor->position() + i + existLength); if (a == b) ++extraLength; else break; } toInsert += extraChars; // Insert the remainder of the name const int length = editor->position() - basePosition + existLength + extraLength; editor->setCursorPosition(basePosition); editor->replace(length, toInsert); if (cursorOffset) editor->setCursorPosition(editor->position() + cursorOffset); } bool ClangCompletionAssistInterface::objcEnabled() const { return m_clangWrapper->objcEnabled(); } ClangCompletionAssistInterface::ClangCompletionAssistInterface(ClangCompleter::Ptr clangWrapper, QTextDocument *document, int position, const QString &fileName, AssistReason reason, const QStringList &options, const QStringList &includePaths, const QStringList &frameworkPaths, const PchInfo::Ptr &pchInfo) : DefaultAssistInterface(document, position, fileName, reason) , m_clangWrapper(clangWrapper) , m_options(options) , m_includePaths(includePaths) , m_frameworkPaths(frameworkPaths) , m_savedPchPointer(pchInfo) { Q_ASSERT(!clangWrapper.isNull()); CppModelManagerInterface *mmi = CppModelManagerInterface::instance(); Q_ASSERT(mmi); m_unsavedFiles = Utils::createUnsavedFiles(mmi->workingCopy()); } ClangCompletionAssistProcessor::ClangCompletionAssistProcessor() : m_preprocessorCompletions(QStringList() << QLatin1String("define") << QLatin1String("error") << QLatin1String("include") << QLatin1String("line") << QLatin1String("pragma") << QLatin1String("pragma once") << QLatin1String("pragma omp atomic") << QLatin1String("pragma omp parallel") << QLatin1String("pragma omp for") << QLatin1String("pragma omp ordered") << QLatin1String("pragma omp parallel for") << QLatin1String("pragma omp section") << QLatin1String("pragma omp sections") << QLatin1String("pragma omp parallel sections") << QLatin1String("pragma omp single") << QLatin1String("pragma omp master") << QLatin1String("pragma omp critical") << QLatin1String("pragma omp barrier") << QLatin1String("pragma omp flush") << QLatin1String("pragma omp threadprivate") << QLatin1String("undef") << QLatin1String("if") << QLatin1String("ifdef") << QLatin1String("ifndef") << QLatin1String("elif") << QLatin1String("else") << QLatin1String("endif")) , m_model(new ClangAssistProposalModel) , m_hintProposal(0) { } ClangCompletionAssistProcessor::~ClangCompletionAssistProcessor() { } IAssistProposal *ClangCompletionAssistProcessor::perform(const IAssistInterface *interface) { m_interface.reset(static_cast(interface)); if (interface->reason() != ExplicitlyInvoked && !accepts()) return 0; int index = startCompletionHelper(); if (index != -1) { if (m_hintProposal) return m_hintProposal; m_model->m_sortable = (m_model->m_completionOperator != T_EOF_SYMBOL); return createContentProposal(); } return 0; } int ClangCompletionAssistProcessor::startCompletionHelper() { //### TODO: clean-up this method, some calculated values might not be used anymore. Q_ASSERT(m_model); const int startOfName = findStartOfName(); m_startPosition = startOfName; m_model->m_completionOperator = T_EOF_SYMBOL; int endOfOperator = m_startPosition; // Skip whitespace preceding this position while (m_interface->characterAt(endOfOperator - 1).isSpace()) --endOfOperator; const QString fileName = m_interface->fileName(); int endOfExpression = startOfOperator(endOfOperator, &m_model->m_completionOperator, /*want function call =*/ true); if (m_model->m_completionOperator == T_EOF_SYMBOL) { endOfOperator = m_startPosition; } else if (m_model->m_completionOperator == T_DOXY_COMMENT) { for (int i = 1; i < T_DOXY_LAST_TAG; ++i) addCompletionItem(QString::fromLatin1(doxygenTagSpell(i)), m_icons.keywordIcon()); return m_startPosition; } // Pre-processor completion //### TODO: check if clang can do pp completion if (m_model->m_completionOperator == T_POUND) { completePreprocessor(); m_startPosition = startOfName; return m_startPosition; } // Include completion if (m_model->m_completionOperator == T_STRING_LITERAL || m_model->m_completionOperator == T_ANGLE_STRING_LITERAL || m_model->m_completionOperator == T_SLASH) { QTextCursor c(m_interface->textDocument()); c.setPosition(endOfExpression); if (completeInclude(c)) m_startPosition = startOfName; return m_startPosition; } ExpressionUnderCursor expressionUnderCursor; QTextCursor tc(m_interface->textDocument()); if (m_model->m_completionOperator == T_COMMA) { tc.setPosition(endOfExpression); const int start = expressionUnderCursor.startOfFunctionCall(tc); if (start == -1) { m_model->m_completionOperator = T_EOF_SYMBOL; return -1; } endOfExpression = start; m_startPosition = start + 1; m_model->m_completionOperator = T_LPAREN; } QString expression; int startOfExpression = m_interface->position(); tc.setPosition(endOfExpression); if (m_model->m_completionOperator) { expression = expressionUnderCursor(tc); startOfExpression = endOfExpression - expression.length(); if (m_model->m_completionOperator == T_LPAREN) { if (expression.endsWith(QLatin1String("SIGNAL"))) m_model->m_completionOperator = T_SIGNAL; else if (expression.endsWith(QLatin1String("SLOT"))) m_model->m_completionOperator = T_SLOT; else if (m_interface->position() != endOfOperator) { // We don't want a function completion when the cursor isn't at the opening brace expression.clear(); m_model->m_completionOperator = T_EOF_SYMBOL; m_startPosition = startOfName; startOfExpression = m_interface->position(); } } } else if (expression.isEmpty()) { while (startOfExpression > 0 && m_interface->characterAt(startOfExpression).isSpace()) --startOfExpression; } int line = 0, column = 0; // Convenience::convertPosition(m_interface->document(), startOfExpression, &line, &column); Convenience::convertPosition(m_interface->textDocument(), endOfOperator, &line, &column); return startCompletionInternal(fileName, line, column, endOfOperator); } int ClangCompletionAssistProcessor::startOfOperator(int pos, unsigned *kind, bool wantFunctionCall) const { const QChar ch = pos > -1 ? m_interface->characterAt(pos - 1) : QChar(); const QChar ch2 = pos > 0 ? m_interface->characterAt(pos - 2) : QChar(); const QChar ch3 = pos > 1 ? m_interface->characterAt(pos - 3) : QChar(); int start = pos - activationSequenceChar(ch, ch2, ch3, kind, wantFunctionCall); if (start != pos) { QTextCursor tc(m_interface->textDocument()); tc.setPosition(pos); // Include completion: make sure the quote character is the first one on the line if (*kind == T_STRING_LITERAL) { QTextCursor s = tc; s.movePosition(QTextCursor::StartOfLine, QTextCursor::KeepAnchor); QString sel = s.selectedText(); if (sel.indexOf(QLatin1Char('"')) < sel.length() - 1) { *kind = T_EOF_SYMBOL; start = pos; } } if (*kind == T_COMMA) { ExpressionUnderCursor expressionUnderCursor; if (expressionUnderCursor.startOfFunctionCall(tc) == -1) { *kind = T_EOF_SYMBOL; start = pos; } } SimpleLexer tokenize; LanguageFeatures lf = tokenize.languageFeatures(); lf.qtMocRunEnabled = true; lf.objCEnabled = true; tokenize.setLanguageFeatures(lf); tokenize.setSkipComments(false); const QList &tokens = tokenize(tc.block().text(), BackwardsScanner::previousBlockState(tc.block())); const int tokenIdx = SimpleLexer::tokenBefore(tokens, qMax(0, tc.positionInBlock() - 1)); // get the token at the left of the cursor const CPlusPlus::Token tk = (tokenIdx == -1) ? CPlusPlus::Token() : tokens.at(tokenIdx); if (*kind == T_DOXY_COMMENT && !(tk.is(T_DOXY_COMMENT) || tk.is(T_CPP_DOXY_COMMENT))) { *kind = T_EOF_SYMBOL; start = pos; } // Don't complete in comments or strings, but still check for include completion else if (tk.is(T_COMMENT) || tk.is(T_CPP_COMMENT) || (tk.isLiteral() && (*kind != T_STRING_LITERAL && *kind != T_ANGLE_STRING_LITERAL && *kind != T_SLASH))) { *kind = T_EOF_SYMBOL; start = pos; } // Include completion: can be triggered by slash, but only in a string else if (*kind == T_SLASH && (tk.isNot(T_STRING_LITERAL) && tk.isNot(T_ANGLE_STRING_LITERAL))) { *kind = T_EOF_SYMBOL; start = pos; } else if (*kind == T_LPAREN) { if (tokenIdx > 0) { const CPlusPlus::Token &previousToken = tokens.at(tokenIdx - 1); // look at the token at the left of T_LPAREN switch (previousToken.kind()) { case T_IDENTIFIER: case T_GREATER: case T_SIGNAL: case T_SLOT: break; // good default: // that's a bad token :) *kind = T_EOF_SYMBOL; start = pos; } } } // Check for include preprocessor directive else if (*kind == T_STRING_LITERAL || *kind == T_ANGLE_STRING_LITERAL || *kind == T_SLASH) { bool include = false; if (tokens.size() >= 3) { if (tokens.at(0).is(T_POUND) && tokens.at(1).is(T_IDENTIFIER) && (tokens.at(2).is(T_STRING_LITERAL) || tokens.at(2).is(T_ANGLE_STRING_LITERAL))) { const CPlusPlus::Token &directiveToken = tokens.at(1); QString directive = tc.block().text().mid(directiveToken.begin(), directiveToken.length()); if (directive == QLatin1String("include") || directive == QLatin1String("include_next") || directive == QLatin1String("import")) { include = true; } } } if (!include) { *kind = T_EOF_SYMBOL; start = pos; } } } return start; } int ClangCompletionAssistProcessor::findStartOfName(int pos) const { if (pos == -1) pos = m_interface->position(); QChar chr; // Skip to the start of a name do { chr = m_interface->characterAt(--pos); } while (chr.isLetterOrNumber() || chr == QLatin1Char('_')); return pos + 1; } bool ClangCompletionAssistProcessor::accepts() const { const int pos = m_interface->position(); unsigned token = T_EOF_SYMBOL; const int start = startOfOperator(pos, &token, /*want function call=*/ true); if (start != pos) { if (token == T_POUND) { const int column = pos - m_interface->textDocument()->findBlock(start).position(); if (column != 1) return false; } return true; } else { // Trigger completion after three characters of a name have been typed, when not editing an existing name QChar characterUnderCursor = m_interface->characterAt(pos); if (!characterUnderCursor.isLetterOrNumber() && characterUnderCursor != QLatin1Char('_')) { const int startOfName = findStartOfName(pos); if (pos - startOfName >= 3) { const QChar firstCharacter = m_interface->characterAt(startOfName); if (firstCharacter.isLetter() || firstCharacter == QLatin1Char('_')) { // Finally check that we're not inside a comment or string (code copied from startOfOperator) QTextCursor tc(m_interface->textDocument()); tc.setPosition(pos); SimpleLexer tokenize; LanguageFeatures lf = tokenize.languageFeatures(); lf.qtMocRunEnabled = true; lf.objCEnabled = true; tokenize.setLanguageFeatures(lf); tokenize.setSkipComments(false); const QList &tokens = tokenize(tc.block().text(), BackwardsScanner::previousBlockState(tc.block())); const int tokenIdx = SimpleLexer::tokenBefore(tokens, qMax(0, tc.positionInBlock() - 1)); const CPlusPlus::Token tk = (tokenIdx == -1) ? CPlusPlus::Token() : tokens.at(tokenIdx); if (!tk.isComment() && !tk.isLiteral()) { return true; } else if (tk.isLiteral() && tokens.size() == 3 && tokens.at(0).kind() == T_POUND && tokens.at(1).kind() == T_IDENTIFIER) { const QString &line = tc.block().text(); const CPlusPlus::Token &idToken = tokens.at(1); const QStringRef &identifier = line.midRef(idToken.begin(), idToken.end() - idToken.begin()); if (identifier == QLatin1String("include") || identifier == QLatin1String("include_next") || (m_interface->objcEnabled() && identifier == QLatin1String("import"))) { return true; } } } } } } return false; } IAssistProposal *ClangCompletionAssistProcessor::createContentProposal() { m_model->loadContent(m_completions); return new ClangAssistProposal(m_startPosition, m_model.take()); } /// Seach backwards in the document starting from pos to find the first opening /// parenthesis. Nested parenthesis are skipped. static int findOpenParen(QTextDocument *doc, int start) { unsigned parenCount = 1; for (int pos = start; pos >= 0; --pos) { const QChar ch = doc->characterAt(pos); if (ch == QLatin1Char('(')) { --parenCount; if (parenCount == 0) return pos; } else if (ch == QLatin1Char(')')) { ++parenCount; } } return -1; } static QByteArray modifyInput(QTextDocument *doc, int endOfExpression) { int comma = endOfExpression; while (comma > 0) { const QChar ch = doc->characterAt(comma); if (ch == QLatin1Char(',')) break; if (ch == QLatin1Char(';') || ch == QLatin1Char('{') || ch == QLatin1Char('}')) { // Safety net: we don't seem to have "connect(pointer, SIGNAL(" as // input, so stop searching. comma = -1; break; } --comma; } if (comma < 0) return QByteArray(); const int openBrace = findOpenParen(doc, comma); if (openBrace < 0) return QByteArray(); QByteArray modifiedInput = doc->toPlainText().toUtf8(); const int len = endOfExpression - comma; QByteArray replacement(len - 4, ' '); replacement.append(")->"); modifiedInput.replace(comma, len, replacement); modifiedInput.insert(openBrace, '('); return modifiedInput; } int ClangCompletionAssistProcessor::startCompletionInternal(const QString fileName, unsigned line, unsigned column, int endOfExpression) { bool signalCompletion = false; bool slotCompletion = false; QByteArray modifiedInput; if (m_model->m_completionOperator == T_SIGNAL) { signalCompletion = true; modifiedInput = modifyInput(m_interface->textDocument(), endOfExpression); } else if (m_model->m_completionOperator == T_SLOT) { slotCompletion = true; modifiedInput = modifyInput(m_interface->textDocument(), endOfExpression); } else if (m_model->m_completionOperator == T_LPAREN) { // Find the expression that precedes the current name int index = endOfExpression; while (m_interface->characterAt(index - 1).isSpace()) --index; QTextCursor tc(m_interface->textDocument()); tc.setPosition(index); ExpressionUnderCursor euc; index = euc.startOfFunctionCall(tc); int nameStart = findStartOfName(index); QTextCursor tc2(m_interface->textDocument()); tc2.setPosition(nameStart); tc2.setPosition(index, QTextCursor::KeepAnchor); const QString functionName = tc2.selectedText().trimmed(); int l = line, c = column; Convenience::convertPosition(m_interface->textDocument(), nameStart, &l, &c); if (DebugTiming) qDebug()<<"complete constructor or function @" << line<<":"<"< completions = unfilteredCompletion( m_interface.data(), fileName, l, c, QByteArray(), signalCompletion || slotCompletion); QList functionCompletions; foreach (const CodeCompletionResult &ccr, completions) { if (ccr.completionKind() == CodeCompletionResult::FunctionCompletionKind || ccr.completionKind() == CodeCompletionResult::ConstructorCompletionKind || ccr.completionKind() == CodeCompletionResult::DestructorCompletionKind || ccr.completionKind() == CodeCompletionResult::SignalCompletionKind || ccr.completionKind() == CodeCompletionResult::SlotCompletionKind) if (ccr.text() == functionName) functionCompletions.append(ccr); } if (!functionCompletions.isEmpty()) { IFunctionHintProposalModel *model = new ClangFunctionHintModel(functionCompletions); m_hintProposal = new FunctionHintProposal(m_startPosition, model); return m_startPosition; } } const QIcon snippetIcon = QIcon(QLatin1String(SNIPPET_ICON_PATH)); QList completions = unfilteredCompletion( m_interface.data(), fileName, line, column, modifiedInput, signalCompletion || slotCompletion); QHash items; foreach (const CodeCompletionResult &ccr, completions) { if (!ccr.isValid()) continue; if (signalCompletion && ccr.completionKind() != CodeCompletionResult::SignalCompletionKind) continue; if (slotCompletion && ccr.completionKind() != CodeCompletionResult::SlotCompletionKind) continue; const QString txt(ccr.text()); ClangAssistProposalItem *item = items.value(txt, 0); if (item) { item->addOverload(ccr); } else { item = new ClangAssistProposalItem; items.insert(txt, item); item->setText(txt); item->setDetail(ccr.hint()); item->setOrder(ccr.priority()); const QString snippet = ccr.snippet(); if (!snippet.isEmpty()) item->setData(snippet); else item->setData(qVariantFromValue(ccr)); } // FIXME: show the effective accessebility instead of availability switch (ccr.completionKind()) { case CodeCompletionResult::ClassCompletionKind: item->setIcon(m_icons.iconForType(Icons::ClassIconType)); break; case CodeCompletionResult::EnumCompletionKind: item->setIcon(m_icons.iconForType(Icons::EnumIconType)); break; case CodeCompletionResult::EnumeratorCompletionKind: item->setIcon(m_icons.iconForType(Icons::EnumeratorIconType)); break; case CodeCompletionResult::ConstructorCompletionKind: // fall through case CodeCompletionResult::DestructorCompletionKind: // fall through case CodeCompletionResult::FunctionCompletionKind: case CodeCompletionResult::ObjCMessageCompletionKind: switch (ccr.availability()) { case CodeCompletionResult::Available: case CodeCompletionResult::Deprecated: item->setIcon(m_icons.iconForType(Icons::FuncPublicIconType)); break; default: item->setIcon(m_icons.iconForType(Icons::FuncPrivateIconType)); break; } break; case CodeCompletionResult::SignalCompletionKind: item->setIcon(m_icons.iconForType(Icons::SignalIconType)); break; case CodeCompletionResult::SlotCompletionKind: switch (ccr.availability()) { case CodeCompletionResult::Available: case CodeCompletionResult::Deprecated: item->setIcon(m_icons.iconForType(Icons::SlotPublicIconType)); break; case CodeCompletionResult::NotAccessible: case CodeCompletionResult::NotAvailable: item->setIcon(m_icons.iconForType(Icons::SlotPrivateIconType)); break; } break; case CodeCompletionResult::NamespaceCompletionKind: item->setIcon(m_icons.iconForType(Icons::NamespaceIconType)); break; case CodeCompletionResult::PreProcessorCompletionKind: item->setIcon(m_icons.iconForType(Icons::MacroIconType)); break; case CodeCompletionResult::VariableCompletionKind: switch (ccr.availability()) { case CodeCompletionResult::Available: case CodeCompletionResult::Deprecated: item->setIcon(m_icons.iconForType(Icons::VarPublicIconType)); break; default: item->setIcon(m_icons.iconForType(Icons::VarPrivateIconType)); break; } break; case CodeCompletionResult::KeywordCompletionKind: item->setIcon(m_icons.iconForType(Icons::KeywordIconType)); break; case CodeCompletionResult::ClangSnippetKind: item->setIcon(snippetIcon); break; default: break; } } foreach (ClangAssistProposalItem *item, items.values()) m_completions.append(item); return m_startPosition; } /** * @brief Creates completion proposals for #include and given cursor * @param cursor - cursor placed after opening bracked or quote * @return false if completions list is empty */ bool ClangCompletionAssistProcessor::completeInclude(const QTextCursor &cursor) { QString directoryPrefix; if (m_model->m_completionOperator == T_SLASH) { QTextCursor c = cursor; c.movePosition(QTextCursor::StartOfLine, QTextCursor::KeepAnchor); QString sel = c.selectedText(); int startCharPos = sel.indexOf(QLatin1Char('"')); if (startCharPos == -1) { startCharPos = sel.indexOf(QLatin1Char('<')); m_model->m_completionOperator = T_ANGLE_STRING_LITERAL; } else { m_model->m_completionOperator = T_STRING_LITERAL; } if (startCharPos != -1) directoryPrefix = sel.mid(startCharPos + 1, sel.length() - 1); } // Make completion for all relevant includes QStringList includePaths = m_interface->includePaths(); const QString ¤tFilePath = QFileInfo(m_interface->fileName()).path(); if (!includePaths.contains(currentFilePath)) includePaths.append(currentFilePath); const Core::MimeType mimeType = Core::MimeDatabase::findByType(QLatin1String("text/x-c++hdr")); const QStringList suffixes = mimeType.suffixes(); foreach (const QString &includePath, includePaths) { QString realPath = includePath; if (!directoryPrefix.isEmpty()) { realPath += QLatin1Char('/'); realPath += directoryPrefix; } completeIncludePath(realPath, suffixes); } foreach (const QString &frameworkPath, m_interface->frameworkPaths()) { QString realPath = frameworkPath; if (!directoryPrefix.isEmpty()) { realPath += QLatin1Char('/'); realPath += directoryPrefix; realPath += QLatin1String(".framework/Headers"); } completeIncludePath(realPath, suffixes); } return !m_completions.isEmpty(); } /** * @brief Adds #include completion proposals using given include path * @param realPath - one of directories where compiler searches includes * @param suffixes - file suffixes for C/C++ header files */ void ClangCompletionAssistProcessor::completeIncludePath(const QString &realPath, const QStringList &suffixes) { QDirIterator i(realPath, QDir::Files | QDir::Dirs | QDir::NoDotAndDotDot); const QString hint = QObject::tr("Location: ", "Parent folder for proposed #include completion") + QDir::cleanPath(realPath); while (i.hasNext()) { const QString fileName = i.next(); const QFileInfo fileInfo = i.fileInfo(); const QString suffix = fileInfo.suffix(); if (suffix.isEmpty() || suffixes.contains(suffix)) { QString text = fileName.mid(realPath.length() + 1); if (fileInfo.isDir()) text += QLatin1Char('/'); ClangAssistProposalItem *item = new ClangAssistProposalItem; item->setText(text); item->setDetail(hint); item->setIcon(m_icons.keywordIcon()); item->keepCompletionOperator(m_model->m_completionOperator); m_completions.append(item); } } } void ClangCompletionAssistProcessor::completePreprocessor() { foreach (const QString &preprocessorCompletion, m_preprocessorCompletions) addCompletionItem(preprocessorCompletion, m_icons.iconForType(Icons::MacroIconType)); if (m_interface->objcEnabled()) addCompletionItem(QLatin1String("import"), m_icons.iconForType(Icons::MacroIconType)); } void ClangCompletionAssistProcessor::addCompletionItem(const QString &text, const QIcon &icon, int order, const QVariant &data) { ClangAssistProposalItem *item = new ClangAssistProposalItem; item->setText(text); item->setIcon(icon); item->setOrder(order); item->setData(data); item->keepCompletionOperator(m_model->m_completionOperator); m_completions.append(item); }