// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "fontsettings.h" #include "fontsettingspage.h" #include "texteditortr.h" #include #include #include #include #include #include #include #include #include #include #include #include using namespace Utils; const char fontFamilyKey[] = "FontFamily"; const char fontSizeKey[] = "FontSize"; const char fontZoomKey[] = "FontZoom"; const char lineSpacingKey[] = "LineSpacing"; const char antialiasKey[] = "FontAntialias"; const char schemeFileNamesKey[] = "ColorSchemes"; const bool DEFAULT_ANTIALIAS = true; const char g_sourceCodePro[] = "Source Code Pro"; namespace TextEditor { // -- FontSettings FontSettings::FontSettings() : m_family(defaultFixedFontFamily()), m_fontSize(defaultFontSize()), m_fontZoom(100), m_lineSpacing(100), m_antialias(DEFAULT_ANTIALIAS) { } void FontSettings::clear() { m_family = defaultFixedFontFamily(); m_fontSize = defaultFontSize(); m_fontZoom = 100; m_lineSpacing = 100; m_antialias = DEFAULT_ANTIALIAS; m_scheme.clear(); clearCaches(); } static Key settingsGroup() { return keyFromString(Utils::settingsKey(TextEditor::Constants::TEXT_EDITOR_SETTINGS_CATEGORY)); } void FontSettings::toSettings(QtcSettings *s) const { s->beginGroup(settingsGroup()); if (m_family != defaultFixedFontFamily() || s->contains(fontFamilyKey)) s->setValue(fontFamilyKey, m_family); if (m_fontSize != defaultFontSize() || s->contains(fontSizeKey)) s->setValue(fontSizeKey, m_fontSize); if (m_fontZoom!= 100 || s->contains(fontZoomKey)) s->setValue(fontZoomKey, m_fontZoom); if (m_lineSpacing != 100 || s->contains(lineSpacingKey)) s->setValue(lineSpacingKey, m_lineSpacing); if (m_antialias != DEFAULT_ANTIALIAS || s->contains(antialiasKey)) s->setValue(antialiasKey, m_antialias); auto schemeFileNames = s->value(schemeFileNamesKey).toMap(); if (m_schemeFileName != defaultSchemeFileName() || schemeFileNames.contains(Utils::creatorTheme()->id())) { schemeFileNames.insert(Utils::creatorTheme()->id(), m_schemeFileName.toSettings()); s->setValue(schemeFileNamesKey, schemeFileNames); } s->endGroup(); } bool FontSettings::fromSettings(const FormatDescriptions &descriptions, const QtcSettings *s) { clear(); Key group = settingsGroup(); if (!s->childGroups().contains(stringFromKey(group))) return false; group = group + '/'; m_family = s->value(group + fontFamilyKey, defaultFixedFontFamily()).toString(); m_fontSize = s->value(group + fontSizeKey, m_fontSize).toInt(); m_fontZoom= s->value(group + fontZoomKey, m_fontZoom).toInt(); m_lineSpacing = s->value(group + lineSpacingKey, m_lineSpacing).toInt(); m_antialias = s->value(group + antialiasKey, DEFAULT_ANTIALIAS).toBool(); if (s->contains(group + schemeFileNamesKey)) { // Load the selected color scheme for the current theme auto schemeFileNames = s->value(group + schemeFileNamesKey).toMap(); if (schemeFileNames.contains(Utils::creatorTheme()->id())) { const FilePath scheme = FilePath::fromSettings(schemeFileNames.value(Utils::creatorTheme()->id())); loadColorScheme(scheme, descriptions); } } return true; } bool FontSettings::equals(const FontSettings &f) const { return m_family == f.m_family && m_schemeFileName == f.m_schemeFileName && m_fontSize == f.m_fontSize && m_lineSpacing == f.m_lineSpacing && m_fontZoom == f.m_fontZoom && m_antialias == f.m_antialias && m_scheme == f.m_scheme; } auto qHash(const TextStyle &textStyle) { return ::qHash(quint8(textStyle)); } static bool isOverlayCategory(TextStyle category) { return category == C_OCCURRENCES || category == C_OCCURRENCES_RENAME || category == C_SEARCH_RESULT || category == C_SEARCH_RESULT_ALT1 || category == C_SEARCH_RESULT_ALT2 || category == C_PARENTHESES_MISMATCH; } /** * Returns the QTextCharFormat of the given format category. */ QTextCharFormat FontSettings::toTextCharFormat(TextStyle category) const { auto textCharFormatIterator = m_formatCache.find(category); if (textCharFormatIterator != m_formatCache.end()) return *textCharFormatIterator; const Format &f = m_scheme.formatFor(category); QTextCharFormat tf; if (category == C_TEXT) { tf.setFontFamily(m_family); tf.setFontPointSize(m_fontSize * m_fontZoom / 100.); tf.setFontStyleStrategy(m_antialias ? QFont::PreferAntialias : QFont::NoAntialias); } if (category == C_OCCURRENCES_UNUSED) tf.setToolTip(Tr::tr("Unused variable")); if (f.foreground().isValid() && !isOverlayCategory(category)) tf.setForeground(f.foreground()); if (f.background().isValid()) { if (category == C_TEXT || f.background() != m_scheme.formatFor(C_TEXT).background()) tf.setBackground(f.background()); } else if (isOverlayCategory(category)) { // overlays without a background schouldn't get painted tf.setBackground(QColor()); } tf.setFontWeight(f.bold() ? QFont::Bold : fontNormalWeight()); tf.setFontItalic(f.italic()); tf.setUnderlineColor(f.underlineColor()); tf.setUnderlineStyle(f.underlineStyle()); m_formatCache.insert(category, tf); return tf; } auto qHash(TextStyles textStyles) { return ::qHash(reinterpret_cast(textStyles)); } bool operator==(const TextStyles &first, const TextStyles &second) { return first.mainStyle == second.mainStyle && first.mixinStyles == second.mixinStyles; } namespace { double clamp(double value) { return std::max(0.0, std::min(1.0, value)); } QBrush mixBrush(const QBrush &original, double relativeSaturation, double relativeLightness) { const QColor originalColor = original.color().toHsl(); QColor mixedColor(QColor::Hsl); double mixedSaturation = clamp(originalColor.hslSaturationF() + relativeSaturation); double mixedLightness = clamp(originalColor.lightnessF() + relativeLightness); mixedColor.setHslF(originalColor.hslHueF(), mixedSaturation, mixedLightness); return mixedColor; } } void FontSettings::addMixinStyle(QTextCharFormat &textCharFormat, const MixinTextStyles &mixinStyles) const { const int normalWeight = fontNormalWeight(); for (TextStyle mixinStyle : mixinStyles) { const Format &format = m_scheme.formatFor(mixinStyle); if (format.foreground().isValid()) { textCharFormat.setForeground(format.foreground()); } else { if (textCharFormat.hasProperty(QTextFormat::ForegroundBrush)) { textCharFormat.setForeground(mixBrush(textCharFormat.foreground(), format.relativeForegroundSaturation(), format.relativeForegroundLightness())); } } if (format.background().isValid()) { textCharFormat.setBackground(format.background()); } else { if (textCharFormat.hasProperty(QTextFormat::BackgroundBrush)) { textCharFormat.setBackground(mixBrush(textCharFormat.background(), format.relativeBackgroundSaturation(), format.relativeBackgroundLightness())); } } if (!textCharFormat.fontItalic()) textCharFormat.setFontItalic(format.italic()); if (textCharFormat.fontWeight() == normalWeight) textCharFormat.setFontWeight(format.bold() ? QFont::Bold : normalWeight); if (textCharFormat.underlineStyle() == QTextCharFormat::NoUnderline) { textCharFormat.setUnderlineStyle(format.underlineStyle()); textCharFormat.setUnderlineColor(format.underlineColor()); } }; } void FontSettings::clearCaches() { m_formatCache.clear(); m_textCharFormatCache.clear(); } QTextCharFormat FontSettings::toTextCharFormat(TextStyles textStyles) const { auto textCharFormatIterator = m_textCharFormatCache.find(textStyles); if (textCharFormatIterator != m_textCharFormatCache.end()) return *textCharFormatIterator; QTextCharFormat textCharFormat = toTextCharFormat(textStyles.mainStyle); addMixinStyle(textCharFormat, textStyles.mixinStyles); m_textCharFormatCache.insert(textStyles, textCharFormat); return textCharFormat; } /** * Returns the list of QTextCharFormats that corresponds to the list of * requested format categories. */ QVector FontSettings::toTextCharFormats(const QVector &categories) const { QVector rc; const int size = categories.size(); rc.reserve(size); for (int i = 0; i < size; i++) rc.append(toTextCharFormat(categories.at(i))); return rc; } /** * Returns the configured font family. */ QString FontSettings::family() const { return m_family; } void FontSettings::setFamily(const QString &family) { m_family = family; clearCaches(); } /** * Returns the configured font size. */ int FontSettings::fontSize() const { return m_fontSize; } void FontSettings::setFontSize(int size) { m_fontSize = size; clearCaches(); } /** * Returns the configured font zoom factor in percent. */ int FontSettings::fontZoom() const { return m_fontZoom; } void FontSettings::setFontZoom(int zoom) { m_fontZoom = zoom; clearCaches(); } qreal FontSettings::lineSpacing() const { QFont currentFont = font(); currentFont.setPointSize(std::max(m_fontSize * m_fontZoom / 100, 1)); qreal spacing = QFontMetricsF(currentFont).lineSpacing(); if (m_lineSpacing != 100) spacing *= qreal(m_lineSpacing) / 100; return spacing; } int FontSettings::relativeLineSpacing() const { return m_lineSpacing; } void FontSettings::setRelativeLineSpacing(int relativeLineSpacing) { m_lineSpacing = relativeLineSpacing; } QFont FontSettings::font() const { QFont f(family(), fontSize()); f.setStyleStrategy(m_antialias ? QFont::PreferAntialias : QFont::NoAntialias); f.setWeight(fontNormalWeight()); return f; } QFont::Weight FontSettings::fontNormalWeight() const { // TODO: Fix this when we upgrade "Source Code Pro" to a version greater than 2.0.30 QFont::Weight weight = QFont::Normal; if (Utils::HostOsInfo::isMacHost() && m_family == g_sourceCodePro) weight = QFont::Medium; return weight; } /** * Returns the configured antialiasing behavior. */ bool FontSettings::antialias() const { return m_antialias; } void FontSettings::setAntialias(bool antialias) { m_antialias = antialias; clearCaches(); } /** * Returns the format for the given font category. */ Format &FontSettings::formatFor(TextStyle category) { return m_scheme.formatFor(category); } Format FontSettings::formatFor(TextStyle category) const { return m_scheme.formatFor(category); } /** * Returns the file name of the currently selected color scheme. */ Utils::FilePath FontSettings::colorSchemeFileName() const { return m_schemeFileName; } /** * Sets the file name of the color scheme. Does not load the scheme from the * given file. If you want to load a scheme, use loadColorScheme() instead. */ void FontSettings::setColorSchemeFileName(const Utils::FilePath &filePath) { m_schemeFileName = filePath; } bool FontSettings::loadColorScheme(const Utils::FilePath &filePath, const FormatDescriptions &descriptions) { clearCaches(); bool loaded = true; m_schemeFileName = filePath; if (!m_scheme.load(m_schemeFileName)) { loaded = false; m_schemeFileName.clear(); qWarning() << "Failed to load color scheme:" << filePath; } // Apply default formats to undefined categories for (const FormatDescription &desc : descriptions) { const TextStyle id = desc.id(); if (!m_scheme.contains(id)) { if ((id == C_NAMESPACE || id == C_CONCEPT) && m_scheme.contains(C_TYPE)) { m_scheme.setFormatFor(id, m_scheme.formatFor(C_TYPE)); continue; } if (id == C_MACRO && m_scheme.contains(C_FUNCTION)) { m_scheme.setFormatFor(C_MACRO, m_scheme.formatFor(C_FUNCTION)); continue; } Format format; const Format &descFormat = desc.format(); // Default fallback for background and foreground is C_TEXT, which is set through // the editor's palette, i.e. we leave these as invalid colors in that case if (descFormat != format || !m_scheme.contains(C_TEXT)) { format.setForeground(descFormat.foreground()); format.setBackground(descFormat.background()); } format.setRelativeForegroundSaturation(descFormat.relativeForegroundSaturation()); format.setRelativeForegroundLightness(descFormat.relativeForegroundLightness()); format.setRelativeBackgroundSaturation(descFormat.relativeBackgroundSaturation()); format.setRelativeBackgroundLightness(descFormat.relativeBackgroundLightness()); format.setBold(descFormat.bold()); format.setItalic(descFormat.italic()); format.setUnderlineColor(descFormat.underlineColor()); format.setUnderlineStyle(descFormat.underlineStyle()); m_scheme.setFormatFor(id, format); } } return loaded; } bool FontSettings::saveColorScheme(const Utils::FilePath &fileName) { const bool saved = m_scheme.save(fileName, Core::ICore::dialogParent()); if (saved) m_schemeFileName = fileName; return saved; } /** * Returns the currently active color scheme. */ const ColorScheme &FontSettings::colorScheme() const { return m_scheme; } void FontSettings::setColorScheme(const ColorScheme &scheme) { m_scheme = scheme; clearCaches(); } static QString defaultFontFamily() { if (Utils::HostOsInfo::isMacHost()) return QLatin1String("Menlo"); const QString sourceCodePro(g_sourceCodePro); const QFontDatabase dataBase; if (dataBase.hasFamily(sourceCodePro)) return sourceCodePro; if (Utils::HostOsInfo::isAnyUnixHost()) return QLatin1String("Monospace"); return QLatin1String("Courier"); } QString FontSettings::defaultFixedFontFamily() { static QString rc; if (rc.isEmpty()) { QFont f = QFont(defaultFontFamily()); f.setStyleHint(QFont::TypeWriter); rc = f.family(); } return rc; } int FontSettings::defaultFontSize() { if (Utils::HostOsInfo::isMacHost()) return 12; if (Utils::HostOsInfo::isAnyUnixHost()) return 9; return 10; } /** * Returns the default scheme file name, or the path to a shipped scheme when * one exists with the given \a fileName. */ FilePath FontSettings::defaultSchemeFileName(const QString &fileName) { FilePath defaultScheme = Core::ICore::resourcePath("styles"); if (!fileName.isEmpty() && (defaultScheme / fileName).exists()) { defaultScheme = defaultScheme / fileName; } else { const QString themeScheme = Utils::creatorTheme()->defaultTextEditorColorScheme(); if (!themeScheme.isEmpty() && (defaultScheme / themeScheme).exists()) defaultScheme = defaultScheme / themeScheme; else defaultScheme = defaultScheme / "default.xml"; } return defaultScheme; } } // namespace TextEditor