2018-11-08 10:35:23 +01:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** 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 "clangformatutils.h"
|
|
|
|
|
|
2018-11-13 09:29:09 +01:00
|
|
|
#include "clangformatconstants.h"
|
2019-03-05 13:12:44 +01:00
|
|
|
#include "clangformatsettings.h"
|
2018-11-13 09:29:09 +01:00
|
|
|
|
2018-11-08 10:35:23 +01:00
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <cpptools/cppcodestylesettings.h>
|
|
|
|
|
#include <texteditor/tabsettings.h>
|
|
|
|
|
#include <projectexplorer/project.h>
|
|
|
|
|
#include <projectexplorer/session.h>
|
|
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
#include <QCryptographicHash>
|
|
|
|
|
|
2018-11-08 10:35:23 +01:00
|
|
|
using namespace clang;
|
|
|
|
|
using namespace format;
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
using namespace CppTools;
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
using namespace TextEditor;
|
|
|
|
|
|
|
|
|
|
namespace ClangFormat {
|
|
|
|
|
|
|
|
|
|
static void applyTabSettings(clang::format::FormatStyle &style, const TabSettings &settings)
|
|
|
|
|
{
|
|
|
|
|
style.IndentWidth = static_cast<unsigned>(settings.m_indentSize);
|
|
|
|
|
style.TabWidth = static_cast<unsigned>(settings.m_tabSize);
|
|
|
|
|
|
|
|
|
|
if (settings.m_tabPolicy == TabSettings::TabsOnlyTabPolicy)
|
|
|
|
|
style.UseTab = FormatStyle::UT_Always;
|
|
|
|
|
else if (settings.m_tabPolicy == TabSettings::SpacesOnlyTabPolicy)
|
|
|
|
|
style.UseTab = FormatStyle::UT_Never;
|
|
|
|
|
else
|
|
|
|
|
style.UseTab = FormatStyle::UT_ForContinuationAndIndentation;
|
|
|
|
|
|
|
|
|
|
if (settings.m_continuationAlignBehavior == TabSettings::NoContinuationAlign) {
|
|
|
|
|
style.ContinuationIndentWidth = 0;
|
|
|
|
|
style.AlignAfterOpenBracket = FormatStyle::BAS_DontAlign;
|
|
|
|
|
} else {
|
|
|
|
|
style.ContinuationIndentWidth = style.IndentWidth;
|
|
|
|
|
style.AlignAfterOpenBracket = FormatStyle::BAS_Align;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void applyCppCodeStyleSettings(clang::format::FormatStyle &style,
|
|
|
|
|
const CppCodeStyleSettings &settings)
|
|
|
|
|
{
|
|
|
|
|
style.IndentCaseLabels = settings.indentSwitchLabels;
|
|
|
|
|
style.AlignOperands = settings.alignAssignments;
|
|
|
|
|
style.NamespaceIndentation = FormatStyle::NI_None;
|
|
|
|
|
if (settings.indentNamespaceBody)
|
|
|
|
|
style.NamespaceIndentation = FormatStyle::NI_All;
|
|
|
|
|
|
|
|
|
|
style.BraceWrapping.IndentBraces = false;
|
|
|
|
|
if (settings.indentBlockBraces) {
|
|
|
|
|
if (settings.indentClassBraces && settings.indentEnumBraces
|
|
|
|
|
&& settings.indentNamespaceBraces && settings.indentFunctionBraces) {
|
|
|
|
|
style.BraceWrapping.IndentBraces = true;
|
|
|
|
|
} else {
|
|
|
|
|
style.BreakBeforeBraces = FormatStyle::BS_GNU;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (settings.bindStarToIdentifier || settings.bindStarToRightSpecifier)
|
|
|
|
|
style.PointerAlignment = FormatStyle::PAS_Right;
|
|
|
|
|
else
|
|
|
|
|
style.PointerAlignment = FormatStyle::PAS_Left;
|
|
|
|
|
|
|
|
|
|
style.AccessModifierOffset = settings.indentAccessSpecifiers
|
|
|
|
|
? 0
|
|
|
|
|
: - static_cast<int>(style.IndentWidth);
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
static bool useGlobalOverriddenSettings()
|
|
|
|
|
{
|
|
|
|
|
return ClangFormatSettings::instance().overrideDefaultFile();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString currentProjectUniqueId()
|
2018-11-08 10:35:23 +01:00
|
|
|
{
|
|
|
|
|
const Project *project = SessionManager::startupProject();
|
2019-03-05 13:12:44 +01:00
|
|
|
if (!project)
|
|
|
|
|
return QString();
|
2018-11-08 10:35:23 +01:00
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
return QString::fromUtf8(QCryptographicHash::hash(project->projectFilePath().toString().toUtf8(),
|
|
|
|
|
QCryptographicHash::Md5)
|
|
|
|
|
.toHex(0));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool useProjectOverriddenSettings()
|
|
|
|
|
{
|
|
|
|
|
const Project *project = SessionManager::startupProject();
|
|
|
|
|
return project ? project->namedSettings(Constants::OVERRIDE_FILE_ID).toBool() : false;
|
2018-11-08 10:35:23 +01:00
|
|
|
}
|
|
|
|
|
|
2018-11-13 11:47:02 +01:00
|
|
|
static Utils::FileName globalPath()
|
2018-11-08 10:35:23 +01:00
|
|
|
{
|
|
|
|
|
return Utils::FileName::fromString(Core::ICore::userResourcePath());
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
static Utils::FileName projectPath()
|
2019-01-22 14:16:30 +01:00
|
|
|
{
|
2019-03-05 13:12:44 +01:00
|
|
|
const Project *project = SessionManager::startupProject();
|
|
|
|
|
if (project)
|
|
|
|
|
return globalPath().appendPath("clang-format").appendPath(currentProjectUniqueId());
|
2019-02-21 13:45:18 +01:00
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
return Utils::FileName();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QString configForFile(Utils::FileName fileName, bool checkForSettings)
|
|
|
|
|
{
|
|
|
|
|
QDir overrideDir;
|
|
|
|
|
if (!checkForSettings || useProjectOverriddenSettings()) {
|
|
|
|
|
overrideDir = projectPath().toString();
|
|
|
|
|
if (!overrideDir.isEmpty() && overrideDir.exists(Constants::SETTINGS_FILE_NAME))
|
|
|
|
|
return overrideDir.filePath(Constants::SETTINGS_FILE_NAME);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!checkForSettings || useGlobalOverriddenSettings()) {
|
|
|
|
|
overrideDir = globalPath().toString();
|
|
|
|
|
if (!overrideDir.isEmpty() && overrideDir.exists(Constants::SETTINGS_FILE_NAME))
|
|
|
|
|
return overrideDir.filePath(Constants::SETTINGS_FILE_NAME);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QDir parentDir(fileName.parentDir().toString());
|
|
|
|
|
while (!parentDir.exists(Constants::SETTINGS_FILE_NAME)
|
|
|
|
|
&& !parentDir.exists(Constants::SETTINGS_FILE_ALT_NAME)) {
|
|
|
|
|
if (!parentDir.cdUp())
|
2019-02-21 13:45:18 +01:00
|
|
|
return QString();
|
2018-11-13 11:47:02 +01:00
|
|
|
}
|
2019-02-21 13:45:18 +01:00
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
if (parentDir.exists(Constants::SETTINGS_FILE_NAME))
|
|
|
|
|
return parentDir.filePath(Constants::SETTINGS_FILE_NAME);
|
|
|
|
|
return parentDir.filePath(Constants::SETTINGS_FILE_ALT_NAME);
|
2018-11-08 10:35:23 +01:00
|
|
|
}
|
|
|
|
|
|
2019-02-21 13:45:18 +01:00
|
|
|
static clang::format::FormatStyle constructStyle(bool isGlobal,
|
|
|
|
|
const QByteArray &baseStyle = QByteArray())
|
2018-11-08 10:35:23 +01:00
|
|
|
{
|
2019-02-21 13:45:18 +01:00
|
|
|
if (!baseStyle.isEmpty()) {
|
|
|
|
|
// Try to get the style for this base style.
|
|
|
|
|
Expected<FormatStyle> style = getStyle(baseStyle.toStdString(),
|
|
|
|
|
"dummy.cpp",
|
|
|
|
|
baseStyle.toStdString());
|
|
|
|
|
if (style)
|
|
|
|
|
return *style;
|
|
|
|
|
|
|
|
|
|
handleAllErrors(style.takeError(), [](const ErrorInfoBase &) {
|
|
|
|
|
// do nothing
|
|
|
|
|
});
|
|
|
|
|
// Fallthrough to the default style.
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-08 10:35:23 +01:00
|
|
|
FormatStyle style = getLLVMStyle();
|
2018-11-13 08:01:13 +01:00
|
|
|
style.BreakBeforeBraces = FormatStyle::BS_Custom;
|
|
|
|
|
|
2018-11-08 10:35:23 +01:00
|
|
|
const CppCodeStyleSettings codeStyleSettings = isGlobal
|
|
|
|
|
? CppCodeStyleSettings::currentGlobalCodeStyle()
|
2018-11-08 16:21:54 +01:00
|
|
|
: CppCodeStyleSettings::currentProjectCodeStyle()
|
|
|
|
|
.value_or(CppCodeStyleSettings::currentGlobalCodeStyle());
|
2018-11-08 10:35:23 +01:00
|
|
|
const TabSettings tabSettings = isGlobal
|
|
|
|
|
? CppCodeStyleSettings::currentGlobalTabSettings()
|
|
|
|
|
: CppCodeStyleSettings::currentProjectTabSettings();
|
|
|
|
|
|
|
|
|
|
applyTabSettings(style, tabSettings);
|
|
|
|
|
applyCppCodeStyleSettings(style, codeStyleSettings);
|
|
|
|
|
|
|
|
|
|
return style;
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-13 11:47:02 +01:00
|
|
|
void createStyleFileIfNeeded(bool isGlobal)
|
2018-11-08 10:35:23 +01:00
|
|
|
{
|
2018-11-13 11:47:02 +01:00
|
|
|
Utils::FileName path = isGlobal ? globalPath() : projectPath();
|
2019-01-22 14:16:30 +01:00
|
|
|
const QString configFile = path.appendPath(Constants::SETTINGS_FILE_NAME).toString();
|
2018-11-13 11:47:02 +01:00
|
|
|
|
2018-11-08 10:35:23 +01:00
|
|
|
if (QFile::exists(configFile))
|
|
|
|
|
return;
|
|
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
QDir().mkpath(path.parentDir().toString());
|
2018-11-08 10:35:23 +01:00
|
|
|
std::fstream newStyleFile(configFile.toStdString(), std::fstream::out);
|
|
|
|
|
if (newStyleFile.is_open()) {
|
|
|
|
|
newStyleFile << clang::format::configurationAsText(constructStyle(isGlobal));
|
|
|
|
|
newStyleFile.close();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-21 13:45:18 +01:00
|
|
|
static QByteArray configBaseStyleName(const QString &configFile)
|
|
|
|
|
{
|
|
|
|
|
if (configFile.isEmpty())
|
|
|
|
|
return QByteArray();
|
|
|
|
|
|
|
|
|
|
QFile config(configFile);
|
|
|
|
|
if (!config.open(QIODevice::ReadOnly))
|
|
|
|
|
return QByteArray();
|
|
|
|
|
|
|
|
|
|
const QByteArray content = config.readAll();
|
|
|
|
|
const char basedOnStyle[] = "BasedOnStyle:";
|
|
|
|
|
int basedOnStyleIndex = content.indexOf(basedOnStyle);
|
|
|
|
|
if (basedOnStyleIndex < 0)
|
|
|
|
|
return QByteArray();
|
|
|
|
|
|
|
|
|
|
basedOnStyleIndex += sizeof(basedOnStyle) - 1;
|
|
|
|
|
const int endOfLineIndex = content.indexOf('\n', basedOnStyleIndex);
|
|
|
|
|
return content
|
|
|
|
|
.mid(basedOnStyleIndex, endOfLineIndex < 0 ? -1 : endOfLineIndex - basedOnStyleIndex)
|
|
|
|
|
.trimmed();
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
static clang::format::FormatStyle styleForFile(Utils::FileName fileName, bool checkForSettings)
|
2018-11-08 10:35:23 +01:00
|
|
|
{
|
2019-03-05 13:12:44 +01:00
|
|
|
QString configFile = configForFile(fileName, checkForSettings);
|
|
|
|
|
if (configFile.isEmpty())
|
|
|
|
|
return constructStyle(true);
|
2018-11-08 10:35:23 +01:00
|
|
|
|
2018-11-13 11:47:02 +01:00
|
|
|
Expected<FormatStyle> style = format::getStyle("file",
|
|
|
|
|
fileName.toString().toStdString(),
|
|
|
|
|
"none");
|
2018-11-08 10:35:23 +01:00
|
|
|
if (style)
|
|
|
|
|
return *style;
|
|
|
|
|
|
|
|
|
|
handleAllErrors(style.takeError(), [](const ErrorInfoBase &) {
|
|
|
|
|
// do nothing
|
|
|
|
|
});
|
|
|
|
|
|
2019-03-05 13:12:44 +01:00
|
|
|
return constructStyle(true, configBaseStyleName(configFile));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clang::format::FormatStyle styleForFile(Utils::FileName fileName)
|
|
|
|
|
{
|
|
|
|
|
return styleForFile(fileName, true);
|
2018-11-08 10:35:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clang::format::FormatStyle currentProjectStyle()
|
|
|
|
|
{
|
2019-03-05 13:12:44 +01:00
|
|
|
return styleForFile(projectPath().appendPath(Constants::SAMPLE_FILE_NAME), false);
|
2018-11-08 10:35:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clang::format::FormatStyle currentGlobalStyle()
|
|
|
|
|
{
|
2019-03-05 13:12:44 +01:00
|
|
|
return styleForFile(globalPath().appendPath(Constants::SAMPLE_FILE_NAME), false);
|
2018-11-08 10:35:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|