forked from qt-creator/qt-creator
Instead of checking recursively every possible object just check the required keys for an object and validate it on construction or assignment from json. This will reduce the implementation effort for protocol extensions and also reduce the false positives we might get if the protocol gets updated. Change-Id: I3df24e62430d2c7575d26c1581e6a9606e7da4c1 Reviewed-by: hjk <hjk@qt.io> Reviewed-by: Christian Stenger <christian.stenger@qt.io>
149 lines
5.5 KiB
C++
149 lines
5.5 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2019 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 "languageclientformatter.h"
|
|
|
|
#include "client.h"
|
|
#include "languageclientutils.h"
|
|
|
|
#include <texteditor/tabsettings.h>
|
|
#include <texteditor/textdocument.h>
|
|
#include <utils/mimetypes/mimedatabase.h>
|
|
|
|
#include <QTextDocument>
|
|
|
|
using namespace LanguageServerProtocol;
|
|
using namespace Utils;
|
|
|
|
namespace LanguageClient {
|
|
|
|
LanguageClientFormatter::LanguageClientFormatter(TextEditor::TextDocument *document, Client *client)
|
|
: m_client(client)
|
|
, m_document(document)
|
|
{
|
|
m_cancelConnection = QObject::connect(document->document(),
|
|
&QTextDocument::contentsChanged,
|
|
[this]() {
|
|
if (m_ignoreCancel)
|
|
m_ignoreCancel = false;
|
|
else
|
|
cancelCurrentRequest();
|
|
});
|
|
}
|
|
|
|
LanguageClientFormatter::~LanguageClientFormatter()
|
|
{
|
|
QObject::disconnect(m_cancelConnection);
|
|
cancelCurrentRequest();
|
|
}
|
|
|
|
static const FormattingOptions formattingOptions(const TextEditor::TabSettings &settings)
|
|
{
|
|
FormattingOptions options;
|
|
options.setTabSize(settings.m_tabSize);
|
|
options.setInsertSpace(settings.m_tabPolicy == TextEditor::TabSettings::SpacesOnlyTabPolicy);
|
|
return options;
|
|
}
|
|
|
|
QFutureWatcher<ChangeSet> *LanguageClientFormatter::format(
|
|
const QTextCursor &cursor, const TextEditor::TabSettings &tabSettings)
|
|
{
|
|
cancelCurrentRequest();
|
|
m_progress = QFutureInterface<ChangeSet>();
|
|
|
|
const FilePath &filePath = m_document->filePath();
|
|
const DynamicCapabilities dynamicCapabilities = m_client->dynamicCapabilities();
|
|
const QString method(DocumentRangeFormattingRequest::methodName);
|
|
if (optional<bool> registered = dynamicCapabilities.isRegistered(method)) {
|
|
if (!registered.value())
|
|
return nullptr;
|
|
const TextDocumentRegistrationOptions option(dynamicCapabilities.option(method).toObject());
|
|
if (option.isValid()
|
|
&& !option.filterApplies(filePath, Utils::mimeTypeForName(m_document->mimeType()))) {
|
|
return nullptr;
|
|
}
|
|
} else {
|
|
const Utils::optional<Utils::variant<bool, WorkDoneProgressOptions>> &provider
|
|
= m_client->capabilities().documentRangeFormattingProvider();
|
|
if (!provider.has_value())
|
|
return nullptr;
|
|
if (Utils::holds_alternative<bool>(*provider) && !Utils::get<bool>(*provider))
|
|
return nullptr;
|
|
}
|
|
DocumentRangeFormattingParams params;
|
|
const DocumentUri uri = DocumentUri::fromFilePath(filePath);
|
|
params.setTextDocument(TextDocumentIdentifier(uri));
|
|
params.setOptions(formattingOptions(tabSettings));
|
|
if (!cursor.hasSelection()) {
|
|
QTextCursor c = cursor;
|
|
c.select(QTextCursor::LineUnderCursor);
|
|
params.setRange(Range(c));
|
|
} else {
|
|
params.setRange(Range(cursor));
|
|
}
|
|
DocumentRangeFormattingRequest request(params);
|
|
request.setResponseCallback([this](const DocumentRangeFormattingRequest::Response &response) {
|
|
handleResponse(response);
|
|
});
|
|
m_currentRequest = request.id();
|
|
m_client->sendContent(request);
|
|
// ignore first contents changed, because this function is called inside a begin/endEdit block
|
|
m_ignoreCancel = true;
|
|
m_progress.reportStarted();
|
|
auto watcher = new QFutureWatcher<ChangeSet>();
|
|
watcher->setFuture(m_progress.future());
|
|
QObject::connect(watcher, &QFutureWatcher<Text::Replacements>::canceled, [this]() {
|
|
cancelCurrentRequest();
|
|
});
|
|
return watcher;
|
|
}
|
|
|
|
void LanguageClientFormatter::cancelCurrentRequest()
|
|
{
|
|
if (m_currentRequest.has_value()) {
|
|
m_progress.reportCanceled();
|
|
m_progress.reportFinished();
|
|
m_client->cancelRequest(*m_currentRequest);
|
|
m_ignoreCancel = false;
|
|
m_currentRequest = nullopt;
|
|
}
|
|
}
|
|
|
|
void LanguageClientFormatter::handleResponse(const DocumentRangeFormattingRequest::Response &response)
|
|
{
|
|
m_currentRequest = nullopt;
|
|
if (const optional<DocumentRangeFormattingRequest::Response::Error> &error = response.error())
|
|
m_client->log(*error);
|
|
ChangeSet changeSet;
|
|
if (optional<LanguageClientArray<TextEdit>> result = response.result()) {
|
|
if (!result->isNull())
|
|
changeSet = editsToChangeSet(result->toList(), m_document->document());
|
|
}
|
|
m_progress.reportResult(changeSet);
|
|
m_progress.reportFinished();
|
|
}
|
|
|
|
} // namespace LanguageClient
|