LanguageClient: only send contents of TextDocuments to the server

Change-Id: Ifdcd6ba161336814744c51f15e35a756960cb199
Reviewed-by: Christian Stenger <christian.stenger@qt.io>
This commit is contained in:
David Schulz
2019-09-11 11:15:39 +02:00
parent 74688679b2
commit d3c94edd93
6 changed files with 79 additions and 75 deletions

View File

@@ -280,7 +280,7 @@ Client::State Client::state() const
return m_state; return m_state;
} }
bool Client::openDocument(Core::IDocument *document) bool Client::openDocument(TextEditor::TextDocument *document)
{ {
using namespace TextEditor; using namespace TextEditor;
if (!isSupportedDocument(document)) if (!isSupportedDocument(document))
@@ -291,9 +291,9 @@ bool Client::openDocument(Core::IDocument *document)
if (!registered.value()) if (!registered.value())
return false; return false;
const TextDocumentRegistrationOptions option( const TextDocumentRegistrationOptions option(
m_dynamicCapabilities.option(method).toObject()); m_dynamicCapabilities.option(method).toObject());
if (option.isValid(nullptr) if (option.isValid(nullptr)
&& !option.filterApplies(filePath, Utils::mimeTypeForName(document->mimeType()))) { && !option.filterApplies(filePath, Utils::mimeTypeForName(document->mimeType()))) {
return false; return false;
} }
} else if (Utils::optional<ServerCapabilities::TextDocumentSync> _sync } else if (Utils::optional<ServerCapabilities::TextDocumentSync> _sync
@@ -305,39 +305,35 @@ bool Client::openDocument(Core::IDocument *document)
} }
auto uri = DocumentUri::fromFileName(filePath); auto uri = DocumentUri::fromFileName(filePath);
showDiagnostics(uri); showDiagnostics(uri);
auto textDocument = qobject_cast<TextDocument *>(document);
connect(document,
&TextDocument::contentsChangedWithPosition,
this,
[this, document](int position, int charsRemoved, int charsAdded) {
documentContentsChanged(document, position, charsRemoved, charsAdded);
});
auto *oldCompletionProvider = qobject_cast<DocumentContentCompletionProvider *>(
document->completionAssistProvider());
// only replace the completion assist provider if it is the default one or null
if (oldCompletionProvider || !document->completionAssistProvider())
document->setCompletionAssistProvider(m_clientProviders.completionAssistProvider);
m_resetAssistProvider[document] = {oldCompletionProvider,
document->functionHintAssistProvider(),
document->quickFixAssistProvider()};
document->setFunctionHintAssistProvider(m_clientProviders.functionHintProvider);
document->setQuickFixAssistProvider(m_clientProviders.quickFixAssistProvider);
connect(document, &QObject::destroyed, this, [this, document] {
m_resetAssistProvider.remove(document);
});
const QString &text = document->plainText();
m_openedDocument.insert(document->filePath(), text);
TextDocumentItem item; TextDocumentItem item;
item.setLanguageId(TextDocumentItem::mimeTypeToLanguageId(document->mimeType())); item.setLanguageId(TextDocumentItem::mimeTypeToLanguageId(document->mimeType()));
item.setUri(uri); item.setUri(uri);
item.setText(QString::fromUtf8(document->contents())); item.setText(text);
item.setVersion(textDocument ? textDocument->document()->revision() : 0); item.setVersion(document->document()->revision());
if (textDocument) {
connect(textDocument,
&TextDocument::contentsChangedWithPosition,
this,
[this, textDocument](int position, int charsRemoved, int charsAdded) {
documentContentsChanged(textDocument, position, charsRemoved, charsAdded);
});
auto *oldCompletionProvider = qobject_cast<DocumentContentCompletionProvider *>(
textDocument->completionAssistProvider());
// only replace the completion assist provider if it is the default one or null
if (oldCompletionProvider || !textDocument->completionAssistProvider())
textDocument->setCompletionAssistProvider(m_clientProviders.completionAssistProvider);
m_resetAssistProvider[textDocument] = {oldCompletionProvider,
textDocument->functionHintAssistProvider(),
textDocument->quickFixAssistProvider()};
textDocument->setFunctionHintAssistProvider(m_clientProviders.functionHintProvider);
textDocument->setQuickFixAssistProvider(m_clientProviders.quickFixAssistProvider);
connect(textDocument, &QObject::destroyed, this, [this, textDocument]{
m_resetAssistProvider.remove(textDocument);
});
m_openedDocument.insert(document->filePath(), textDocument->plainText());
} else {
m_openedDocument.insert(document->filePath(), QString());
}
sendContent(DidOpenTextDocumentNotification(DidOpenTextDocumentParams(item))); sendContent(DidOpenTextDocumentNotification(DidOpenTextDocumentParams(item)));
return true; return true;
@@ -367,15 +363,14 @@ void Client::cancelRequest(const MessageId &id)
sendContent(CancelRequest(CancelParameter(id))); sendContent(CancelRequest(CancelParameter(id)));
} }
void Client::closeDocument(Core::IDocument *document) void Client::closeDocument(TextEditor::TextDocument *document)
{ {
m_openedDocument.remove(document->filePath()); m_openedDocument.remove(document->filePath());
const DocumentUri &uri = DocumentUri::fromFileName(document->filePath()); const DocumentUri &uri = DocumentUri::fromFileName(document->filePath());
const DidCloseTextDocumentParams params(TextDocumentIdentifier{uri}); const DidCloseTextDocumentParams params(TextDocumentIdentifier{uri});
m_highlights[uri].clear(); m_highlights[uri].clear();
sendContent(uri, DidCloseTextDocumentNotification(params)); sendContent(uri, DidCloseTextDocumentNotification(params));
if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) resetAssistProviders(document);
resetAssistProviders(textDocument);
} }
bool Client::documentOpen(const Core::IDocument *document) const bool Client::documentOpen(const Core::IDocument *document) const
@@ -383,7 +378,7 @@ bool Client::documentOpen(const Core::IDocument *document) const
return m_openedDocument.contains(document->filePath()); return m_openedDocument.contains(document->filePath());
} }
void Client::documentContentsSaved(Core::IDocument *document) void Client::documentContentsSaved(TextEditor::TextDocument *document)
{ {
if (!m_openedDocument.contains(document->filePath())) if (!m_openedDocument.contains(document->filePath()))
return; return;
@@ -413,7 +408,7 @@ void Client::documentContentsSaved(Core::IDocument *document)
DidSaveTextDocumentParams params( DidSaveTextDocumentParams params(
TextDocumentIdentifier(DocumentUri::fromFileName(document->filePath()))); TextDocumentIdentifier(DocumentUri::fromFileName(document->filePath())));
if (includeText) if (includeText)
params.setText(QString::fromUtf8(document->contents())); params.setText(document->plainText());
sendContent(DidSaveTextDocumentNotification(params)); sendContent(DidSaveTextDocumentNotification(params));
} }
@@ -744,7 +739,7 @@ void Client::setSupportedLanguage(const LanguageFilter &filter)
m_languagFilter = filter; m_languagFilter = filter;
} }
bool Client::isSupportedDocument(const Core::IDocument *document) const bool Client::isSupportedDocument(const TextEditor::TextDocument *document) const
{ {
QTC_ASSERT(document, return false); QTC_ASSERT(document, return false);
return m_languagFilter.isSupported(document); return m_languagFilter.isSupported(document);
@@ -836,13 +831,11 @@ void Client::showDiagnostics(Core::IDocument *doc)
showDiagnostics(DocumentUri::fromFileName(doc->filePath())); showDiagnostics(DocumentUri::fromFileName(doc->filePath()));
} }
void Client::hideDiagnostics(Core::IDocument *doc) void Client::hideDiagnostics(TextEditor::TextDocument *doc)
{ {
if (auto *textDocument = qobject_cast<TextEditor::TextDocument *>(doc)) { DocumentUri uri = DocumentUri::fromFileName(doc->filePath());
DocumentUri uri = DocumentUri::fromFileName(doc->filePath()); for (TextMark *mark : m_diagnostics.value(uri))
for (TextMark *mark : m_diagnostics.value(uri)) doc->removeMark(mark);
textDocument->removeMark(mark);
}
} }
const ServerCapabilities &Client::capabilities() const const ServerCapabilities &Client::capabilities() const

View File

@@ -95,10 +95,10 @@ public:
bool reachable() const { return m_state == Initialized; } bool reachable() const { return m_state == Initialized; }
// document synchronization // document synchronization
bool openDocument(Core::IDocument *document); bool openDocument(TextEditor::TextDocument *document);
void closeDocument(Core::IDocument *document); void closeDocument(TextEditor::TextDocument *document);
bool documentOpen(const Core::IDocument *document) const; bool documentOpen(const Core::IDocument *document) const;
void documentContentsSaved(Core::IDocument *document); void documentContentsSaved(TextEditor::TextDocument *document);
void documentWillSave(Core::IDocument *document); void documentWillSave(Core::IDocument *document);
void documentContentsChanged(TextEditor::TextDocument *document, void documentContentsChanged(TextEditor::TextDocument *document,
int position, int position,
@@ -129,7 +129,7 @@ public:
void cancelRequest(const LanguageServerProtocol::MessageId &id); void cancelRequest(const LanguageServerProtocol::MessageId &id);
void setSupportedLanguage(const LanguageFilter &filter); void setSupportedLanguage(const LanguageFilter &filter);
bool isSupportedDocument(const Core::IDocument *document) const; bool isSupportedDocument(const TextEditor::TextDocument *document) const;
bool isSupportedFile(const Utils::FilePath &filePath, const QString &mimeType) const; bool isSupportedFile(const Utils::FilePath &filePath, const QString &mimeType) const;
bool isSupportedUri(const LanguageServerProtocol::DocumentUri &uri) const; bool isSupportedUri(const LanguageServerProtocol::DocumentUri &uri) const;
@@ -155,7 +155,7 @@ public:
{ log(responseError.toString(), flag); } { log(responseError.toString(), flag); }
void showDiagnostics(Core::IDocument *doc); void showDiagnostics(Core::IDocument *doc);
void hideDiagnostics(Core::IDocument *doc); void hideDiagnostics(TextEditor::TextDocument *doc);
const LanguageServerProtocol::ServerCapabilities &capabilities() const; const LanguageServerProtocol::ServerCapabilities &capabilities() const;
const DynamicCapabilities &dynamicCapabilities() const; const DynamicCapabilities &dynamicCapabilities() const;

View File

@@ -201,7 +201,7 @@ LanguageClientManager *LanguageClientManager::instance()
return managerInstance; return managerInstance;
} }
QList<Client *> LanguageClientManager::clientsSupportingDocument(const Core::IDocument *doc) QList<Client *> LanguageClientManager::clientsSupportingDocument(const TextEditor::TextDocument *doc)
{ {
QTC_ASSERT(managerInstance, return {}); QTC_ASSERT(managerInstance, return {});
QTC_ASSERT(doc, return {};); QTC_ASSERT(doc, return {};);
@@ -222,7 +222,7 @@ void LanguageClientManager::applySettings()
&BaseSettings::needsRestart); &BaseSettings::needsRestart);
for (BaseSettings *setting : restarts) { for (BaseSettings *setting : restarts) {
QList<Core::IDocument *> documents; QList<TextEditor::TextDocument *> documents;
for (Client *client : clientForSetting(setting)) { for (Client *client : clientForSetting(setting)) {
documents << managerInstance->m_clientForDocument.keys(client); documents << managerInstance->m_clientForDocument.keys(client);
shutdownClient(client); shutdownClient(client);
@@ -232,7 +232,7 @@ void LanguageClientManager::applySettings()
switch (setting->m_startBehavior) { switch (setting->m_startBehavior) {
case BaseSettings::AlwaysOn: { case BaseSettings::AlwaysOn: {
Client *client = startClient(setting); Client *client = startClient(setting);
for (Core::IDocument *document : documents) for (TextEditor::TextDocument *document : documents)
managerInstance->m_clientForDocument[document] = client; managerInstance->m_clientForDocument[document] = client;
break; break;
} }
@@ -242,7 +242,7 @@ void LanguageClientManager::applySettings()
return filter.isSupported(doc); return filter.isSupported(doc);
})) { })) {
Client *client = startClient(setting); Client *client = startClient(setting);
for (Core::IDocument *document : documents) for (TextEditor::TextDocument *document : documents)
managerInstance->m_clientForDocument[document] = client; managerInstance->m_clientForDocument[document] = client;
} }
break; break;
@@ -295,13 +295,13 @@ const BaseSettings *LanguageClientManager::settingForClient(Client *client)
return nullptr; return nullptr;
} }
Client *LanguageClientManager::clientForDocument(Core::IDocument *document) Client *LanguageClientManager::clientForDocument(TextEditor::TextDocument *document)
{ {
QTC_ASSERT(managerInstance, return nullptr); QTC_ASSERT(managerInstance, return nullptr);
return document == nullptr ? nullptr : managerInstance->m_clientForDocument[document].data(); return document == nullptr ? nullptr : managerInstance->m_clientForDocument[document].data();
} }
bool LanguageClientManager::reOpenDocumentWithClient(Core::IDocument *document, Client *client) bool LanguageClientManager::reOpenDocumentWithClient(TextEditor::TextDocument *document, Client *client)
{ {
Utils::ExecuteOnDestruction outlineUpdater(&TextEditor::IOutlineWidgetFactory::updateOutline); Utils::ExecuteOnDestruction outlineUpdater(&TextEditor::IOutlineWidgetFactory::updateOutline);
if (Client *currentClient = clientForDocument(document)) { if (Client *currentClient = clientForDocument(document)) {
@@ -358,7 +358,7 @@ void LanguageClientManager::clientFinished(Client *client)
} else { } else {
if (unexpectedFinish && !m_shuttingDown) if (unexpectedFinish && !m_shuttingDown)
client->log(tr("Unexpectedly finished."), Core::MessageManager::Flash); client->log(tr("Unexpectedly finished."), Core::MessageManager::Flash);
for (Core::IDocument *document : m_clientForDocument.keys(client)) for (TextEditor::TextDocument *document : m_clientForDocument.keys(client))
m_clientForDocument.remove(document); m_clientForDocument.remove(document);
deleteClient(client); deleteClient(client);
if (m_shuttingDown && m_clients.isEmpty()) if (m_shuttingDown && m_clients.isEmpty())
@@ -394,7 +394,7 @@ void LanguageClientManager::editorOpened(Core::IEditor *editor)
}); });
}); });
updateEditorToolBar(editor); updateEditorToolBar(editor);
if (Core::IDocument *document = editor->document()) { if (TextEditor::TextDocument *document = textEditor->textDocument()) {
if (Client *client = m_clientForDocument[document]) if (Client *client = m_clientForDocument[document])
widget->addHoverHandler(client->hoverHandler()); widget->addHoverHandler(client->hoverHandler());
} }
@@ -404,6 +404,10 @@ void LanguageClientManager::editorOpened(Core::IEditor *editor)
void LanguageClientManager::documentOpened(Core::IDocument *document) 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 // check whether we have to start servers for this document
bool opened = false; bool opened = false;
for (BaseSettings *setting : LanguageClientSettings::currentPageSettings()) { for (BaseSettings *setting : LanguageClientSettings::currentPageSettings()) {
@@ -434,7 +438,7 @@ void LanguageClientManager::documentOpened(Core::IDocument *document)
continue; continue;
for (auto client : clients) { for (auto client : clients) {
if (openDocumentWithClient(document, client)) { if (openDocumentWithClient(textDocument, client)) {
opened = true; opened = true;
continue; continue;
} }
@@ -443,7 +447,8 @@ void LanguageClientManager::documentOpened(Core::IDocument *document)
} }
} }
bool LanguageClientManager::openDocumentWithClient(Core::IDocument *document, Client *client) bool LanguageClientManager::openDocumentWithClient(TextEditor::TextDocument *document,
Client *client)
{ {
if (!client || client->state() == Client::Error) if (!client || client->state() == Client::Error)
return false; return false;
@@ -455,21 +460,27 @@ bool LanguageClientManager::openDocumentWithClient(Core::IDocument *document, Cl
void LanguageClientManager::documentClosed(Core::IDocument *document) void LanguageClientManager::documentClosed(Core::IDocument *document)
{ {
for (Client *client : reachableClients()) if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) {
client->closeDocument(document); for (Client *client : reachableClients())
m_clientForDocument.remove(document); client->closeDocument(textDocument);
m_clientForDocument.remove(textDocument);
}
} }
void LanguageClientManager::documentContentsSaved(Core::IDocument *document) void LanguageClientManager::documentContentsSaved(Core::IDocument *document)
{ {
for (Client *interface : reachableClients()) if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) {
interface->documentContentsSaved(document); for (Client *interface : reachableClients())
interface->documentContentsSaved(textDocument);
}
} }
void LanguageClientManager::documentWillSave(Core::IDocument *document) void LanguageClientManager::documentWillSave(Core::IDocument *document)
{ {
for (Client *interface : reachableClients()) if (auto textDocument = qobject_cast<TextEditor::TextDocument *>(document)) {
interface->documentWillSave(document); for (Client *interface : reachableClients())
interface->documentWillSave(textDocument);
}
} }
void LanguageClientManager::findLinkAt(const Utils::FilePath &filePath, void LanguageClientManager::findLinkAt(const Utils::FilePath &filePath,

View File

@@ -70,14 +70,14 @@ public:
static LanguageClientManager *instance(); static LanguageClientManager *instance();
static QList<Client *> clientsSupportingDocument(const Core::IDocument *doc); static QList<Client *> clientsSupportingDocument(const TextEditor::TextDocument *doc);
static void applySettings(); static void applySettings();
static QList<BaseSettings *> currentSettings(); static QList<BaseSettings *> currentSettings();
static QVector<Client *> clientForSetting(const BaseSettings *setting); static QVector<Client *> clientForSetting(const BaseSettings *setting);
static const BaseSettings *settingForClient(Client *setting); static const BaseSettings *settingForClient(Client *setting);
static Client *clientForDocument(Core::IDocument *document); static Client *clientForDocument(TextEditor::TextDocument *document);
static bool reOpenDocumentWithClient(Core::IDocument *document, Client *client); static bool reOpenDocumentWithClient(TextEditor::TextDocument *document, Client *client);
signals: signals:
void shutdownFinished(); void shutdownFinished();
@@ -87,7 +87,7 @@ private:
void editorOpened(Core::IEditor *editor); void editorOpened(Core::IEditor *editor);
void documentOpened(Core::IDocument *document); void documentOpened(Core::IDocument *document);
bool openDocumentWithClient(Core::IDocument *document, Client *client); bool openDocumentWithClient(TextEditor::TextDocument *document, Client *client);
void documentClosed(Core::IDocument *document); void documentClosed(Core::IDocument *document);
void documentContentsSaved(Core::IDocument *document); void documentContentsSaved(Core::IDocument *document);
void documentWillSave(Core::IDocument *document); void documentWillSave(Core::IDocument *document);
@@ -108,7 +108,7 @@ private:
QVector<Client *> m_clients; QVector<Client *> m_clients;
QList<BaseSettings *> m_currentSettings; // owned QList<BaseSettings *> m_currentSettings; // owned
QMap<QString, QVector<Client *>> m_clientsForSetting; QMap<QString, QVector<Client *>> m_clientsForSetting;
QHash<Core::IDocument *, QPointer<Client>> m_clientForDocument; QHash<TextEditor::TextDocument *, QPointer<Client>> m_clientForDocument;
QHash<LanguageServerProtocol::MessageId, QList<Client *>> m_exclusiveRequests; QHash<LanguageServerProtocol::MessageId, QList<Client *>> m_exclusiveRequests;
DocumentLocatorFilter m_currentDocumentLocatorFilter; DocumentLocatorFilter m_currentDocumentLocatorFilter;
WorkspaceLocatorFilter m_workspaceLocatorFilter; WorkspaceLocatorFilter m_workspaceLocatorFilter;

View File

@@ -201,8 +201,8 @@ void updateEditorToolBar(Core::IEditor *editor)
if (!widget) if (!widget)
return; return;
Core::IDocument *document = editor->document(); TextDocument *document = textEditor->textDocument();
Client *client = LanguageClientManager::clientForDocument(document); Client *client = LanguageClientManager::clientForDocument(textEditor->textDocument());
static QMap<QWidget *, QAction *> actions; static QMap<QWidget *, QAction *> actions;

View File

@@ -56,10 +56,10 @@ DocumentLocatorFilter::DocumentLocatorFilter()
void DocumentLocatorFilter::updateCurrentClient() void DocumentLocatorFilter::updateCurrentClient()
{ {
Core::IDocument *document = Core::EditorManager::currentDocument();
resetSymbols(); resetSymbols();
disconnect(m_resetSymbolsConnection); disconnect(m_resetSymbolsConnection);
TextEditor::TextDocument *document = TextEditor::TextDocument::currentTextDocument();
if (Client *client = LanguageClientManager::clientForDocument(document)) { if (Client *client = LanguageClientManager::clientForDocument(document)) {
if (m_symbolCache != client->documentSymbolCache()) { if (m_symbolCache != client->documentSymbolCache()) {
disconnect(m_updateSymbolsConnection); disconnect(m_updateSymbolsConnection);