| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2015-01-14 18:07:15 +01:00
										 |  |  | ** Copyright (C) 2015 The Qt Company Ltd. | 
					
						
							|  |  |  | ** Contact: http://www.qt.io/licensing
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** 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 | 
					
						
							| 
									
										
										
										
											2015-01-14 18:07:15 +01:00
										 |  |  | ** a written agreement between you and The Qt Company.  For licensing terms and | 
					
						
							|  |  |  | ** conditions see http://www.qt.io/terms-conditions.  For further information
 | 
					
						
							| 
									
										
										
										
											2014-10-01 13:21:18 +02:00
										 |  |  | ** use the contact form at http://www.qt.io/contact-us.
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** Alternatively, this file may be used under the terms of the GNU Lesser | 
					
						
							| 
									
										
										
										
											2014-10-01 13:21:18 +02:00
										 |  |  | ** General Public License version 2.1 or version 3 as published by the Free | 
					
						
							|  |  |  | ** Software Foundation and appearing in the file LICENSE.LGPLv21 and | 
					
						
							|  |  |  | ** LICENSE.LGPLv3 included in the packaging of this file.  Please review the | 
					
						
							|  |  |  | ** following information to ensure the GNU Lesser General Public License | 
					
						
							|  |  |  | ** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
 | 
					
						
							|  |  |  | ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2015-01-14 18:07:15 +01:00
										 |  |  | ** In addition, as a special exception, The Qt Company gives you certain additional | 
					
						
							|  |  |  | ** rights.  These rights are described in The Qt Company LGPL Exception | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | #include "clangassistproposalitem.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 09:43:50 +01:00
										 |  |  | #include "clangactivationsequenceprocessor.h"
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | #include "clangassistproposal.h"
 | 
					
						
							|  |  |  | #include "clangassistproposalmodel.h"
 | 
					
						
							|  |  |  | #include "clangcompletionassistprocessor.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | #include "clangcompletioncontextanalyzer.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  | #include "clangeditordocumentprocessor.h"
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | #include "clangfunctionhintmodel.h"
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | #include "clangutils.h"
 | 
					
						
							| 
									
										
										
										
											2015-11-30 09:43:50 +01:00
										 |  |  | #include "clangcompletionchunkstotextconverter.h"
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | #include <utils/qtcassert.h>
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  | #include <cpptools/editordocumenthandle.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | #include <texteditor/codeassist/assistproposalitem.h>
 | 
					
						
							|  |  |  | #include <texteditor/codeassist/functionhintproposal.h>
 | 
					
						
							|  |  |  | #include <texteditor/codeassist/ifunctionhintproposalmodel.h>
 | 
					
						
							|  |  |  | #include <texteditor/convenience.h>
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <cpptools/cppdoxygen.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | #include <cplusplus/BackwardsScanner.h>
 | 
					
						
							|  |  |  | #include <cplusplus/ExpressionUnderCursor.h>
 | 
					
						
							|  |  |  | #include <cplusplus/SimpleLexer.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QDirIterator>
 | 
					
						
							|  |  |  | #include <QTextBlock>
 | 
					
						
							|  |  |  | #include <filecontainer.h>
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-04 09:32:46 +01:00
										 |  |  | #include <utils/mimetypes/mimedatabase.h>
 | 
					
						
							| 
									
										
										
										
											2014-06-20 15:44:56 +04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | namespace ClangCodeModel { | 
					
						
							|  |  |  | namespace Internal { | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-22 17:33:27 +02:00
										 |  |  | using ClangBackEnd::CodeCompletion; | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | using TextEditor::AssistProposalItem; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | const char SNIPPET_ICON_PATH[] = ":/texteditor/images/snippet.png"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | QList<AssistProposalItem *> toAssistProposalItems(const CodeCompletions &completions) | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     static CPlusPlus::Icons m_icons; // de-deduplicate
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool signalCompletion = false; // TODO
 | 
					
						
							|  |  |  |     bool slotCompletion = false; // TODO
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QIcon snippetIcon = QIcon(QLatin1String(SNIPPET_ICON_PATH)); | 
					
						
							|  |  |  |     QHash<QString, ClangAssistProposalItem *> items; | 
					
						
							|  |  |  |     foreach (const CodeCompletion &ccr, completions) { | 
					
						
							|  |  |  |         if (ccr.text().isEmpty()) // TODO: Make isValid()?
 | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         if (signalCompletion && ccr.completionKind() != CodeCompletion::SignalCompletionKind) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         if (slotCompletion && ccr.completionKind() != CodeCompletion::SlotCompletionKind) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-14 16:08:14 +02:00
										 |  |  |         QString name; | 
					
						
							|  |  |  |         if (ccr.completionKind() == CodeCompletion::KeywordCompletionKind) | 
					
						
							|  |  |  |             name = CompletionChunksToTextConverter::convertToName(ccr.chunks()); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             name = ccr.text().toString(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ClangAssistProposalItem *item = items.value(name, 0); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         if (item) { | 
					
						
							|  |  |  |             item->addOverload(ccr); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             item = new ClangAssistProposalItem; | 
					
						
							| 
									
										
										
										
											2015-07-14 16:08:14 +02:00
										 |  |  |             items.insert(name, item); | 
					
						
							|  |  |  |             item->setText(name); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |             item->setOrder(ccr.priority()); | 
					
						
							| 
									
										
										
										
											2015-08-25 16:19:19 +02:00
										 |  |  |             QString detail = CompletionChunksToTextConverter::convertToToolTip(ccr.chunks()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!ccr.briefComment().isEmpty()) | 
					
						
							|  |  |  |                 detail += QStringLiteral("\n\n") + ccr.briefComment().toString(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             item->setDetail(detail); | 
					
						
							| 
									
										
										
										
											2015-07-15 17:01:50 +02:00
										 |  |  |             item->setCodeCompletion(ccr); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         // FIXME: show the effective accessebility instead of availability
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  |         using CPlusPlus::Icons; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         switch (ccr.completionKind()) { | 
					
						
							|  |  |  |         case CodeCompletion::ClassCompletionKind: | 
					
						
							|  |  |  |         case CodeCompletion::TemplateClassCompletionKind: | 
					
						
							|  |  |  |             item->setIcon(m_icons.iconForType(Icons::ClassIconType)); break; | 
					
						
							|  |  |  |         case CodeCompletion::EnumerationCompletionKind: item->setIcon(m_icons.iconForType(Icons::EnumIconType)); break; | 
					
						
							|  |  |  |         case CodeCompletion::EnumeratorCompletionKind: item->setIcon(m_icons.iconForType(Icons::EnumeratorIconType)); break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case CodeCompletion::ConstructorCompletionKind: // fall through
 | 
					
						
							|  |  |  |         case CodeCompletion::DestructorCompletionKind: // fall through
 | 
					
						
							|  |  |  |         case CodeCompletion::FunctionCompletionKind: | 
					
						
							|  |  |  |         case CodeCompletion::TemplateFunctionCompletionKind: | 
					
						
							|  |  |  |         case CodeCompletion::ObjCMessageCompletionKind: | 
					
						
							|  |  |  |             switch (ccr.availability()) { | 
					
						
							|  |  |  |             case CodeCompletion::Available: | 
					
						
							|  |  |  |             case CodeCompletion::Deprecated: | 
					
						
							|  |  |  |                 item->setIcon(m_icons.iconForType(Icons::FuncPublicIconType)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 item->setIcon(m_icons.iconForType(Icons::FuncPrivateIconType)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         case CodeCompletion::SignalCompletionKind: | 
					
						
							|  |  |  |             item->setIcon(m_icons.iconForType(Icons::SignalIconType)); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         case CodeCompletion::SlotCompletionKind: | 
					
						
							|  |  |  |             switch (ccr.availability()) { | 
					
						
							|  |  |  |             case CodeCompletion::Available: | 
					
						
							|  |  |  |             case CodeCompletion::Deprecated: | 
					
						
							|  |  |  |                 item->setIcon(m_icons.iconForType(Icons::SlotPublicIconType)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case CodeCompletion::NotAccessible: | 
					
						
							|  |  |  |             case CodeCompletion::NotAvailable: | 
					
						
							|  |  |  |                 item->setIcon(m_icons.iconForType(Icons::SlotPrivateIconType)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case CodeCompletion::NamespaceCompletionKind: item->setIcon(m_icons.iconForType(Icons::NamespaceIconType)); break; | 
					
						
							|  |  |  |         case CodeCompletion::PreProcessorCompletionKind: item->setIcon(m_icons.iconForType(Icons::MacroIconType)); break; | 
					
						
							|  |  |  |         case CodeCompletion::VariableCompletionKind: | 
					
						
							|  |  |  |             switch (ccr.availability()) { | 
					
						
							|  |  |  |             case CodeCompletion::Available: | 
					
						
							|  |  |  |             case CodeCompletion::Deprecated: | 
					
						
							|  |  |  |                 item->setIcon(m_icons.iconForType(Icons::VarPublicIconType)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 item->setIcon(m_icons.iconForType(Icons::VarPrivateIconType)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case CodeCompletion::KeywordCompletionKind: | 
					
						
							|  |  |  |             item->setIcon(m_icons.iconForType(Icons::KeywordIconType)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case CodeCompletion::ClangSnippetKind: | 
					
						
							|  |  |  |             item->setIcon(snippetIcon); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case CodeCompletion::Other: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-15 16:25:04 +02:00
										 |  |  |     QList<AssistProposalItem *> results; | 
					
						
							|  |  |  |     results.reserve(items.size()); | 
					
						
							|  |  |  |     std::copy(items.cbegin(), items.cend(), std::back_inserter(results)); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-15 16:25:04 +02:00
										 |  |  |     return results; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | bool isFunctionHintLikeCompletion(CodeCompletion::Kind kind) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return kind == CodeCompletion::FunctionCompletionKind | 
					
						
							|  |  |  |         || kind == CodeCompletion::ConstructorCompletionKind | 
					
						
							|  |  |  |         || kind == CodeCompletion::DestructorCompletionKind | 
					
						
							|  |  |  |         || kind == CodeCompletion::SignalCompletionKind | 
					
						
							|  |  |  |         || kind == CodeCompletion::SlotCompletionKind; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-22 17:33:27 +02:00
										 |  |  | CodeCompletions matchingFunctionCompletions(const CodeCompletions completions, | 
					
						
							|  |  |  |                                             const QString &functionName) | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-22 17:33:27 +02:00
										 |  |  |     CodeCompletions matching; | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     foreach (const CodeCompletion &completion, completions) { | 
					
						
							|  |  |  |         if (isFunctionHintLikeCompletion(completion.completionKind()) | 
					
						
							|  |  |  |                 && completion.text().toString() == functionName) { | 
					
						
							|  |  |  |             matching.append(completion); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return matching; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | } // Anonymous
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | using namespace CPlusPlus; | 
					
						
							|  |  |  | using namespace TextEditor; | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | ClangCompletionAssistProcessor::ClangCompletionAssistProcessor() | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     : m_completionOperator(T_EOF_SYMBOL) | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ClangCompletionAssistProcessor::~ClangCompletionAssistProcessor() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-04 00:04:18 +02:00
										 |  |  | IAssistProposal *ClangCompletionAssistProcessor::perform(const AssistInterface *interface) | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     m_interface.reset(static_cast<const ClangCompletionAssistInterface *>(interface)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 12:15:15 +02:00
										 |  |  |     if (interface->reason() != ExplicitlyInvoked && !accepts()) { | 
					
						
							|  |  |  |         setPerformWasApplicable(false); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2015-07-29 12:15:15 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     return startCompletionHelper(); // == 0 if results are calculated asynchronously
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-22 13:09:44 +02:00
										 |  |  | bool ClangCompletionAssistProcessor::handleAvailableAsyncCompletions( | 
					
						
							| 
									
										
										
										
											2015-07-22 13:18:08 +02:00
										 |  |  |         const CodeCompletions &completions) | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-22 13:09:44 +02:00
										 |  |  |     bool handled = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     switch (m_sentRequestType) { | 
					
						
							|  |  |  |     case CompletionRequestType::NormalCompletion: | 
					
						
							| 
									
										
										
										
											2015-07-22 13:18:08 +02:00
										 |  |  |         handleAvailableCompletions(completions); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case CompletionRequestType::FunctionHintCompletion: | 
					
						
							| 
									
										
										
										
											2015-07-22 13:09:44 +02:00
										 |  |  |         handled = handleAvailableFunctionHintCompletions(completions); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         QTC_CHECK(!"Unhandled ClangCompletionAssistProcessor::CompletionRequestType"); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-22 13:09:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return handled; | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | const TextEditorWidget *ClangCompletionAssistProcessor::textEditorWidget() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_interface->textEditorWidget(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | /// Seach backwards in the document starting from pos to find the first opening
 | 
					
						
							|  |  |  | /// parenthesis. Nested parenthesis are skipped.
 | 
					
						
							|  |  |  | static int findOpenParen(QTextDocument *document, int start) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned parenCount = 1; | 
					
						
							|  |  |  |     for (int position = start; position >= 0; --position) { | 
					
						
							|  |  |  |         const QChar ch = document->characterAt(position); | 
					
						
							|  |  |  |         if (ch == QLatin1Char('(')) { | 
					
						
							|  |  |  |             --parenCount; | 
					
						
							|  |  |  |             if (parenCount == 0) | 
					
						
							|  |  |  |                 return position; | 
					
						
							|  |  |  |         } else if (ch == QLatin1Char(')')) { | 
					
						
							|  |  |  |             ++parenCount; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 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; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         --comma; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     if (comma < 0) | 
					
						
							|  |  |  |         return QByteArray(); | 
					
						
							|  |  |  |     const int openBrace = findOpenParen(doc, comma); | 
					
						
							|  |  |  |     if (openBrace < 0) | 
					
						
							|  |  |  |         return QByteArray(); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     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; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | IAssistProposal *ClangCompletionAssistProcessor::startCompletionHelper() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ClangCompletionContextAnalyzer analyzer(m_interface.data(), m_interface->languageFeatures()); | 
					
						
							|  |  |  |     analyzer.analyze(); | 
					
						
							|  |  |  |     m_completionOperator = analyzer.completionOperator(); | 
					
						
							|  |  |  |     m_positionForProposal = analyzer.positionForProposal(); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     QByteArray modifiedFileContent; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     const ClangCompletionContextAnalyzer::CompletionAction action = analyzer.completionAction(); | 
					
						
							|  |  |  |     switch (action) { | 
					
						
							|  |  |  |     case ClangCompletionContextAnalyzer::CompleteDoxygenKeyword: | 
					
						
							|  |  |  |         if (completeDoxygenKeywords()) | 
					
						
							|  |  |  |             return createProposal(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ClangCompletionContextAnalyzer::CompleteIncludePath: | 
					
						
							|  |  |  |         if (completeInclude(analyzer.positionEndOfExpression())) | 
					
						
							|  |  |  |             return createProposal(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ClangCompletionContextAnalyzer::CompletePreprocessorDirective: | 
					
						
							|  |  |  |         if (completePreprocessorDirectives()) | 
					
						
							|  |  |  |             return createProposal(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ClangCompletionContextAnalyzer::CompleteSignal: | 
					
						
							|  |  |  |     case ClangCompletionContextAnalyzer::CompleteSlot: | 
					
						
							|  |  |  |         modifiedFileContent = modifyInput(m_interface->textDocument(), | 
					
						
							|  |  |  |                                           analyzer.positionEndOfExpression()); | 
					
						
							|  |  |  |         // Fall through!
 | 
					
						
							|  |  |  |     case ClangCompletionContextAnalyzer::PassThroughToLibClang: { | 
					
						
							|  |  |  |         m_addSnippets = m_completionOperator == T_EOF_SYMBOL; | 
					
						
							|  |  |  |         m_sentRequestType = NormalCompletion; | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  |         const bool requestSent = sendCompletionRequest(analyzer.positionForClang(), | 
					
						
							|  |  |  |                                                        modifiedFileContent); | 
					
						
							|  |  |  |         setPerformWasApplicable(requestSent); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case ClangCompletionContextAnalyzer::PassThroughToLibClangAfterLeftParen: { | 
					
						
							|  |  |  |         m_sentRequestType = FunctionHintCompletion; | 
					
						
							|  |  |  |         m_functionName = analyzer.functionName(); | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  |         const bool requestSent = sendCompletionRequest(analyzer.positionForClang(), QByteArray()); | 
					
						
							|  |  |  |         setPerformWasApplicable(requestSent); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-23 12:01:54 +02:00
										 |  |  | // TODO: Extract duplicated logic from InternalCppCompletionAssistProcessor::startOfOperator
 | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  | int ClangCompletionAssistProcessor::startOfOperator(int positionInDocument, | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                                                     unsigned *kind, | 
					
						
							|  |  |  |                                                     bool wantFunctionCall) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |     auto activationSequence = m_interface->textAt(positionInDocument - 3, 3); | 
					
						
							|  |  |  |     ActivationSequenceProcessor activationSequenceProcessor(activationSequence, | 
					
						
							|  |  |  |                                                             positionInDocument, | 
					
						
							|  |  |  |                                                             wantFunctionCall); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |     *kind = activationSequenceProcessor.completionKind(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-23 16:06:48 +02:00
										 |  |  |     int start = activationSequenceProcessor.operatorStartPosition(); | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |     if (start != positionInDocument) { | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         QTextCursor tc(m_interface->textDocument()); | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |         tc.setPosition(positionInDocument); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // 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; | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |                 start = positionInDocument; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (*kind == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2015-02-26 09:14:38 +02:00
										 |  |  |             ExpressionUnderCursor expressionUnderCursor(m_interface->languageFeatures()); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |             if (expressionUnderCursor.startOfFunctionCall(tc) == -1) { | 
					
						
							|  |  |  |                 *kind = T_EOF_SYMBOL; | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |                 start = positionInDocument; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         SimpleLexer tokenize; | 
					
						
							| 
									
										
										
										
											2015-02-26 08:42:05 +02:00
										 |  |  |         tokenize.setLanguageFeatures(m_interface->languageFeatures()); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         tokenize.setSkipComments(false); | 
					
						
							| 
									
										
										
										
											2014-11-06 09:35:29 +01:00
										 |  |  |         const Tokens &tokens = tokenize(tc.block().text(), BackwardsScanner::previousBlockState(tc.block())); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         const int tokenIdx = SimpleLexer::tokenBefore(tokens, qMax(0, tc.positionInBlock() - 1)); // get the token at the left of the cursor
 | 
					
						
							| 
									
										
										
										
											2015-02-04 17:01:07 +02:00
										 |  |  |         const Token tk = (tokenIdx == -1) ? Token() : tokens.at(tokenIdx); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (*kind == T_DOXY_COMMENT && !(tk.is(T_DOXY_COMMENT) || tk.is(T_CPP_DOXY_COMMENT))) { | 
					
						
							|  |  |  |             *kind = T_EOF_SYMBOL; | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |             start = positionInDocument; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         // Don't complete in comments or strings, but still check for include completion
 | 
					
						
							| 
									
										
										
										
											2015-04-23 12:01:54 +02:00
										 |  |  |         else if (tk.is(T_COMMENT) || tk.is(T_CPP_COMMENT) | 
					
						
							|  |  |  |                  || tk.is(T_CPP_DOXY_COMMENT) || tk.is(T_DOXY_COMMENT) | 
					
						
							|  |  |  |                  || (tk.isLiteral() && (*kind != T_STRING_LITERAL | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                                      && *kind != T_ANGLE_STRING_LITERAL | 
					
						
							|  |  |  |                                      && *kind != T_SLASH))) { | 
					
						
							|  |  |  |             *kind = T_EOF_SYMBOL; | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |             start = positionInDocument; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         // 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; | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |             start = positionInDocument; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         else if (*kind == T_LPAREN) { | 
					
						
							|  |  |  |             if (tokenIdx > 0) { | 
					
						
							| 
									
										
										
										
											2015-02-04 17:01:07 +02:00
										 |  |  |                 const Token &previousToken = tokens.at(tokenIdx - 1); // look at the token at the left of T_LPAREN
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                 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; | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |                     start = positionInDocument; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // 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))) { | 
					
						
							| 
									
										
										
										
											2015-02-04 17:01:07 +02:00
										 |  |  |                     const Token &directiveToken = tokens.at(1); | 
					
						
							| 
									
										
										
										
											2013-12-13 18:41:15 +01:00
										 |  |  |                     QString directive = tc.block().text().mid(directiveToken.bytesBegin(), | 
					
						
							|  |  |  |                                                               directiveToken.bytes()); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                     if (directive == QLatin1String("include") || | 
					
						
							|  |  |  |                             directive == QLatin1String("include_next") || | 
					
						
							|  |  |  |                             directive == QLatin1String("import")) { | 
					
						
							|  |  |  |                         include = true; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!include) { | 
					
						
							|  |  |  |                 *kind = T_EOF_SYMBOL; | 
					
						
							| 
									
										
										
										
											2015-07-08 14:07:13 +02:00
										 |  |  |                 start = positionInDocument; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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); | 
					
						
							| 
									
										
										
										
											2014-11-06 09:35:29 +01:00
										 |  |  |                     const Tokens &tokens = tokenize(tc.block().text(), BackwardsScanner::previousBlockState(tc.block())); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                     const int tokenIdx = SimpleLexer::tokenBefore(tokens, qMax(0, tc.positionInBlock() - 1)); | 
					
						
							| 
									
										
										
										
											2015-02-04 17:01:07 +02:00
										 |  |  |                     const Token tk = (tokenIdx == -1) ? Token() : tokens.at(tokenIdx); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     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(); | 
					
						
							| 
									
										
										
										
											2015-02-04 17:01:07 +02:00
										 |  |  |                         const Token &idToken = tokens.at(1); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                         const QStringRef &identifier = | 
					
						
							| 
									
										
										
										
											2013-12-13 18:41:15 +01:00
										 |  |  |                                 line.midRef(idToken.bytesBegin(), | 
					
						
							|  |  |  |                                             idToken.bytesEnd() - idToken.bytesBegin()); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |                         if (identifier == QLatin1String("include") | 
					
						
							|  |  |  |                                 || identifier == QLatin1String("include_next") | 
					
						
							|  |  |  |                                 || (m_interface->objcEnabled() && identifier == QLatin1String("import"))) { | 
					
						
							|  |  |  |                             return true; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * @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; | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     if (m_completionOperator == T_SLASH) { | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         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('<')); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |             m_completionOperator = T_ANGLE_STRING_LITERAL; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |             m_completionOperator = T_STRING_LITERAL; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (startCharPos != -1) | 
					
						
							|  |  |  |             directoryPrefix = sel.mid(startCharPos + 1, sel.length() - 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Make completion for all relevant includes
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  |     CppTools::ProjectPart::HeaderPaths headerPaths = m_interface->headerPaths(); | 
					
						
							|  |  |  |     const CppTools::ProjectPart::HeaderPath currentFilePath(QFileInfo(m_interface->fileName()).path(), | 
					
						
							|  |  |  |                                                             CppTools::ProjectPart::HeaderPath::IncludePath); | 
					
						
							| 
									
										
										
										
											2014-06-25 17:23:19 +02:00
										 |  |  |     if (!headerPaths.contains(currentFilePath)) | 
					
						
							|  |  |  |         headerPaths.append(currentFilePath); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-17 14:34:28 +01:00
										 |  |  |     ::Utils::MimeDatabase mdb; | 
					
						
							|  |  |  |     const ::Utils::MimeType mimeType = mdb.mimeTypeForName(QLatin1String("text/x-c++hdr")); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |     const QStringList suffixes = mimeType.suffixes(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  |     foreach (const CppTools::ProjectPart::HeaderPath &headerPath, headerPaths) { | 
					
						
							| 
									
										
										
										
											2014-06-25 17:23:19 +02:00
										 |  |  |         QString realPath = headerPath.path; | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         if (!directoryPrefix.isEmpty()) { | 
					
						
							|  |  |  |             realPath += QLatin1Char('/'); | 
					
						
							|  |  |  |             realPath += directoryPrefix; | 
					
						
							| 
									
										
										
										
											2014-06-25 17:23:19 +02:00
										 |  |  |             if (headerPath.isFrameworkPath()) | 
					
						
							|  |  |  |                 realPath += QLatin1String(".framework/Headers"); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         completeIncludePath(realPath, suffixes); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return !m_completions.isEmpty(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | bool ClangCompletionAssistProcessor::completeInclude(int position) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QTextCursor textCursor(m_interface->textDocument()); // TODO: Simplify, move into function
 | 
					
						
							|  |  |  |     textCursor.setPosition(position); | 
					
						
							|  |  |  |     return completeInclude(textCursor); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * @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); | 
					
						
							| 
									
										
										
										
											2014-03-03 15:53:34 +01:00
										 |  |  |     //: Parent folder for proposed #include completion
 | 
					
						
							|  |  |  |     const QString hint = tr("Location: %1").arg(QDir::toNativeSeparators(QDir::cleanPath(realPath))); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |     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()); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |             item->keepCompletionOperator(m_completionOperator); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |             m_completions.append(item); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | bool ClangCompletionAssistProcessor::completePreprocessorDirectives() | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     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)); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return !m_completions.isEmpty(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ClangCompletionAssistProcessor::completeDoxygenKeywords() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  |     for (int i = 1; i < CppTools::T_DOXY_LAST_TAG; ++i) | 
					
						
							|  |  |  |         addCompletionItem(QString::fromLatin1(CppTools::doxygenTagSpell(i)), m_icons.keywordIcon()); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     return !m_completions.isEmpty(); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     item->keepCompletionOperator(m_completionOperator); | 
					
						
							| 
									
										
										
										
											2013-12-10 14:37:32 +01:00
										 |  |  |     m_completions.append(item); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-20 13:24:06 +02:00
										 |  |  | ClangCompletionAssistProcessor::UnsavedFileContentInfo | 
					
						
							|  |  |  | ClangCompletionAssistProcessor::unsavedFileContent(const QByteArray &customFileContent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const bool hasCustomModification = !customFileContent.isEmpty(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UnsavedFileContentInfo info; | 
					
						
							|  |  |  |     info.isDocumentModified = hasCustomModification || m_interface->textDocument()->isModified(); | 
					
						
							|  |  |  |     info.unsavedContent = hasCustomModification | 
					
						
							|  |  |  |                         ? customFileContent | 
					
						
							|  |  |  |                         : m_interface->textDocument()->toPlainText().toUtf8(); | 
					
						
							|  |  |  |     return info; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-13 15:56:41 +02:00
										 |  |  | void ClangCompletionAssistProcessor::sendFileContent(const QByteArray &customFileContent) | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-14 16:40:17 +02:00
										 |  |  |     // TODO: Revert custom modification after the completions
 | 
					
						
							| 
									
										
										
										
											2015-07-20 13:24:06 +02:00
										 |  |  |     const UnsavedFileContentInfo info = unsavedFileContent(customFileContent); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 13:59:08 +02:00
										 |  |  |     IpcCommunicator &ipcCommunicator = m_interface->ipcCommunicator(); | 
					
						
							| 
									
										
										
										
											2015-10-13 15:56:41 +02:00
										 |  |  |     ipcCommunicator.updateTranslationUnitsForEditor({{m_interface->fileName(), | 
					
						
							|  |  |  |                                                       Utf8String(), | 
					
						
							|  |  |  |                                                       Utf8String::fromByteArray(info.unsavedContent), | 
					
						
							|  |  |  |                                                       info.isDocumentModified, | 
					
						
							|  |  |  |                                                       uint(m_interface->textDocument()->revision())}}); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  | namespace { | 
					
						
							|  |  |  | CppTools::CppEditorDocumentHandle *cppDocument(const QString &filePath) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return CppTools::CppModelManager::instance()->cppEditorDocument(filePath); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool shouldSendDocumentForCompletion(const QString &filePath, | 
					
						
							|  |  |  |                                      int completionPosition) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto *document = cppDocument(filePath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (document) { | 
					
						
							| 
									
										
										
										
											2015-10-13 15:56:41 +02:00
										 |  |  |         auto &sendTracker = document->sendTracker(); | 
					
						
							|  |  |  |         return sendTracker.shouldSendRevisionWithCompletionPosition(int(document->revision()), | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  |                                                                     completionPosition); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  | bool shouldSendCodeCompletion(const QString &filePath, | 
					
						
							|  |  |  |                               int completionPosition) | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     auto *document = cppDocument(filePath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (document) { | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  |         auto &sendTracker = document->sendTracker(); | 
					
						
							|  |  |  |         return sendTracker.shouldSendCompletion(completionPosition); | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void setLastDocumentRevision(const QString &filePath) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto *document = cppDocument(filePath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (document) | 
					
						
							|  |  |  |         document->sendTracker().setLastSentRevision(int(document->revision())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void setLastCompletionPosition(const QString &filePath, | 
					
						
							|  |  |  |                                int completionPosition) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto *document = cppDocument(filePath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (document) | 
					
						
							|  |  |  |         document->sendTracker().setLastCompletionPosition(completionPosition); | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString projectPartIdForEditorDocument(const QString &filePath) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto projectPart = ClangEditorDocumentProcessor::get(filePath)->projectPart(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QString projectPartId; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (projectPart) | 
					
						
							|  |  |  |         projectPartId = projectPart->id(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return projectPartId; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  | bool ClangCompletionAssistProcessor::sendCompletionRequest(int position, | 
					
						
							| 
									
										
										
										
											2015-07-14 16:40:17 +02:00
										 |  |  |                                                            const QByteArray &customFileContent) | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     int line, column; | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  |     TextEditor::Convenience::convertPosition(m_interface->textDocument(), position, &line, &column); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     ++column; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QString filePath = m_interface->fileName(); | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  |     auto &ipcCommunicator = m_interface->ipcCommunicator(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (shouldSendCodeCompletion(filePath, position) | 
					
						
							|  |  |  |             || ipcCommunicator.isNotWaitingForCompletion()) { | 
					
						
							|  |  |  |         if (shouldSendDocumentForCompletion(filePath, position)) { | 
					
						
							|  |  |  |             sendFileContent(customFileContent); | 
					
						
							|  |  |  |             setLastDocumentRevision(filePath); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const QString projectPartId = projectPartIdForEditorDocument(filePath); | 
					
						
							|  |  |  |         ipcCommunicator.completeCode(this, filePath, uint(line), uint(column), projectPartId); | 
					
						
							|  |  |  |         setLastCompletionPosition(filePath, position); | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2015-09-28 17:42:43 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-01 13:21:02 +01:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TextEditor::IAssistProposal *ClangCompletionAssistProcessor::createProposal() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ClangAssistProposalModel *model = new ClangAssistProposalModel; | 
					
						
							|  |  |  |     model->loadContent(m_completions); | 
					
						
							|  |  |  |     return new ClangAssistProposal(m_positionForProposal, model); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-22 13:18:08 +02:00
										 |  |  | void ClangCompletionAssistProcessor::handleAvailableCompletions(const CodeCompletions &completions) | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     QTC_CHECK(m_completions.isEmpty()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_completions = toAssistProposalItems(completions); | 
					
						
							|  |  |  |     if (m_addSnippets) | 
					
						
							|  |  |  |         addSnippets(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setAsyncProposalAvailable(createProposal()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-22 13:09:44 +02:00
										 |  |  | bool ClangCompletionAssistProcessor::handleAvailableFunctionHintCompletions( | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |         const CodeCompletions &completions) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QTC_CHECK(!m_functionName.isEmpty()); | 
					
						
							|  |  |  |     const auto relevantCompletions = matchingFunctionCompletions(completions, m_functionName); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!relevantCompletions.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2015-07-22 13:18:08 +02:00
										 |  |  |         auto *model = new ClangFunctionHintModel(relevantCompletions); | 
					
						
							|  |  |  |         auto *proposal = new FunctionHintProposal(m_positionForProposal, model); | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         setAsyncProposalAvailable(proposal); | 
					
						
							| 
									
										
										
										
											2015-07-22 13:09:44 +02:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2015-07-22 13:09:44 +02:00
										 |  |  |         m_addSnippets = false; | 
					
						
							|  |  |  |         m_functionName.clear(); | 
					
						
							|  |  |  |         m_sentRequestType = NormalCompletion; | 
					
						
							|  |  |  |         sendCompletionRequest(m_interface->position(), QByteArray()); | 
					
						
							|  |  |  |         return false; // We are not yet finished.
 | 
					
						
							| 
									
										
										
										
											2015-05-08 15:48:17 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Internal
 | 
					
						
							|  |  |  | } // namespace ClangCodeModel
 | 
					
						
							| 
									
										
										
										
											2015-07-06 12:05:02 +02:00
										 |  |  | 
 |