forked from qt-creator/qt-creator
The capabilities are not changeable for the user, but only there to check whether a server is capable of a specific task. This will also allow us to have more specialized settings widgets for specific servers like for the java language server without the need to add the capabilities to each of those special widgets. Also add the dynamic capabilities to the widget so users have a complete overview of the capabilities. Change-Id: I9f2ed6ed11b458f0d4c67be3df632fd810023286 Reviewed-by: Christian Stenger <christian.stenger@qt.io>
564 lines
22 KiB
C++
564 lines
22 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2018 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 "languageclientmanager.h"
|
|
|
|
#include "languageclientplugin.h"
|
|
#include "languageclientutils.h"
|
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
#include <coreplugin/find/searchresultwindow.h>
|
|
#include <coreplugin/icore.h>
|
|
#include <languageserverprotocol/messages.h>
|
|
#include <projectexplorer/project.h>
|
|
#include <projectexplorer/projectexplorer.h>
|
|
#include <projectexplorer/session.h>
|
|
#include <texteditor/textdocument.h>
|
|
#include <texteditor/texteditor.h>
|
|
#include <texteditor/textmark.h>
|
|
#include <utils/executeondestruction.h>
|
|
#include <utils/mimetypes/mimedatabase.h>
|
|
#include <utils/theme/theme.h>
|
|
#include <utils/utilsicons.h>
|
|
|
|
#include <QTextBlock>
|
|
#include <QTimer>
|
|
|
|
using namespace LanguageServerProtocol;
|
|
|
|
namespace LanguageClient {
|
|
|
|
static LanguageClientManager *managerInstance = nullptr;
|
|
|
|
LanguageClientManager::LanguageClientManager(QObject *parent)
|
|
: QObject (parent)
|
|
{
|
|
using namespace Core;
|
|
using namespace ProjectExplorer;
|
|
JsonRpcMessageHandler::registerMessageProvider<PublishDiagnosticsNotification>();
|
|
JsonRpcMessageHandler::registerMessageProvider<SemanticHighlightNotification>();
|
|
JsonRpcMessageHandler::registerMessageProvider<ApplyWorkspaceEditRequest>();
|
|
JsonRpcMessageHandler::registerMessageProvider<LogMessageNotification>();
|
|
JsonRpcMessageHandler::registerMessageProvider<ShowMessageRequest>();
|
|
JsonRpcMessageHandler::registerMessageProvider<ShowMessageNotification>();
|
|
JsonRpcMessageHandler::registerMessageProvider<WorkSpaceFolderRequest>();
|
|
JsonRpcMessageHandler::registerMessageProvider<RegisterCapabilityRequest>();
|
|
JsonRpcMessageHandler::registerMessageProvider<UnregisterCapabilityRequest>();
|
|
connect(EditorManager::instance(), &EditorManager::editorOpened,
|
|
this, &LanguageClientManager::editorOpened);
|
|
connect(EditorManager::instance(), &EditorManager::documentOpened,
|
|
this, &LanguageClientManager::documentOpened);
|
|
connect(EditorManager::instance(), &EditorManager::documentClosed,
|
|
this, &LanguageClientManager::documentClosed);
|
|
connect(EditorManager::instance(), &EditorManager::saved,
|
|
this, &LanguageClientManager::documentContentsSaved);
|
|
connect(EditorManager::instance(), &EditorManager::aboutToSave,
|
|
this, &LanguageClientManager::documentWillSave);
|
|
connect(SessionManager::instance(), &SessionManager::projectAdded,
|
|
this, &LanguageClientManager::projectAdded);
|
|
connect(SessionManager::instance(), &SessionManager::projectRemoved,
|
|
this, &LanguageClientManager::projectRemoved);
|
|
}
|
|
|
|
LanguageClientManager::~LanguageClientManager()
|
|
{
|
|
QTC_ASSERT(m_clients.isEmpty(), qDeleteAll(m_clients));
|
|
qDeleteAll(m_currentSettings);
|
|
managerInstance = nullptr;
|
|
}
|
|
|
|
void LanguageClientManager::init()
|
|
{
|
|
if (managerInstance)
|
|
return;
|
|
QTC_ASSERT(LanguageClientPlugin::instance(), return);
|
|
managerInstance = new LanguageClientManager(LanguageClientPlugin::instance());
|
|
}
|
|
|
|
void LanguageClientManager::clientStarted(Client *client)
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
QTC_ASSERT(client, return);
|
|
if (managerInstance->m_shuttingDown) {
|
|
clientFinished(client);
|
|
return;
|
|
}
|
|
if (!managerInstance->m_clients.contains(client)) {
|
|
managerInstance->m_clients << client;
|
|
connect(client, &Client::finished, managerInstance, [client]() { clientFinished(client); });
|
|
connect(client,
|
|
&Client::initialized,
|
|
managerInstance,
|
|
[client](const LanguageServerProtocol::ServerCapabilities &capabilities) {
|
|
managerInstance->m_currentDocumentLocatorFilter.updateCurrentClient();
|
|
managerInstance->m_inspector.clientInitialized(client->name(), capabilities);
|
|
});
|
|
connect(client,
|
|
&Client::capabilitiesChanged,
|
|
managerInstance,
|
|
[client](const DynamicCapabilities &capabilities) {
|
|
managerInstance->m_inspector.updateCapabilities(client->name(), capabilities);
|
|
});
|
|
}
|
|
|
|
client->initialize();
|
|
}
|
|
|
|
void LanguageClientManager::clientFinished(Client *client)
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
constexpr int restartTimeoutS = 5;
|
|
const bool unexpectedFinish = client->state() != Client::Shutdown
|
|
&& client->state() != Client::ShutdownRequested;
|
|
if (unexpectedFinish && !managerInstance->m_shuttingDown && client->reset()) {
|
|
client->disconnect(managerInstance);
|
|
client->log(tr("Unexpectedly finished. Restarting in %1 seconds.").arg(restartTimeoutS));
|
|
QTimer::singleShot(restartTimeoutS * 1000, client, [client]() { client->start(); });
|
|
for (TextEditor::TextDocument *document : managerInstance->m_clientForDocument.keys(client))
|
|
client->deactivateDocument(document);
|
|
} else {
|
|
if (unexpectedFinish && !managerInstance->m_shuttingDown)
|
|
client->log(tr("Unexpectedly finished."));
|
|
for (TextEditor::TextDocument *document : managerInstance->m_clientForDocument.keys(client))
|
|
managerInstance->m_clientForDocument.remove(document);
|
|
deleteClient(client);
|
|
if (managerInstance->m_shuttingDown && managerInstance->m_clients.isEmpty())
|
|
emit managerInstance->shutdownFinished();
|
|
}
|
|
}
|
|
|
|
Client *LanguageClientManager::startClient(BaseSettings *setting, ProjectExplorer::Project *project)
|
|
{
|
|
QTC_ASSERT(managerInstance, return nullptr);
|
|
QTC_ASSERT(setting, return nullptr);
|
|
QTC_ASSERT(setting->isValid(), return nullptr);
|
|
Client *client = setting->createClient();
|
|
QTC_ASSERT(client, return nullptr);
|
|
client->setCurrentProject(project);
|
|
client->start();
|
|
managerInstance->m_clientsForSetting[setting->m_id].append(client);
|
|
return client;
|
|
}
|
|
|
|
QVector<Client *> LanguageClientManager::clients()
|
|
{
|
|
QTC_ASSERT(managerInstance, return {});
|
|
return managerInstance->m_clients;
|
|
}
|
|
|
|
void LanguageClientManager::addExclusiveRequest(const MessageId &id, Client *client)
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
managerInstance->m_exclusiveRequests[id] << client;
|
|
}
|
|
|
|
void LanguageClientManager::reportFinished(const MessageId &id, Client *byClient)
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
for (Client *client : qAsConst(managerInstance->m_exclusiveRequests[id])) {
|
|
if (client != byClient)
|
|
client->cancelRequest(id);
|
|
}
|
|
managerInstance->m_exclusiveRequests.remove(id);
|
|
}
|
|
|
|
void LanguageClientManager::shutdownClient(Client *client)
|
|
{
|
|
if (!client)
|
|
return;
|
|
if (client->reachable())
|
|
client->shutdown();
|
|
else if (client->state() != Client::Shutdown && client->state() != Client::ShutdownRequested)
|
|
deleteClient(client);
|
|
}
|
|
|
|
void LanguageClientManager::deleteClient(Client *client)
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
QTC_ASSERT(client, return);
|
|
client->disconnect();
|
|
managerInstance->m_clients.removeAll(client);
|
|
for (QVector<Client *> &clients : managerInstance->m_clientsForSetting)
|
|
clients.removeAll(client);
|
|
if (managerInstance->m_shuttingDown)
|
|
delete client;
|
|
else
|
|
client->deleteLater();
|
|
}
|
|
|
|
void LanguageClientManager::shutdown()
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
if (managerInstance->m_shuttingDown)
|
|
return;
|
|
managerInstance->m_shuttingDown = true;
|
|
for (Client *client : qAsConst(managerInstance->m_clients))
|
|
shutdownClient(client);
|
|
QTimer::singleShot(3000, managerInstance, [](){
|
|
for (Client *client : qAsConst(managerInstance->m_clients))
|
|
deleteClient(client);
|
|
emit managerInstance->shutdownFinished();
|
|
});
|
|
}
|
|
|
|
LanguageClientManager *LanguageClientManager::instance()
|
|
{
|
|
return managerInstance;
|
|
}
|
|
|
|
QList<Client *> LanguageClientManager::clientsSupportingDocument(const TextEditor::TextDocument *doc)
|
|
{
|
|
QTC_ASSERT(managerInstance, return {});
|
|
QTC_ASSERT(doc, return {};);
|
|
return Utils::filtered(managerInstance->reachableClients(), [doc](Client *client) {
|
|
return client->isSupportedDocument(doc);
|
|
}).toList();
|
|
}
|
|
|
|
void LanguageClientManager::applySettings()
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
qDeleteAll(managerInstance->m_currentSettings);
|
|
managerInstance->m_currentSettings
|
|
= Utils::transform(LanguageClientSettings::pageSettings(), &BaseSettings::copy);
|
|
const QList<BaseSettings *> restarts = LanguageClientSettings::changedSettings();
|
|
LanguageClientSettings::toSettings(Core::ICore::settings(), managerInstance->m_currentSettings);
|
|
|
|
for (BaseSettings *setting : restarts) {
|
|
QList<TextEditor::TextDocument *> documents;
|
|
const QVector<Client *> currentClients = clientForSetting(setting);
|
|
for (Client *client : currentClients) {
|
|
documents << managerInstance->m_clientForDocument.keys(client);
|
|
shutdownClient(client);
|
|
}
|
|
for (auto document : qAsConst(documents))
|
|
managerInstance->m_clientForDocument.remove(document);
|
|
if (!setting->isValid() || !setting->m_enabled)
|
|
continue;
|
|
switch (setting->m_startBehavior) {
|
|
case BaseSettings::AlwaysOn: {
|
|
Client *client = startClient(setting);
|
|
for (TextEditor::TextDocument *document : qAsConst(documents))
|
|
managerInstance->m_clientForDocument[document] = client;
|
|
break;
|
|
}
|
|
case BaseSettings::RequiresFile: {
|
|
const QList<Core::IDocument *> &openedDocuments = Core::DocumentModel::openedDocuments();
|
|
for (Core::IDocument *document : openedDocuments) {
|
|
if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) {
|
|
if (setting->m_languageFilter.isSupported(document))
|
|
documents << textDocument;
|
|
}
|
|
}
|
|
if (!documents.isEmpty()) {
|
|
Client *client = startClient(setting);
|
|
for (TextEditor::TextDocument *document : qAsConst(documents))
|
|
client->openDocument(document);
|
|
}
|
|
break;
|
|
}
|
|
case BaseSettings::RequiresProject: {
|
|
const QList<Core::IDocument *> &openedDocuments = Core::DocumentModel::openedDocuments();
|
|
QHash<ProjectExplorer::Project *, Client *> clientForProject;
|
|
for (Core::IDocument *document : openedDocuments) {
|
|
auto textDocument = qobject_cast<TextEditor::TextDocument *>(document);
|
|
if (!textDocument || !setting->m_languageFilter.isSupported(textDocument))
|
|
continue;
|
|
const Utils::FilePath filePath = textDocument->filePath();
|
|
for (ProjectExplorer::Project *project :
|
|
ProjectExplorer::SessionManager::projects()) {
|
|
if (project->isKnownFile(filePath)) {
|
|
Client *client = clientForProject[project];
|
|
if (!client) {
|
|
client = startClient(setting, project);
|
|
if (!client)
|
|
continue;
|
|
clientForProject[project] = client;
|
|
}
|
|
client->openDocument(textDocument);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
QList<BaseSettings *> LanguageClientManager::currentSettings()
|
|
{
|
|
QTC_ASSERT(managerInstance, return {});
|
|
return managerInstance->m_currentSettings;
|
|
}
|
|
|
|
void LanguageClientManager::registerClientSettings(BaseSettings *settings)
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
LanguageClientSettings::addSettings(settings);
|
|
managerInstance->applySettings();
|
|
}
|
|
|
|
void LanguageClientManager::enableClientSettings(const QString &settingsId)
|
|
{
|
|
QTC_ASSERT(managerInstance, return);
|
|
LanguageClientSettings::enableSettings(settingsId);
|
|
managerInstance->applySettings();
|
|
}
|
|
|
|
QVector<Client *> LanguageClientManager::clientForSetting(const BaseSettings *setting)
|
|
{
|
|
QTC_ASSERT(managerInstance, return {});
|
|
auto instance = managerInstance;
|
|
return instance->m_clientsForSetting.value(setting->m_id);
|
|
}
|
|
|
|
const BaseSettings *LanguageClientManager::settingForClient(Client *client)
|
|
{
|
|
QTC_ASSERT(managerInstance, return nullptr);
|
|
for (auto it = managerInstance->m_clientsForSetting.cbegin();
|
|
it != managerInstance->m_clientsForSetting.cend(); ++it) {
|
|
const QString &id = it.key();
|
|
for (const Client *settingClient : it.value()) {
|
|
if (settingClient == client) {
|
|
return Utils::findOrDefault(managerInstance->m_currentSettings,
|
|
[id](BaseSettings *setting) {
|
|
return setting->m_id == id;
|
|
});
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Client *LanguageClientManager::clientForDocument(TextEditor::TextDocument *document)
|
|
{
|
|
QTC_ASSERT(managerInstance, return nullptr);
|
|
return document == nullptr ? nullptr
|
|
: managerInstance->m_clientForDocument.value(document).data();
|
|
}
|
|
|
|
Client *LanguageClientManager::clientForFilePath(const Utils::FilePath &filePath)
|
|
{
|
|
return clientForDocument(TextEditor::TextDocument::textDocumentForFilePath(filePath));
|
|
}
|
|
|
|
Client *LanguageClientManager::clientForUri(const DocumentUri &uri)
|
|
{
|
|
return clientForFilePath(uri.toFilePath());
|
|
}
|
|
|
|
void LanguageClientManager::openDocumentWithClient(TextEditor::TextDocument *document, Client *client)
|
|
{
|
|
Client *currentClient = clientForDocument(document);
|
|
if (client == currentClient)
|
|
return;
|
|
if (currentClient)
|
|
currentClient->deactivateDocument(document);
|
|
managerInstance->m_clientForDocument[document] = client;
|
|
if (client) {
|
|
if (!client->documentOpen(document))
|
|
client->openDocument(document);
|
|
else
|
|
client->activateDocument(document);
|
|
}
|
|
TextEditor::IOutlineWidgetFactory::updateOutline();
|
|
}
|
|
|
|
void LanguageClientManager::logBaseMessage(const LspLogMessage::MessageSender sender,
|
|
const QString &clientName,
|
|
const BaseMessage &message)
|
|
{
|
|
instance()->m_inspector.log(sender, clientName, message);
|
|
}
|
|
|
|
void LanguageClientManager::showInspector()
|
|
{
|
|
QWidget *inspectorWidget = instance()->m_inspector.createWidget();
|
|
inspectorWidget->setAttribute(Qt::WA_DeleteOnClose);
|
|
inspectorWidget->show();
|
|
}
|
|
|
|
QVector<Client *> LanguageClientManager::reachableClients()
|
|
{
|
|
return Utils::filtered(m_clients, &Client::reachable);
|
|
}
|
|
|
|
void LanguageClientManager::editorOpened(Core::IEditor *editor)
|
|
{
|
|
using namespace TextEditor;
|
|
if (auto *textEditor = qobject_cast<BaseTextEditor *>(editor)) {
|
|
if (TextEditorWidget *widget = textEditor->editorWidget()) {
|
|
connect(widget, &TextEditorWidget::requestLinkAt, this,
|
|
[document = textEditor->textDocument()]
|
|
(const QTextCursor &cursor, Utils::ProcessLinkCallback &callback, bool resolveTarget) {
|
|
if (auto client = clientForDocument(document))
|
|
client->symbolSupport().findLinkAt(document, cursor, callback, resolveTarget);
|
|
});
|
|
connect(widget, &TextEditorWidget::requestUsages, this,
|
|
[document = textEditor->textDocument()](const QTextCursor &cursor) {
|
|
if (auto client = clientForDocument(document))
|
|
client->symbolSupport().findUsages(document, cursor);
|
|
});
|
|
connect(widget, &TextEditorWidget::requestRename, this,
|
|
[document = textEditor->textDocument()](const QTextCursor &cursor) {
|
|
if (auto client = clientForDocument(document))
|
|
client->symbolSupport().renameSymbol(document, cursor);
|
|
});
|
|
connect(widget, &TextEditorWidget::cursorPositionChanged, this, [this, widget]() {
|
|
// TODO This would better be a compressing timer
|
|
QTimer::singleShot(50, this, [widget = QPointer<TextEditorWidget>(widget)]() {
|
|
if (!widget)
|
|
return;
|
|
if (Client *client = clientForDocument(widget->textDocument()))
|
|
if (client->reachable())
|
|
client->cursorPositionChanged(widget);
|
|
});
|
|
});
|
|
updateEditorToolBar(editor);
|
|
if (TextEditor::TextDocument *document = textEditor->textDocument()) {
|
|
if (Client *client = m_clientForDocument[document])
|
|
widget->addHoverHandler(client->hoverHandler());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void LanguageClientManager::documentOpened(Core::IDocument *document)
|
|
{
|
|
auto textDocument = qobject_cast<TextEditor::TextDocument *>(document);
|
|
if (!textDocument)
|
|
return;
|
|
|
|
// check whether we have to start servers for this document
|
|
const QList<BaseSettings *> settings = currentSettings();
|
|
for (BaseSettings *setting : settings) {
|
|
QVector<Client *> clients = clientForSetting(setting);
|
|
if (setting->isValid() && setting->m_enabled
|
|
&& setting->m_languageFilter.isSupported(document)) {
|
|
if (setting->m_startBehavior == BaseSettings::RequiresProject) {
|
|
const Utils::FilePath &filePath = document->filePath();
|
|
for (ProjectExplorer::Project *project :
|
|
ProjectExplorer::SessionManager::projects()) {
|
|
// check whether file is part of this project
|
|
if (!project->isKnownFile(filePath))
|
|
continue;
|
|
|
|
// check whether we already have a client running for this project
|
|
if (Utils::findOrDefault(clients,
|
|
[project](const QPointer<Client> &client) {
|
|
return client->project() == project;
|
|
})) {
|
|
continue;
|
|
}
|
|
clients << startClient(setting, project);
|
|
}
|
|
} else if (setting->m_startBehavior == BaseSettings::RequiresFile && clients.isEmpty()) {
|
|
clients << startClient(setting);
|
|
}
|
|
for (auto client : qAsConst(clients))
|
|
client->openDocument(textDocument);
|
|
}
|
|
}
|
|
}
|
|
|
|
void LanguageClientManager::documentClosed(Core::IDocument *document)
|
|
{
|
|
if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) {
|
|
for (Client *client : qAsConst(m_clients))
|
|
client->closeDocument(textDocument);
|
|
m_clientForDocument.remove(textDocument);
|
|
}
|
|
}
|
|
|
|
void LanguageClientManager::documentContentsSaved(Core::IDocument *document)
|
|
{
|
|
if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) {
|
|
const QVector<Client *> &clients = reachableClients();
|
|
for (Client *client : clients)
|
|
client->documentContentsSaved(textDocument);
|
|
}
|
|
}
|
|
|
|
void LanguageClientManager::documentWillSave(Core::IDocument *document)
|
|
{
|
|
if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) {
|
|
const QVector<Client *> &clients = reachableClients();
|
|
for (Client *client : clients)
|
|
client->documentWillSave(textDocument);
|
|
}
|
|
}
|
|
|
|
void LanguageClientManager::updateProject(ProjectExplorer::Project *project)
|
|
{
|
|
for (BaseSettings *setting : qAsConst(m_currentSettings)) {
|
|
if (setting->isValid()
|
|
&& setting->m_enabled
|
|
&& setting->m_startBehavior == BaseSettings::RequiresProject) {
|
|
if (Utils::findOrDefault(clientForSetting(setting),
|
|
[project](const QPointer<Client> &client) {
|
|
return client->project() == project;
|
|
})
|
|
== nullptr) {
|
|
Client *newClient = nullptr;
|
|
const QList<Core::IDocument *> &openedDocuments = Core::DocumentModel::openedDocuments();
|
|
for (Core::IDocument *doc : openedDocuments) {
|
|
if (setting->m_languageFilter.isSupported(doc)
|
|
&& project->isKnownFile(doc->filePath())) {
|
|
if (auto textDoc = qobject_cast<TextEditor::TextDocument *>(doc)) {
|
|
if (!newClient)
|
|
newClient = startClient(setting, project);
|
|
if (!newClient)
|
|
break;
|
|
newClient->openDocument(textDoc);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
const QVector<Client *> &clients = reachableClients();
|
|
for (Client *client : clients)
|
|
client->projectOpened(project);
|
|
}
|
|
|
|
void LanguageClientManager::projectAdded(ProjectExplorer::Project *project)
|
|
{
|
|
connect(project, &ProjectExplorer::Project::fileListChanged, this, [this, project]() {
|
|
updateProject(project);
|
|
});
|
|
}
|
|
|
|
void LanguageClientManager::projectRemoved(ProjectExplorer::Project *project)
|
|
{
|
|
project->disconnect(this);
|
|
for (Client *client : qAsConst(m_clients))
|
|
client->projectClosed(project);
|
|
}
|
|
|
|
} // namespace LanguageClient
|