2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2019 The Qt Company Ltd.
|
|
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0
|
2018-08-29 15:58:13 +02:00
|
|
|
|
|
|
|
|
#include "clangformatindenter.h"
|
2022-05-12 12:51:51 +02:00
|
|
|
#include "clangformatconstants.h"
|
2019-01-28 07:54:05 +01:00
|
|
|
#include "clangformatsettings.h"
|
2018-10-11 14:11:43 +02:00
|
|
|
#include "clangformatutils.h"
|
|
|
|
|
|
2022-05-12 12:51:51 +02:00
|
|
|
#include <coreplugin/icore.h>
|
2022-12-12 15:30:29 +01:00
|
|
|
|
|
|
|
|
#include <cppeditor/cppcodestylepreferencesfactory.h>
|
|
|
|
|
|
2022-05-12 12:51:51 +02:00
|
|
|
#include <extensionsystem/pluginmanager.h>
|
|
|
|
|
#include <extensionsystem/pluginspec.h>
|
2022-12-12 15:30:29 +01:00
|
|
|
|
|
|
|
|
#include <utils/genericconstants.h>
|
|
|
|
|
|
|
|
|
|
#include <projectexplorer/project.h>
|
|
|
|
|
#include <projectexplorer/session.h>
|
|
|
|
|
|
2019-01-16 09:37:54 +01:00
|
|
|
#include <texteditor/tabsettings.h>
|
2019-01-28 08:13:33 +01:00
|
|
|
#include <texteditor/textdocumentlayout.h>
|
2018-11-21 14:54:47 +01:00
|
|
|
|
2018-08-29 15:58:13 +02:00
|
|
|
using namespace clang;
|
|
|
|
|
using namespace format;
|
|
|
|
|
using namespace TextEditor;
|
|
|
|
|
|
|
|
|
|
namespace ClangFormat {
|
|
|
|
|
|
2022-05-12 12:51:51 +02:00
|
|
|
static bool isBeautifierPluginActivated()
|
|
|
|
|
{
|
|
|
|
|
const QVector<ExtensionSystem::PluginSpec *> specs = ExtensionSystem::PluginManager::plugins();
|
|
|
|
|
return std::find_if(specs.begin(),
|
|
|
|
|
specs.end(),
|
|
|
|
|
[](ExtensionSystem::PluginSpec *spec) {
|
|
|
|
|
return spec->name() == "Beautifier";
|
|
|
|
|
})
|
|
|
|
|
!= specs.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool isBeautifierOnSaveActivated()
|
|
|
|
|
{
|
|
|
|
|
if (!isBeautifierPluginActivated())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
QSettings *s = Core::ICore::settings();
|
|
|
|
|
bool activated = false;
|
|
|
|
|
s->beginGroup(Utils::Constants::BEAUTIFIER_SETTINGS_GROUP);
|
|
|
|
|
s->beginGroup(Utils::Constants::BEAUTIFIER_GENERAL_GROUP);
|
|
|
|
|
if (s->value(Utils::Constants::BEAUTIFIER_AUTO_FORMAT_ON_SAVE, false).toBool())
|
|
|
|
|
activated = true;
|
|
|
|
|
s->endGroup();
|
|
|
|
|
s->endGroup();
|
|
|
|
|
return activated;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-16 09:37:54 +01:00
|
|
|
ClangFormatIndenter::ClangFormatIndenter(QTextDocument *doc)
|
|
|
|
|
: ClangFormatBaseIndenter(doc)
|
|
|
|
|
{}
|
2018-08-29 15:58:13 +02:00
|
|
|
|
2019-01-28 07:54:05 +01:00
|
|
|
bool ClangFormatIndenter::formatCodeInsteadOfIndent() const
|
|
|
|
|
{
|
2022-05-12 12:51:51 +02:00
|
|
|
return ClangFormatSettings::instance().mode() == ClangFormatSettings::Mode::Formatting;
|
2019-01-28 08:13:33 +01:00
|
|
|
}
|
|
|
|
|
|
2022-08-26 10:30:00 +02:00
|
|
|
std::optional<TabSettings> ClangFormatIndenter::tabSettings() const
|
2018-08-29 15:58:13 +02:00
|
|
|
{
|
2019-03-05 13:12:44 +01:00
|
|
|
FormatStyle style = styleForFile();
|
2018-08-29 15:58:13 +02:00
|
|
|
TabSettings tabSettings;
|
|
|
|
|
|
|
|
|
|
switch (style.UseTab) {
|
|
|
|
|
case FormatStyle::UT_Never:
|
|
|
|
|
tabSettings.m_tabPolicy = TabSettings::SpacesOnlyTabPolicy;
|
2018-09-12 12:57:08 +02:00
|
|
|
break;
|
2018-08-29 15:58:13 +02:00
|
|
|
case FormatStyle::UT_Always:
|
|
|
|
|
tabSettings.m_tabPolicy = TabSettings::TabsOnlyTabPolicy;
|
2018-09-12 12:57:08 +02:00
|
|
|
break;
|
2018-08-29 15:58:13 +02:00
|
|
|
default:
|
|
|
|
|
tabSettings.m_tabPolicy = TabSettings::MixedTabPolicy;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-11 14:11:43 +02:00
|
|
|
tabSettings.m_tabSize = static_cast<int>(style.TabWidth);
|
|
|
|
|
tabSettings.m_indentSize = static_cast<int>(style.IndentWidth);
|
2018-08-29 15:58:13 +02:00
|
|
|
|
2018-11-08 10:35:23 +01:00
|
|
|
if (style.AlignAfterOpenBracket == FormatStyle::BAS_DontAlign)
|
|
|
|
|
tabSettings.m_continuationAlignBehavior = TabSettings::NoContinuationAlign;
|
2018-08-29 15:58:13 +02:00
|
|
|
else
|
|
|
|
|
tabSettings.m_continuationAlignBehavior = TabSettings::ContinuationAlignWithIndent;
|
|
|
|
|
|
|
|
|
|
return tabSettings;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-28 08:13:33 +01:00
|
|
|
int ClangFormatIndenter::lastSaveRevision() const
|
|
|
|
|
{
|
2019-02-21 12:13:46 +01:00
|
|
|
auto *layout = qobject_cast<TextEditor::TextDocumentLayout *>(m_doc->documentLayout());
|
|
|
|
|
if (!layout)
|
|
|
|
|
return 0;
|
|
|
|
|
return layout->lastSaveRevision;
|
2019-01-28 08:13:33 +01:00
|
|
|
}
|
|
|
|
|
|
2019-01-28 12:25:36 +01:00
|
|
|
bool ClangFormatIndenter::formatOnSave() const
|
|
|
|
|
{
|
2022-06-28 16:17:02 +02:00
|
|
|
return ClangFormatSettings::instance().formatOnSave() && !isBeautifierOnSaveActivated()
|
|
|
|
|
&& formatCodeInsteadOfIndent();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ClangFormatIndenter::formatWhileTyping() const
|
|
|
|
|
{
|
|
|
|
|
return ClangFormatSettings::instance().formatWhileTyping() && formatCodeInsteadOfIndent();
|
2019-01-28 12:25:36 +01:00
|
|
|
}
|
|
|
|
|
|
2022-12-12 15:30:29 +01:00
|
|
|
// ClangFormatIndenterWrapper
|
|
|
|
|
|
|
|
|
|
ClangFormatForwardingIndenter::ClangFormatForwardingIndenter(QTextDocument *doc)
|
|
|
|
|
: TextEditor::Indenter(doc)
|
|
|
|
|
, m_clangFormatIndenter(std::make_unique<ClangFormatIndenter>(doc))
|
|
|
|
|
, m_cppIndenter(CppEditor::CppCodeStylePreferencesFactory().createIndenter(doc))
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
ClangFormatForwardingIndenter::~ClangFormatForwardingIndenter() = default;
|
|
|
|
|
|
|
|
|
|
void ClangFormatForwardingIndenter::setFileName(const Utils::FilePath &fileName)
|
|
|
|
|
{
|
|
|
|
|
m_fileName = fileName;
|
|
|
|
|
m_clangFormatIndenter->setFileName(fileName);
|
|
|
|
|
m_cppIndenter->setFileName(fileName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TextEditor::Indenter *ClangFormatForwardingIndenter::currentIndenter() const
|
|
|
|
|
{
|
|
|
|
|
const ProjectExplorer::Project *projectForFile
|
|
|
|
|
= ProjectExplorer::SessionManager::projectForFile(m_fileName);
|
|
|
|
|
|
|
|
|
|
ClangFormatSettings::Mode mode = projectForFile ? static_cast<ClangFormatSettings::Mode>(
|
|
|
|
|
projectForFile->namedSettings(Constants::MODE_ID).toInt())
|
|
|
|
|
: ClangFormatSettings::instance().mode();
|
|
|
|
|
|
|
|
|
|
if (mode == ClangFormatSettings::Disable)
|
|
|
|
|
return m_cppIndenter.get();
|
|
|
|
|
|
|
|
|
|
return m_clangFormatIndenter.get();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ClangFormatForwardingIndenter::isElectricCharacter(const QChar &ch) const
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->isElectricCharacter(ch);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangFormatForwardingIndenter::setCodeStylePreferences(
|
|
|
|
|
TextEditor::ICodeStylePreferences *preferences)
|
|
|
|
|
{
|
|
|
|
|
currentIndenter()->setCodeStylePreferences(preferences);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangFormatForwardingIndenter::invalidateCache()
|
|
|
|
|
{
|
|
|
|
|
currentIndenter()->invalidateCache();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int ClangFormatForwardingIndenter::indentFor(const QTextBlock &block,
|
|
|
|
|
const TextEditor::TabSettings &tabSettings,
|
|
|
|
|
int cursorPositionInEditor)
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->indentFor(block, tabSettings, cursorPositionInEditor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int ClangFormatForwardingIndenter::visualIndentFor(const QTextBlock &block,
|
|
|
|
|
const TextEditor::TabSettings &tabSettings)
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->visualIndentFor(block, tabSettings);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangFormatForwardingIndenter::autoIndent(const QTextCursor &cursor,
|
|
|
|
|
const TextEditor::TabSettings &tabSettings,
|
|
|
|
|
int cursorPositionInEditor)
|
|
|
|
|
{
|
|
|
|
|
currentIndenter()->autoIndent(cursor, tabSettings, cursorPositionInEditor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Utils::Text::Replacements ClangFormatForwardingIndenter::format(
|
|
|
|
|
const TextEditor::RangesInLines &rangesInLines)
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->format(rangesInLines);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool ClangFormatForwardingIndenter::formatOnSave() const
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->formatOnSave();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TextEditor::IndentationForBlock ClangFormatForwardingIndenter::indentationForBlocks(
|
|
|
|
|
const QVector<QTextBlock> &blocks,
|
|
|
|
|
const TextEditor::TabSettings &tabSettings,
|
|
|
|
|
int cursorPositionInEditor)
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->indentationForBlocks(blocks, tabSettings, cursorPositionInEditor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::optional<TextEditor::TabSettings> ClangFormatForwardingIndenter::tabSettings() const
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->tabSettings();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangFormatForwardingIndenter::indentBlock(const QTextBlock &block,
|
|
|
|
|
const QChar &typedChar,
|
|
|
|
|
const TextEditor::TabSettings &tabSettings,
|
|
|
|
|
int cursorPositionInEditor)
|
|
|
|
|
{
|
|
|
|
|
currentIndenter()->indentBlock(block, typedChar, tabSettings, cursorPositionInEditor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangFormatForwardingIndenter::indent(const QTextCursor &cursor,
|
|
|
|
|
const QChar &typedChar,
|
|
|
|
|
const TextEditor::TabSettings &tabSettings,
|
|
|
|
|
int cursorPositionInEditor)
|
|
|
|
|
{
|
|
|
|
|
currentIndenter()->indent(cursor, typedChar, tabSettings, cursorPositionInEditor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClangFormatForwardingIndenter::reindent(const QTextCursor &cursor,
|
|
|
|
|
const TextEditor::TabSettings &tabSettings,
|
|
|
|
|
int cursorPositionInEditor)
|
|
|
|
|
{
|
|
|
|
|
currentIndenter()->reindent(cursor, tabSettings, cursorPositionInEditor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::optional<int> ClangFormatForwardingIndenter::margin() const
|
|
|
|
|
{
|
|
|
|
|
return currentIndenter()->margin();
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-29 15:58:13 +02:00
|
|
|
} // namespace ClangFormat
|