2016-05-10 15:10:15 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2017 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
|
**
|
|
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "clangautomationutils.h"
|
|
|
|
|
|
2017-02-28 12:01:31 +01:00
|
|
|
#include "../clangcompletionassistinterface.h"
|
|
|
|
|
#include "../clangcompletionassistprovider.h"
|
2016-05-10 15:10:15 +02:00
|
|
|
|
|
|
|
|
#include <texteditor/codeassist/assistinterface.h>
|
|
|
|
|
#include <texteditor/codeassist/assistproposalitem.h>
|
|
|
|
|
#include <texteditor/codeassist/completionassistprovider.h>
|
|
|
|
|
#include <texteditor/codeassist/genericproposalmodel.h>
|
|
|
|
|
#include <texteditor/codeassist/iassistprocessor.h>
|
|
|
|
|
#include <texteditor/codeassist/iassistproposal.h>
|
|
|
|
|
#include <texteditor/textdocument.h>
|
|
|
|
|
#include <texteditor/texteditor.h>
|
|
|
|
|
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
|
|
|
|
#include <QCoreApplication>
|
|
|
|
|
#include <QElapsedTimer>
|
2020-06-19 10:17:21 +02:00
|
|
|
#include <QScopedPointer>
|
2016-05-10 15:10:15 +02:00
|
|
|
|
|
|
|
|
namespace ClangCodeModel {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
class WaitForAsyncCompletions
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
enum WaitResult { GotResults, GotInvalidResults, Timeout };
|
|
|
|
|
|
|
|
|
|
WaitResult wait(TextEditor::IAssistProcessor *processor,
|
|
|
|
|
TextEditor::AssistInterface *assistInterface,
|
|
|
|
|
int timeoutInMs)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(processor, return Timeout);
|
|
|
|
|
QTC_ASSERT(assistInterface, return Timeout);
|
|
|
|
|
|
|
|
|
|
bool gotResults = false;
|
|
|
|
|
|
|
|
|
|
processor->setAsyncCompletionAvailableHandler(
|
|
|
|
|
[this, &gotResults] (TextEditor::IAssistProposal *proposal) {
|
|
|
|
|
QTC_ASSERT(proposal, return);
|
|
|
|
|
proposalModel = proposal->model();
|
|
|
|
|
delete proposal;
|
|
|
|
|
gotResults = true;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Are there any immediate results?
|
|
|
|
|
if (TextEditor::IAssistProposal *proposal = processor->perform(assistInterface)) {
|
|
|
|
|
proposalModel = proposal->model();
|
|
|
|
|
delete proposal;
|
|
|
|
|
QTC_ASSERT(proposalModel, return GotInvalidResults);
|
|
|
|
|
return GotResults;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// There are not any, so wait for async results.
|
|
|
|
|
QElapsedTimer timer;
|
|
|
|
|
timer.start();
|
|
|
|
|
while (!gotResults) {
|
2020-06-19 10:17:21 +02:00
|
|
|
if (timer.elapsed() >= timeoutInMs) {
|
|
|
|
|
processor->cancel();
|
2016-05-10 15:10:15 +02:00
|
|
|
return Timeout;
|
2020-06-19 10:17:21 +02:00
|
|
|
}
|
2016-05-10 15:10:15 +02:00
|
|
|
QCoreApplication::processEvents();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return proposalModel ? GotResults : GotInvalidResults;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
2018-02-14 14:32:51 +01:00
|
|
|
TextEditor::ProposalModelPtr proposalModel;
|
2016-05-10 15:10:15 +02:00
|
|
|
};
|
|
|
|
|
|
2018-09-03 16:10:43 +02:00
|
|
|
static const ProjectExplorer::HeaderPaths toHeaderPaths(const QStringList &paths)
|
2016-05-10 15:10:15 +02:00
|
|
|
{
|
2018-09-03 16:10:43 +02:00
|
|
|
ProjectExplorer::HeaderPaths result;
|
2016-05-10 15:10:15 +02:00
|
|
|
foreach (const QString &path, paths)
|
2018-09-13 11:44:43 +02:00
|
|
|
result.push_back({path, ProjectExplorer::HeaderPathType::User});
|
2016-05-10 15:10:15 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-14 14:32:51 +01:00
|
|
|
TextEditor::ProposalModelPtr completionResults(TextEditor::BaseTextEditor *textEditor,
|
|
|
|
|
const QStringList &includePaths,
|
|
|
|
|
int timeOutInMs)
|
2016-05-10 15:10:15 +02:00
|
|
|
{
|
|
|
|
|
using namespace TextEditor;
|
|
|
|
|
|
2020-02-11 14:00:09 +01:00
|
|
|
auto textEditorWidget = TextEditorWidget::fromEditor(textEditor);
|
2018-02-14 14:32:51 +01:00
|
|
|
QTC_ASSERT(textEditorWidget, return TextEditor::ProposalModelPtr());
|
2016-05-10 15:10:15 +02:00
|
|
|
AssistInterface *assistInterface = textEditorWidget->createAssistInterface(
|
|
|
|
|
TextEditor::Completion, TextEditor::ExplicitlyInvoked);
|
2018-02-14 14:32:51 +01:00
|
|
|
QTC_ASSERT(assistInterface, return TextEditor::ProposalModelPtr());
|
2016-05-10 15:10:15 +02:00
|
|
|
if (!includePaths.isEmpty()) {
|
|
|
|
|
auto clangAssistInterface = static_cast<ClangCompletionAssistInterface *>(assistInterface);
|
|
|
|
|
clangAssistInterface->setHeaderPaths(toHeaderPaths(includePaths));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CompletionAssistProvider *assistProvider
|
|
|
|
|
= textEditor->textDocument()->completionAssistProvider();
|
|
|
|
|
QTC_ASSERT(qobject_cast<ClangCompletionAssistProvider *>(assistProvider),
|
2018-02-14 14:32:51 +01:00
|
|
|
return TextEditor::ProposalModelPtr());
|
|
|
|
|
QTC_ASSERT(assistProvider, return TextEditor::ProposalModelPtr());
|
|
|
|
|
QTC_ASSERT(assistProvider->runType() == IAssistProvider::Asynchronous,
|
|
|
|
|
return TextEditor::ProposalModelPtr());
|
2016-05-10 15:10:15 +02:00
|
|
|
|
2020-06-19 10:17:21 +02:00
|
|
|
QScopedPointer<IAssistProcessor> processor(assistProvider->createProcessor());
|
2018-02-14 14:32:51 +01:00
|
|
|
QTC_ASSERT(processor, return TextEditor::ProposalModelPtr());
|
2016-05-10 15:10:15 +02:00
|
|
|
|
|
|
|
|
WaitForAsyncCompletions waitForCompletions;
|
2020-06-19 10:17:21 +02:00
|
|
|
const WaitForAsyncCompletions::WaitResult result = waitForCompletions.wait(processor.data(),
|
2016-05-10 15:10:15 +02:00
|
|
|
assistInterface,
|
|
|
|
|
timeOutInMs);
|
2018-02-14 14:32:51 +01:00
|
|
|
QTC_ASSERT(result == WaitForAsyncCompletions::GotResults,
|
|
|
|
|
return TextEditor::ProposalModelPtr());
|
|
|
|
|
return waitForCompletions.proposalModel;
|
2016-05-10 15:10:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace ClangCodeModel
|