diff options
Diffstat (limited to 'widget/nsXPLookAndFeel.cpp')
-rw-r--r-- | widget/nsXPLookAndFeel.cpp | 1588 |
1 files changed, 1588 insertions, 0 deletions
diff --git a/widget/nsXPLookAndFeel.cpp b/widget/nsXPLookAndFeel.cpp new file mode 100644 index 0000000000..fae1444161 --- /dev/null +++ b/widget/nsXPLookAndFeel.cpp @@ -0,0 +1,1588 @@ +/* -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "mozilla/ArrayUtils.h" + +#include "mozilla/LookAndFeel.h" +#include "nscore.h" + +#include "nsXPLookAndFeel.h" +#include "nsLookAndFeel.h" +#include "HeadlessLookAndFeel.h" +#include "RemoteLookAndFeel.h" +#include "nsContentUtils.h" +#include "nsCRT.h" +#include "nsFont.h" +#include "nsIFrame.h" +#include "nsIXULRuntime.h" +#include "nsLayoutUtils.h" +#include "Theme.h" +#include "SurfaceCacheUtils.h" +#include "mozilla/dom/ContentParent.h" +#include "mozilla/dom/ContentChild.h" +#include "mozilla/Preferences.h" +#include "mozilla/Services.h" +#include "mozilla/ServoStyleSet.h" +#include "mozilla/ServoCSSParser.h" +#include "mozilla/StaticPrefs_browser.h" +#include "mozilla/StaticPrefs_editor.h" +#include "mozilla/StaticPrefs_layout.h" +#include "mozilla/StaticPrefs_ui.h" +#include "mozilla/StaticPrefs_widget.h" +#include "mozilla/dom/Document.h" +#include "mozilla/PreferenceSheet.h" +#include "mozilla/gfx/2D.h" +#include "mozilla/widget/WidgetMessageUtils.h" +#include "mozilla/dom/KeyboardEventBinding.h" +#include "mozilla/RelativeLuminanceUtils.h" +#include "mozilla/Telemetry.h" +#include "mozilla/TelemetryScalarEnums.h" + +#include "gfxPlatform.h" +#include "gfxFont.h" + +#include "qcms.h" + +#include <bitset> + +using namespace mozilla; + +using IntID = mozilla::LookAndFeel::IntID; +using FloatID = mozilla::LookAndFeel::FloatID; +using ColorID = mozilla::LookAndFeel::ColorID; +using FontID = mozilla::LookAndFeel::FontID; + +template <typename Index, typename Value, Index kEnd> +class EnumeratedCache { + mozilla::EnumeratedArray<Index, kEnd, Value> mEntries; + std::bitset<size_t(kEnd)> mValidity; + + public: + constexpr EnumeratedCache() = default; + + bool IsValid(Index aIndex) const { return mValidity[size_t(aIndex)]; } + + const Value* Get(Index aIndex) const { + return IsValid(aIndex) ? &mEntries[aIndex] : nullptr; + } + + void Insert(Index aIndex, Value aValue) { + mValidity[size_t(aIndex)] = true; + mEntries[aIndex] = aValue; + } + + void Remove(Index aIndex) { + mValidity[size_t(aIndex)] = false; + mEntries[aIndex] = Value(); + } + + void Clear() { + mValidity.reset(); + for (auto& entry : mEntries) { + entry = Value(); + } + } +}; + +using ColorCache = EnumeratedCache<ColorID, Maybe<nscolor>, ColorID::End>; + +struct ColorCaches { + using UseStandins = LookAndFeel::UseStandins; + + ColorCache mCaches[2][2]; + + constexpr ColorCaches() = default; + + ColorCache& Get(ColorScheme aScheme, UseStandins aUseStandins) { + return mCaches[aScheme == ColorScheme::Dark] + [aUseStandins == UseStandins::Yes]; + } + + void Clear() { + for (auto& c : mCaches) { + for (auto& cache : c) { + cache.Clear(); + } + } + } +}; + +static ColorCaches sColorCaches; + +static EnumeratedCache<FloatID, Maybe<float>, FloatID::End> sFloatCache; +static EnumeratedCache<IntID, Maybe<int32_t>, IntID::End> sIntCache; +static EnumeratedCache<FontID, widget::LookAndFeelFont, FontID::End> sFontCache; + +// To make one of these prefs toggleable from a reftest add a user +// pref in testing/profiles/reftest/user.js. For example, to make +// ui.useAccessibilityTheme toggleable, add: +// +// user_pref("ui.useAccessibilityTheme", 0); +// +// This needs to be of the same length and in the same order as +// LookAndFeel::IntID values. +static const char sIntPrefs[][45] = { + "ui.caretBlinkTime", + "ui.caretBlinkCount", + "ui.caretWidth", + "ui.caretVisibleWithSelection", + "ui.selectTextfieldsOnKeyFocus", + "ui.submenuDelay", + "ui.menusCanOverlapOSBar", + "ui.useOverlayScrollbars", + "ui.allowOverlayScrollbarsOverlap", + "ui.skipNavigatingDisabledMenuItem", + "ui.dragThresholdX", + "ui.dragThresholdY", + "ui.useAccessibilityTheme", + "ui.scrollArrowStyle", + "ui.scrollButtonLeftMouseButtonAction", + "ui.scrollButtonMiddleMouseButtonAction", + "ui.scrollButtonRightMouseButtonAction", + "ui.treeOpenDelay", + "ui.treeCloseDelay", + "ui.treeLazyScrollDelay", + "ui.treeScrollDelay", + "ui.treeScrollLinesMax", + "accessibility.tabfocus", // Weird one... + "ui.chosenMenuItemsShouldBlink", + "ui.windowsAccentColorInTitlebar", + "ui.windowsDefaultTheme", + "ui.dwmCompositor", + "ui.windowsClassic", + "ui.windowsGlass", + "ui.macGraphiteTheme", + "ui.macBigSurTheme", + "ui.macRTL", + "ui.alertNotificationOrigin", + "ui.scrollToClick", + "ui.IMERawInputUnderlineStyle", + "ui.IMESelectedRawTextUnderlineStyle", + "ui.IMEConvertedTextUnderlineStyle", + "ui.IMESelectedConvertedTextUnderlineStyle", + "ui.SpellCheckerUnderlineStyle", + "ui.menuBarDrag", + "ui.scrollbarButtonAutoRepeatBehavior", + "ui.tooltipDelay", + "ui.swipeAnimationEnabled", + "ui.scrollbarDisplayOnMouseMove", + "ui.scrollbarFadeBeginDelay", + "ui.scrollbarFadeDuration", + "ui.contextMenuOffsetVertical", + "ui.contextMenuOffsetHorizontal", + "ui.GtkCSDAvailable", + "ui.GtkCSDMinimizeButton", + "ui.GtkCSDMaximizeButton", + "ui.GtkCSDCloseButton", + "ui.GtkCSDMinimizeButtonPosition", + "ui.GtkCSDMaximizeButtonPosition", + "ui.GtkCSDCloseButtonPosition", + "ui.GtkCSDReversedPlacement", + "ui.systemUsesDarkTheme", + "ui.prefersReducedMotion", + "ui.prefersReducedTransparency", + "ui.invertedColors", + "ui.primaryPointerCapabilities", + "ui.allPointerCapabilities", + "ui.systemScrollbarSize", + "ui.touchDeviceSupportPresent", + "ui.titlebarRadius", + "ui.dynamicRange", + "ui.videoDynamicRange", + "ui.panelAnimations", +}; + +static_assert(ArrayLength(sIntPrefs) == size_t(LookAndFeel::IntID::End), + "Should have a pref for each int value"); + +// This array MUST be kept in the same order as the float id list in +// LookAndFeel.h +// clang-format off +static const char sFloatPrefs[][37] = { + "ui.IMEUnderlineRelativeSize", + "ui.SpellCheckerUnderlineRelativeSize", + "ui.caretAspectRatio", + "ui.textScaleFactor", + "ui.cursorScale", +}; +// clang-format on + +static_assert(ArrayLength(sFloatPrefs) == size_t(LookAndFeel::FloatID::End), + "Should have a pref for each float value"); + +// This array MUST be kept in the same order as the color list in +// specified/color.rs +static const char sColorPrefs[][41] = { + "ui.activeborder", + "ui.activecaption", + "ui.appworkspace", + "ui.background", + "ui.buttonface", + "ui.buttonhighlight", + "ui.buttonshadow", + "ui.buttontext", + "ui.buttonborder", + "ui.captiontext", + "ui.-moz-field", + "ui.-moz-disabledfield", + "ui.-moz-fieldtext", + "ui.mark", + "ui.marktext", + "ui.-moz-comboboxtext", + "ui.-moz-combobox", + "ui.graytext", + "ui.highlight", + "ui.highlighttext", + "ui.inactiveborder", + "ui.inactivecaption", + "ui.inactivecaptiontext", + "ui.infobackground", + "ui.infotext", + "ui.menu", + "ui.menutext", + "ui.scrollbar", + "ui.threeddarkshadow", + "ui.threedface", + "ui.threedhighlight", + "ui.threedlightshadow", + "ui.threedshadow", + "ui.window", + "ui.windowframe", + "ui.windowtext", + "ui.-moz-buttondefault", + "ui.-moz-default-color", + "ui.-moz-default-background-color", + "ui.-moz-dialog", + "ui.-moz-dialogtext", + "ui.-moz-dragtargetzone", + "ui.-moz-cellhighlight", + "ui.-moz_cellhighlighttext", + "ui.selecteditem", + "ui.selecteditemtext", + "ui.-moz-buttonhoverface", + "ui.-moz_buttonhovertext", + "ui.-moz_menuhover", + "ui.-moz_menuhoverdisabled", + "ui.-moz_menuhovertext", + "ui.-moz_menubartext", + "ui.-moz_menubarhovertext", + "ui.-moz_eventreerow", + "ui.-moz_oddtreerow", + "ui.-moz-buttonactivetext", + "ui.-moz-buttonactiveface", + "ui.-moz-buttondisabledface", + "ui.-moz_mac_chrome_active", + "ui.-moz_mac_chrome_inactive", + "ui.-moz-mac-defaultbuttontext", + "ui.-moz-mac-focusring", + "ui.-moz-mac-menuselect", + "ui.-moz-mac-menushadow", + "ui.-moz-mac-menutextdisable", + "ui.-moz-mac-menutextselect", + "ui.-moz_mac_disabledtoolbartext", + "ui.-moz-mac-secondaryhighlight", + "ui.-moz-mac-menupopup", + "ui.-moz-mac-menuitem", + "ui.-moz-mac-active-menuitem", + "ui.-moz-mac-source-list", + "ui.-moz-mac-source-list-selection", + "ui.-moz-mac-active-source-list-selection", + "ui.-moz-mac-tooltip", + "ui.accentcolor", + "ui.accentcolortext", + "ui.-moz-autofill-background", + "ui.-moz-win-mediatext", + "ui.-moz-win-communicationstext", + "ui.-moz-nativehyperlinktext", + "ui.-moz-nativevisitedhyperlinktext", + "ui.-moz-hyperlinktext", + "ui.-moz-activehyperlinktext", + "ui.-moz-visitedhyperlinktext", + "ui.-moz-colheadertext", + "ui.-moz-colheaderhovertext", + "ui.textSelectDisabledBackground", + "ui.textSelectAttentionBackground", + "ui.textSelectAttentionForeground", + "ui.textHighlightBackground", + "ui.textHighlightForeground", + "ui.IMERawInputBackground", + "ui.IMERawInputForeground", + "ui.IMERawInputUnderline", + "ui.IMESelectedRawTextBackground", + "ui.IMESelectedRawTextForeground", + "ui.IMESelectedRawTextUnderline", + "ui.IMEConvertedTextBackground", + "ui.IMEConvertedTextForeground", + "ui.IMEConvertedTextUnderline", + "ui.IMESelectedConvertedTextBackground", + "ui.IMESelectedConvertedTextForeground", + "ui.IMESelectedConvertedTextUnderline", + "ui.SpellCheckerUnderline", + "ui.themedScrollbar", + "ui.themedScrollbarInactive", + "ui.themedScrollbarThumb", + "ui.themedScrollbarThumbHover", + "ui.themedScrollbarThumbActive", + "ui.themedScrollbarThumbInactive", +}; + +static_assert(ArrayLength(sColorPrefs) == size_t(LookAndFeel::ColorID::End), + "Should have a pref for each color value"); + +// This array MUST be kept in the same order as the SystemFont enum. +static const char sFontPrefs[][41] = { + "ui.font.caption", + "ui.font.icon", + "ui.font.menu", + "ui.font.message-box", + "ui.font.small-caption", + "ui.font.status-bar", + "ui.font.-moz-pull-down-menu", + "ui.font.-moz-button", + "ui.font.-moz-list", + "ui.font.-moz-field", +}; + +static_assert(ArrayLength(sFontPrefs) == size_t(LookAndFeel::FontID::End), + "Should have a pref for each font value"); + +const char* nsXPLookAndFeel::GetColorPrefName(ColorID aId) { + return sColorPrefs[size_t(aId)]; +} + +bool nsXPLookAndFeel::sInitialized = false; + +nsXPLookAndFeel* nsXPLookAndFeel::sInstance = nullptr; +bool nsXPLookAndFeel::sShutdown = false; + +auto LookAndFeel::SystemZoomSettings() -> ZoomSettings { + ZoomSettings settings; + switch (StaticPrefs::browser_display_os_zoom_behavior()) { + case 0: + default: + break; + case 1: + settings.mFullZoom = GetTextScaleFactor(); + break; + case 2: + settings.mTextZoom = GetTextScaleFactor(); + break; + } + return settings; +} + +// static +nsXPLookAndFeel* nsXPLookAndFeel::GetInstance() { + if (sInstance) { + return sInstance; + } + + NS_ENSURE_TRUE(!sShutdown, nullptr); + + // If we're in a content process, then the parent process will have supplied + // us with an initial FullLookAndFeel object. + // We grab this data from the ContentChild, + // where it's been temporarily stashed, and initialize our new LookAndFeel + // object with it. + + FullLookAndFeel* lnf = nullptr; + + if (auto* cc = mozilla::dom::ContentChild::GetSingleton()) { + lnf = &cc->BorrowLookAndFeelData(); + } + + if (lnf) { + sInstance = new widget::RemoteLookAndFeel(std::move(*lnf)); + } else if (gfxPlatform::IsHeadless()) { + sInstance = new widget::HeadlessLookAndFeel(); + } else { + sInstance = new nsLookAndFeel(); + } + + // This is only ever used once during initialization, and can be cleared now. + if (lnf) { + *lnf = {}; + } + + widget::Theme::Init(); + return sInstance; +} + +// static +void nsXPLookAndFeel::Shutdown() { + if (sShutdown) { + return; + } + + sShutdown = true; + delete sInstance; + sInstance = nullptr; + + // This keeps strings alive, so need to clear to make leak checking happy. + sFontCache.Clear(); + + widget::Theme::Shutdown(); +} + +static void IntPrefChanged(const nsACString& aPref) { + // Most Int prefs can't change our system colors or fonts, but + // ui.systemUsesDarkTheme can, since it affects the effective color-scheme + // (affecting system colors). + auto changeKind = aPref.EqualsLiteral("ui.systemUsesDarkTheme") + ? widget::ThemeChangeKind::Style + : widget::ThemeChangeKind::MediaQueriesOnly; + LookAndFeel::NotifyChangedAllWindows(changeKind); +} + +static void FloatPrefChanged(const nsACString& aPref) { + // Most float prefs can't change our system colors or fonts, but + // textScaleFactor affects layout. + auto changeKind = aPref.EqualsLiteral("ui.textScaleFactor") + ? widget::ThemeChangeKind::StyleAndLayout + : widget::ThemeChangeKind::MediaQueriesOnly; + LookAndFeel::NotifyChangedAllWindows(changeKind); +} + +static void ColorPrefChanged() { + // Color prefs affect style, because they by definition change system colors. + LookAndFeel::NotifyChangedAllWindows(widget::ThemeChangeKind::Style); +} + +static void FontPrefChanged() { + // Color prefs affect style, because they by definition change system fonts. + LookAndFeel::NotifyChangedAllWindows(widget::ThemeChangeKind::Style); +} + +// static +void nsXPLookAndFeel::OnPrefChanged(const char* aPref, void* aClosure) { + nsDependentCString prefName(aPref); + for (const char* pref : sIntPrefs) { + if (prefName.Equals(pref)) { + IntPrefChanged(prefName); + return; + } + } + + for (const char* pref : sFloatPrefs) { + if (prefName.Equals(pref)) { + FloatPrefChanged(prefName); + return; + } + } + + for (const char* pref : sColorPrefs) { + // We use StringBeginsWith to handle .dark prefs too. + if (StringBeginsWith(prefName, nsDependentCString(pref))) { + ColorPrefChanged(); + return; + } + } + + for (const char* pref : sFontPrefs) { + if (StringBeginsWith(prefName, nsDependentCString(pref))) { + FontPrefChanged(); + return; + } + } +} + +bool LookAndFeel::WindowsNonNativeMenusEnabled() { + switch (StaticPrefs::browser_display_windows_non_native_menus()) { + case 0: + return false; + case 1: + return true; + default: +#ifdef XP_WIN + return IsWin10OrLater(); +#else + return false; +#endif + } +} + +static constexpr struct { + nsLiteralCString mName; + widget::ThemeChangeKind mChangeKind = + widget::ThemeChangeKind::MediaQueriesOnly; +} kMediaQueryPrefs[] = { + {"browser.display.windows.non_native_menus"_ns}, + // Affects whether standins are used for the accent color. + {"widget.non-native-theme.use-theme-accent"_ns, + widget::ThemeChangeKind::Style}, + // These two affect system colors on Windows. + {"widget.windows.uwp-system-colors.enabled"_ns, + widget::ThemeChangeKind::Style}, + // These two affect system colors on Windows. + {"widget.windows.uwp-system-colors.highlight-accent"_ns, + widget::ThemeChangeKind::Style}, + // Affects env(). + {"layout.css.prefers-color-scheme.content-override"_ns, + widget::ThemeChangeKind::Style}, + // Affects media queries and scrollbar sizes, so gotta relayout. + {"widget.gtk.overlay-scrollbars.enabled"_ns, + widget::ThemeChangeKind::StyleAndLayout}, + // Affects zoom settings which includes text and full zoom. + {"browser.display.os-zoom-behavior"_ns, + widget::ThemeChangeKind::StyleAndLayout}, + // This affects not only the media query, but also the native theme, so we + // need to re-layout. + {"browser.theme.toolbar-theme"_ns, widget::ThemeChangeKind::AllBits}, + {"browser.theme.content-theme"_ns}, + {"mathml.legacy_maction_and_semantics_implementations.disabled"_ns}, + {"mathml.ms_lquote_rquote_attributes.disabled"_ns}, + {"dom.element.popover.enabled"_ns}, +}; + +// Read values from the user's preferences. +// This is done once at startup, but since the user's preferences +// haven't actually been read yet at that time, we also have to +// set a callback to inform us of changes to each pref. +void nsXPLookAndFeel::Init() { + MOZ_RELEASE_ASSERT(NS_IsMainThread()); + + // Say we're already initialized, and take the chance that it might fail; + // protects against some other process writing to our static variables. + sInitialized = true; + + RecomputeColorSchemes(); + + if (XRE_IsParentProcess()) { + nsLayoutUtils::RecomputeSmoothScrollDefault(); + } + + // XXX If we could reorganize the pref names, we should separate the branch + // for each types. Then, we could reduce the unnecessary loop from + // nsXPLookAndFeel::OnPrefChanged(). + Preferences::RegisterPrefixCallback(OnPrefChanged, "ui."); + // We really do just want the accessibility.tabfocus pref, not other prefs + // that start with that string. + Preferences::RegisterCallback(OnPrefChanged, "accessibility.tabfocus"); + + for (auto& pref : kMediaQueryPrefs) { + Preferences::RegisterCallback( + [](const char*, void* aChangeKind) { + auto changeKind = + widget::ThemeChangeKind(reinterpret_cast<uintptr_t>(aChangeKind)); + LookAndFeel::NotifyChangedAllWindows(changeKind); + }, + pref.mName, reinterpret_cast<void*>(uintptr_t(pref.mChangeKind))); + } +} + +nsXPLookAndFeel::~nsXPLookAndFeel() { + NS_ASSERTION(sInstance == this, + "This destroying instance isn't the singleton instance"); + sInstance = nullptr; +} + +static bool IsSpecialColor(LookAndFeel::ColorID aID, nscolor aColor) { + using ColorID = LookAndFeel::ColorID; + + if (aColor == NS_SAME_AS_FOREGROUND_COLOR) { + return true; + } + + switch (aID) { + case ColorID::IMESelectedRawTextBackground: + case ColorID::IMESelectedConvertedTextBackground: + case ColorID::IMERawInputBackground: + case ColorID::IMEConvertedTextBackground: + case ColorID::IMESelectedRawTextForeground: + case ColorID::IMESelectedConvertedTextForeground: + case ColorID::IMERawInputForeground: + case ColorID::IMEConvertedTextForeground: + case ColorID::IMERawInputUnderline: + case ColorID::IMEConvertedTextUnderline: + case ColorID::IMESelectedRawTextUnderline: + case ColorID::IMESelectedConvertedTextUnderline: + case ColorID::SpellCheckerUnderline: + return NS_IS_SELECTION_SPECIAL_COLOR(aColor); + default: + break; + } + /* + * In GetColor(), every color that is not a special color is color + * corrected. Use false to make other colors color corrected. + */ + return false; +} + +nscolor nsXPLookAndFeel::GetStandinForNativeColor(ColorID aID, + ColorScheme aScheme) { + if (aScheme == ColorScheme::Dark) { + if (auto color = GenericDarkColor(aID)) { + return *color; + } + } + + // The stand-in colors are taken from what the non-native theme needs (for + // field/button colors), the Windows 7 Aero theme except Mac-specific colors + // which are taken from Mac OS 10.7. + +#define COLOR(name_, r, g, b) \ + case ColorID::name_: \ + return NS_RGB(r, g, b); + +#define COLORA(name_, r, g, b, a) \ + case ColorID::name_: \ + return NS_RGBA(r, g, b, a); + + switch (aID) { + // These are here for the purposes of headless mode. + case ColorID::IMESelectedRawTextBackground: + case ColorID::IMESelectedConvertedTextBackground: + case ColorID::IMERawInputBackground: + case ColorID::IMEConvertedTextBackground: + return NS_TRANSPARENT; + case ColorID::IMESelectedRawTextForeground: + case ColorID::IMESelectedConvertedTextForeground: + case ColorID::IMERawInputForeground: + case ColorID::IMEConvertedTextForeground: + return NS_SAME_AS_FOREGROUND_COLOR; + case ColorID::IMERawInputUnderline: + case ColorID::IMEConvertedTextUnderline: + return NS_40PERCENT_FOREGROUND_COLOR; + case ColorID::Accentcolor: + return widget::sDefaultAccent.ToABGR(); + case ColorID::Accentcolortext: + return widget::sDefaultAccentText.ToABGR(); + COLOR(SpellCheckerUnderline, 0xff, 0x00, 0x00) + COLOR(TextSelectDisabledBackground, 0xaa, 0xaa, 0xaa) + + // CSS 2 colors: + COLOR(Activeborder, 0xB4, 0xB4, 0xB4) + COLOR(Activecaption, 0x99, 0xB4, 0xD1) + COLOR(Appworkspace, 0xAB, 0xAB, 0xAB) + COLOR(Background, 0x00, 0x00, 0x00) + COLOR(Buttonhighlight, 0xFF, 0xFF, 0xFF) + COLOR(Buttonshadow, 0xA0, 0xA0, 0xA0) + + // Buttons and comboboxes should be kept in sync since they are drawn with + // the same colors by the non-native theme. + COLOR(Buttonface, 0xe9, 0xe9, 0xed) + COLORA(MozButtondisabledface, 0xe9, 0xe9, 0xed, 128) + + COLOR(MozCombobox, 0xe9, 0xe9, 0xed) + + COLOR(Buttontext, 0x00, 0x00, 0x00) + COLOR(MozComboboxtext, 0x00, 0x00, 0x00) + + COLOR(Captiontext, 0x00, 0x00, 0x00) + COLOR(Graytext, 0x6D, 0x6D, 0x6D) + COLOR(Highlight, 0x33, 0x99, 0xFF) + COLOR(Highlighttext, 0xFF, 0xFF, 0xFF) + COLOR(Inactiveborder, 0xF4, 0xF7, 0xFC) + COLOR(Inactivecaption, 0xBF, 0xCD, 0xDB) + COLOR(Inactivecaptiontext, 0x43, 0x4E, 0x54) + COLOR(Infobackground, 0xFF, 0xFF, 0xE1) + COLOR(Infotext, 0x00, 0x00, 0x00) + COLOR(Menu, 0xF0, 0xF0, 0xF0) + COLOR(Menutext, 0x00, 0x00, 0x00) + COLOR(Scrollbar, 0xC8, 0xC8, 0xC8) + COLOR(Threeddarkshadow, 0x69, 0x69, 0x69) + COLOR(Threedface, 0xF0, 0xF0, 0xF0) + COLOR(Threedhighlight, 0xFF, 0xFF, 0xFF) + COLOR(Threedlightshadow, 0xE3, 0xE3, 0xE3) + COLOR(Threedshadow, 0xA0, 0xA0, 0xA0) + COLOR(Buttonborder, 0xE3, 0xE3, 0xE3) + COLOR(Mark, 0xFF, 0xFF, 0x00) + COLOR(Marktext, 0x00, 0x00, 0x00) + COLOR(Window, 0xFF, 0xFF, 0xFF) + COLOR(Windowframe, 0x64, 0x64, 0x64) + COLOR(Windowtext, 0x00, 0x00, 0x00) + COLOR(MozButtondefault, 0x69, 0x69, 0x69) + COLOR(Field, 0xFF, 0xFF, 0xFF) + COLORA(MozDisabledfield, 0xFF, 0xFF, 0xFF, 128) + COLOR(Fieldtext, 0x00, 0x00, 0x00) + COLOR(MozDialog, 0xF0, 0xF0, 0xF0) + COLOR(MozDialogtext, 0x00, 0x00, 0x00) + COLOR(MozColheadertext, 0x00, 0x00, 0x00) + COLOR(MozColheaderhovertext, 0x00, 0x00, 0x00) + COLOR(MozDragtargetzone, 0xFF, 0xFF, 0xFF) + COLOR(MozCellhighlight, 0xF0, 0xF0, 0xF0) + COLOR(MozCellhighlighttext, 0x00, 0x00, 0x00) + COLOR(Selecteditem, 0x33, 0x99, 0xFF) + COLOR(Selecteditemtext, 0xFF, 0xFF, 0xFF) + COLOR(MozButtonhoverface, 0xd0, 0xd0, 0xd7) + COLOR(MozButtonhovertext, 0x00, 0x00, 0x00) + COLOR(MozButtonactiveface, 0xb1, 0xb1, 0xb9) + COLOR(MozButtonactivetext, 0x00, 0x00, 0x00) + COLOR(MozMenuhover, 0x33, 0x99, 0xFF) + COLOR(MozMenuhovertext, 0x00, 0x00, 0x00) + COLOR(MozMenubartext, 0x00, 0x00, 0x00) + COLOR(MozMenubarhovertext, 0x00, 0x00, 0x00) + COLOR(MozMenuhoverdisabled, 0xF0, 0xF0, 0xF0) + COLOR(MozEventreerow, 0xFF, 0xFF, 0xFF) + COLOR(MozOddtreerow, 0xFF, 0xFF, 0xFF) + COLOR(MozMacChromeActive, 0xB2, 0xB2, 0xB2) + COLOR(MozMacChromeInactive, 0xE1, 0xE1, 0xE1) + COLOR(MozMacFocusring, 0x60, 0x9D, 0xD7) + COLOR(MozMacMenuselect, 0x38, 0x75, 0xD7) + COLOR(MozMacMenushadow, 0xA3, 0xA3, 0xA3) + COLOR(MozMacMenutextdisable, 0x88, 0x88, 0x88) + COLOR(MozMacMenutextselect, 0xFF, 0xFF, 0xFF) + COLOR(MozMacDisabledtoolbartext, 0x3F, 0x3F, 0x3F) + COLOR(MozMacSecondaryhighlight, 0xD4, 0xD4, 0xD4) + COLOR(MozMacMenupopup, 0xe6, 0xe6, 0xe6) + COLOR(MozMacMenuitem, 0xe6, 0xe6, 0xe6) + COLOR(MozMacActiveMenuitem, 0x0a, 0x64, 0xdc) + COLOR(MozMacSourceList, 0xf7, 0xf7, 0xf7) + COLOR(MozMacSourceListSelection, 0xc8, 0xc8, 0xc8) + COLOR(MozMacActiveSourceListSelection, 0x0a, 0x64, 0xdc) + COLOR(MozMacTooltip, 0xf7, 0xf7, 0xf7) + // Seems to be the default color (hardcoded because of bug 1065998) + COLOR(MozWinMediatext, 0xFF, 0xFF, 0xFF) + COLOR(MozWinCommunicationstext, 0xFF, 0xFF, 0xFF) + COLOR(MozNativehyperlinktext, 0x00, 0x66, 0xCC) + COLOR(MozNativevisitedhyperlinktext, 0x55, 0x1A, 0x8B) + default: + break; + } + return NS_RGB(0xFF, 0xFF, 0xFF); +} + +#undef COLOR +#undef COLORA + +// Taken from in-content/common.inc.css's dark theme. +Maybe<nscolor> nsXPLookAndFeel::GenericDarkColor(ColorID aID) { + nscolor color = NS_RGB(0, 0, 0); + static constexpr nscolor kWindowBackground = NS_RGB(28, 27, 34); + static constexpr nscolor kWindowText = NS_RGB(251, 251, 254); + switch (aID) { + case ColorID::Window: // --in-content-page-background + case ColorID::Background: + color = kWindowBackground; + break; + + case ColorID::Menu: + color = NS_RGB(0x2b, 0x2a, 0x33); + break; + + case ColorID::MozMenuhovertext: + case ColorID::MozMenubarhovertext: + case ColorID::Menutext: + color = NS_RGB(0xfb, 0xfb, 0xfe); + break; + + case ColorID::MozMenuhover: + color = NS_RGB(0x52, 0x52, 0x5e); + break; + + case ColorID::MozMenuhoverdisabled: + color = NS_RGB(0x3a, 0x39, 0x44); + break; + + case ColorID::MozOddtreerow: + case ColorID::MozDialog: // --in-content-box-background + color = NS_RGB(35, 34, 43); + break; + case ColorID::Windowtext: // --in-content-page-color + case ColorID::MozDialogtext: + case ColorID::Fieldtext: + case ColorID::Buttontext: // --in-content-button-text-color (via + // --in-content-page-color) + case ColorID::MozComboboxtext: + case ColorID::MozButtonhovertext: + case ColorID::MozButtonactivetext: + color = kWindowText; + break; + case ColorID::Buttonshadow: + case ColorID::Threedshadow: + case ColorID::Threedlightshadow: + case ColorID::Buttonborder: // --in-content-box-border-color computed + // with kWindowText above + // kWindowBackground. + case ColorID::Graytext: // opacity: 0.4 of kWindowText blended over the + // "Window" background color, which happens to be + // the same :-) + color = NS_ComposeColors(kWindowBackground, NS_RGBA(251, 251, 254, 102)); + break; + case ColorID::MozCellhighlight: + case ColorID::Selecteditem: // --in-content-primary-button-background / + // --in-content-item-selected + color = NS_RGB(0, 221, 255); + break; + case ColorID::Field: + case ColorID::Buttonface: // --in-content-button-background + case ColorID::Threedface: + case ColorID::MozCombobox: + case ColorID::MozCellhighlighttext: + case ColorID::Selecteditemtext: // --in-content-primary-button-text-color / + // --in-content-item-selected-text + color = NS_RGB(43, 42, 51); + break; + case ColorID::Threeddarkshadow: // Same as Threedlightshadow but with the + // background. + case ColorID::MozDisabledfield: // opacity: 0.4 of the face above blended + // over the "Window" background color. + case ColorID::MozButtondisabledface: + color = NS_ComposeColors(kWindowBackground, NS_RGBA(43, 42, 51, 102)); + break; + case ColorID::MozButtonhoverface: // --in-content-button-background-hover + color = NS_RGB(82, 82, 94); + break; + case ColorID::MozButtonactiveface: // --in-content-button-background-active + color = NS_RGB(91, 91, 102); + break; + case ColorID::Highlight: + color = NS_RGBA(0, 221, 255, 78); + break; + case ColorID::Highlighttext: + color = NS_SAME_AS_FOREGROUND_COLOR; + break; + case ColorID::MozNativehyperlinktext: + // If you change this color, you probably also want to change the default + // value of browser.anchor_color.dark. + color = NS_RGB(0x8c, 0x8c, 0xff); + break; + case ColorID::MozNativevisitedhyperlinktext: + // If you change this color, you probably also want to change the default + // value of browser.visited_color.dark. + color = NS_RGB(0xff, 0xad, 0xff); + break; + case ColorID::SpellCheckerUnderline: + // This is the default for active links in dark mode as well + // (browser.active_color.dark). See bug 1755564 for some analysis and + // other options too. + color = NS_RGB(0xff, 0x66, 0x66); + break; + default: + return Nothing(); + } + return Some(color); +} + +// Uncomment the #define below if you want to debug system color use in a skin +// that uses them. When set, it will make all system color pairs that are +// appropriate for foreground/background pairing the same. This means if the +// skin is using system colors correctly you will not be able to see *any* text. +// +// #define DEBUG_SYSTEM_COLOR_USE + +#ifdef DEBUG_SYSTEM_COLOR_USE +static nsresult SystemColorUseDebuggingColor(LookAndFeel::ColorID aID, + nscolor& aResult) { + using ColorID = LookAndFeel::ColorID; + + switch (aID) { + // css2 http://www.w3.org/TR/REC-CSS2/ui.html#system-colors + case ColorID::Activecaption: + // active window caption background + case ColorID::Captiontext: + // text in active window caption + aResult = NS_RGB(0xff, 0x00, 0x00); + break; + + case ColorID::Highlight: + // background of selected item + case ColorID::Highlighttext: + // text of selected item + aResult = NS_RGB(0xff, 0xff, 0x00); + break; + + case ColorID::Inactivecaption: + // inactive window caption + case ColorID::Inactivecaptiontext: + // text in inactive window caption + aResult = NS_RGB(0x66, 0x66, 0x00); + break; + + case ColorID::Infobackground: + // tooltip background color + case ColorID::Infotext: + // tooltip text color + aResult = NS_RGB(0x00, 0xff, 0x00); + break; + + case ColorID::Menu: + // menu background + case ColorID::Menutext: + // menu text + aResult = NS_RGB(0x00, 0xff, 0xff); + break; + + case ColorID::Threedface: + case ColorID::Buttonface: + // 3-D face color + case ColorID::Buttontext: + // text on push buttons + aResult = NS_RGB(0x00, 0x66, 0x66); + break; + + case ColorID::Window: + case ColorID::Windowtext: + aResult = NS_RGB(0x00, 0x00, 0xff); + break; + + // from the CSS3 working draft (not yet finalized) + // http://www.w3.org/tr/2000/wd-css3-userint-20000216.html#color + + case ColorID::Field: + case ColorID::Fieldtext: + aResult = NS_RGB(0xff, 0x00, 0xff); + break; + + case ColorID::MozDialog: + case ColorID::MozDialogtext: + aResult = NS_RGB(0x66, 0x00, 0x66); + break; + + default: + return NS_ERROR_NOT_AVAILABLE; + } + + return NS_OK; +} +#endif + +static nsresult GetPrefColor(const char* aPref, nscolor& aResult) { + nsAutoCString colorStr; + MOZ_TRY(Preferences::GetCString(aPref, colorStr)); + if (!ServoCSSParser::ComputeColor(nullptr, NS_RGB(0, 0, 0), colorStr, + &aResult)) { + return NS_ERROR_FAILURE; + } + return NS_OK; +} + +static nsresult GetColorFromPref(LookAndFeel::ColorID aID, ColorScheme aScheme, + nscolor& aResult) { + const char* prefName = sColorPrefs[size_t(aID)]; + if (aScheme == ColorScheme::Dark) { + nsAutoCString darkPrefName(prefName); + darkPrefName.Append(".dark"); + if (NS_SUCCEEDED(GetPrefColor(darkPrefName.get(), aResult))) { + return NS_OK; + } + } + return GetPrefColor(prefName, aResult); +} + +// All these routines will return NS_OK if they have a value, +// in which case the nsLookAndFeel should use that value; +// otherwise we'll return NS_ERROR_NOT_AVAILABLE, in which case, the +// platform-specific nsLookAndFeel should use its own values instead. +nsresult nsXPLookAndFeel::GetColorValue(ColorID aID, ColorScheme aScheme, + UseStandins aUseStandins, + nscolor& aResult) { + if (!sInitialized) { + Init(); + } + +#ifdef DEBUG_SYSTEM_COLOR_USE + if (NS_SUCCEEDED(SystemColorUseDebuggingColor(aID, aResult))) { + return NS_OK; + } +#endif + + auto& cache = sColorCaches.Get(aScheme, aUseStandins); + if (const auto* cached = cache.Get(aID)) { + if (cached->isNothing()) { + return NS_ERROR_FAILURE; + } + aResult = cached->value(); + return NS_OK; + } + + // NOTE: Servo holds a lock and the main thread is paused, so writing to the + // global cache here is fine. + auto result = GetUncachedColor(aID, aScheme, aUseStandins); + cache.Insert(aID, result); + if (!result) { + return NS_ERROR_FAILURE; + } + aResult = *result; + return NS_OK; +} + +Maybe<nscolor> nsXPLookAndFeel::GetUncachedColor(ColorID aID, + ColorScheme aScheme, + UseStandins aUseStandins) { + if (aUseStandins == UseStandins::Yes) { + return Some(GetStandinForNativeColor(aID, aScheme)); + } + nscolor r; + if (NS_SUCCEEDED(GetColorFromPref(aID, aScheme, r))) { + return Some(r); + } + if (NS_SUCCEEDED(NativeGetColor(aID, aScheme, r))) { + if (gfxPlatform::GetCMSMode() == CMSMode::All && !IsSpecialColor(aID, r)) { + qcms_transform* transform = gfxPlatform::GetCMSInverseRGBTransform(); + if (transform) { + uint8_t color[4]; + color[0] = NS_GET_R(r); + color[1] = NS_GET_G(r); + color[2] = NS_GET_B(r); + color[3] = NS_GET_A(r); + qcms_transform_data(transform, color, color, 1); + r = NS_RGBA(color[0], color[1], color[2], color[3]); + } + } + + return Some(r); + } + return Nothing(); +} + +nsresult nsXPLookAndFeel::GetIntValue(IntID aID, int32_t& aResult) { + if (!sInitialized) { + Init(); + } + + if (const auto* cached = sIntCache.Get(aID)) { + if (cached->isNothing()) { + return NS_ERROR_FAILURE; + } + aResult = cached->value(); + return NS_OK; + } + + if (NS_SUCCEEDED(Preferences::GetInt(sIntPrefs[size_t(aID)], &aResult))) { + sIntCache.Insert(aID, Some(aResult)); + return NS_OK; + } + + if (NS_FAILED(NativeGetInt(aID, aResult))) { + sIntCache.Insert(aID, Nothing()); + return NS_ERROR_FAILURE; + } + + sIntCache.Insert(aID, Some(aResult)); + return NS_OK; +} + +nsresult nsXPLookAndFeel::GetFloatValue(FloatID aID, float& aResult) { + if (!sInitialized) { + Init(); + } + + if (const auto* cached = sFloatCache.Get(aID)) { + if (cached->isNothing()) { + return NS_ERROR_FAILURE; + } + aResult = cached->value(); + return NS_OK; + } + + int32_t pref = 0; + if (NS_SUCCEEDED(Preferences::GetInt(sFloatPrefs[size_t(aID)], &pref))) { + aResult = float(pref) / 100.0f; + sFloatCache.Insert(aID, Some(aResult)); + return NS_OK; + } + + if (NS_FAILED(NativeGetFloat(aID, aResult))) { + sFloatCache.Insert(aID, Nothing()); + return NS_ERROR_FAILURE; + } + + sFloatCache.Insert(aID, Some(aResult)); + return NS_OK; +} + +bool nsXPLookAndFeel::LookAndFeelFontToStyle(const LookAndFeelFont& aFont, + nsString& aName, + gfxFontStyle& aStyle) { + if (!aFont.haveFont()) { + return false; + } + aName = aFont.name(); + aStyle = gfxFontStyle(); + aStyle.size = aFont.size(); + aStyle.weight = FontWeight::FromInt(aFont.weight()); + aStyle.style = + aFont.italic() ? FontSlantStyle::ITALIC : FontSlantStyle::NORMAL; + aStyle.systemFont = true; + return true; +} + +widget::LookAndFeelFont nsXPLookAndFeel::StyleToLookAndFeelFont( + const nsAString& aName, const gfxFontStyle& aStyle) { + LookAndFeelFont font; + font.haveFont() = true; + font.name() = aName; + font.size() = aStyle.size; + font.weight() = aStyle.weight.ToFloat(); + font.italic() = aStyle.style.IsItalic(); + MOZ_ASSERT(aStyle.style.IsNormal() || aStyle.style.IsItalic(), + "Cannot handle oblique font style"); +#ifdef DEBUG + { + // Assert that all the remaining font style properties have their + // default values. + gfxFontStyle candidate = aStyle; + gfxFontStyle defaults{}; + candidate.size = defaults.size; + candidate.weight = defaults.weight; + candidate.style = defaults.style; + MOZ_ASSERT(candidate.Equals(defaults), + "Some font style properties not supported"); + } +#endif + return font; +} + +bool nsXPLookAndFeel::GetFontValue(FontID aID, nsString& aName, + gfxFontStyle& aStyle) { + if (const LookAndFeelFont* cached = sFontCache.Get(aID)) { + return LookAndFeelFontToStyle(*cached, aName, aStyle); + } + + LookAndFeelFont font; + auto GetFontsFromPrefs = [&]() -> bool { + nsDependentCString pref(sFontPrefs[size_t(aID)]); + if (NS_FAILED(Preferences::GetString(pref.get(), aName))) { + return false; + } + font.haveFont() = true; + font.name() = aName; + font.size() = Preferences::GetFloat(nsAutoCString(pref + ".size"_ns).get()); + // This is written this way rather than using the fallback so that an empty + // pref (such like the one about:config creates) doesn't cause system fonts + // to have zero-size. + if (font.size() < 1.0f) { + font.size() = StyleFONT_MEDIUM_PX; + } + font.weight() = Preferences::GetFloat( + nsAutoCString(pref + ".weight"_ns).get(), FontWeight::NORMAL.ToFloat()); + font.italic() = + Preferences::GetBool(nsAutoCString(pref + ".italic"_ns).get()); + return true; + }; + + if (GetFontsFromPrefs()) { + LookAndFeelFontToStyle(font, aName, aStyle); + } else if (NativeGetFont(aID, aName, aStyle)) { + font = StyleToLookAndFeelFont(aName, aStyle); + } else { + MOZ_ASSERT(!font.haveFont()); + } + bool success = font.haveFont(); + sFontCache.Insert(aID, std::move(font)); + return success; +} + +void nsXPLookAndFeel::RefreshImpl() { + // Wipe out our caches. + sColorCaches.Clear(); + sFontCache.Clear(); + sFloatCache.Clear(); + sIntCache.Clear(); + RecomputeColorSchemes(); + + if (XRE_IsParentProcess()) { + nsLayoutUtils::RecomputeSmoothScrollDefault(); + // Clear any cached FullLookAndFeel data, which is now invalid. + widget::RemoteLookAndFeel::ClearCachedData(); + } +} + +static bool sRecordedLookAndFeelTelemetry = false; + +void nsXPLookAndFeel::RecordTelemetry() { + if (!XRE_IsParentProcess()) { + return; + } + + if (sRecordedLookAndFeelTelemetry) { + return; + } + + sRecordedLookAndFeelTelemetry = true; + + int32_t i; + Telemetry::ScalarSet( + Telemetry::ScalarID::WIDGET_DARK_MODE, + NS_SUCCEEDED(GetIntValue(IntID::SystemUsesDarkTheme, i)) && i != 0); + + RecordLookAndFeelSpecificTelemetry(); +} + +namespace mozilla { + +static widget::ThemeChangeKind sGlobalThemeChangeKind{0}; + +void LookAndFeel::NotifyChangedAllWindows(widget::ThemeChangeKind aKind) { + sGlobalThemeChanged = true; + sGlobalThemeChangeKind |= aKind; + + if (nsCOMPtr<nsIObserverService> obs = services::GetObserverService()) { + const char16_t kind[] = {char16_t(aKind), 0}; + obs->NotifyObservers(nullptr, "internal-look-and-feel-changed", kind); + } +} + +void LookAndFeel::DoHandleGlobalThemeChange() { + MOZ_ASSERT(sGlobalThemeChanged); + sGlobalThemeChanged = false; + auto kind = std::exchange(sGlobalThemeChangeKind, widget::ThemeChangeKind(0)); + + // Tell the theme that it changed, so it can flush any handles to stale theme + // data. + // + // We can use the *DoNotUseDirectly functions directly here, because we want + // to notify all possible themes in a given process (but just once). + if (XRE_IsParentProcess() || + !StaticPrefs::widget_non_native_theme_enabled()) { + if (nsCOMPtr<nsITheme> theme = do_GetNativeThemeDoNotUseDirectly()) { + theme->ThemeChanged(); + } + } + if (nsCOMPtr<nsITheme> theme = do_GetBasicNativeThemeDoNotUseDirectly()) { + theme->ThemeChanged(); + } + + // Clear all cached LookAndFeel colors. + LookAndFeel::Refresh(); + + // Reset default background and foreground colors for the document since they + // may be using system colors. + PreferenceSheet::Refresh(); + + // Vector images (SVG) may be using theme colors so we discard all cached + // surfaces. (We could add a vector image only version of DiscardAll, but + // in bug 940625 we decided theme changes are rare enough not to bother.) + image::SurfaceCacheUtils::DiscardAll(); + + if (XRE_IsParentProcess()) { + dom::ContentParent::BroadcastThemeUpdate(kind); + } + + nsContentUtils::AddScriptRunner( + NS_NewRunnableFunction("HandleGlobalThemeChange", [] { + if (nsCOMPtr<nsIObserverService> obs = services::GetObserverService()) { + obs->NotifyObservers(nullptr, "look-and-feel-changed", nullptr); + } + })); +} + +#define BIT_FOR(_c) (1ull << size_t(ColorID::_c)) + +// We want to use a non-native color scheme for the non-native theme (except in +// high-contrast mode), so spoof some of the colors with stand-ins to prevent +// lack of contrast. +static constexpr std::bitset<size_t(ColorID::End)> sNonNativeThemeStandinColors{ + // Used by default button styles. + BIT_FOR(Buttonface) | BIT_FOR(Buttontext) | BIT_FOR(MozButtonhoverface) | + BIT_FOR(MozButtonhovertext) | BIT_FOR(MozButtonactiveface) | + BIT_FOR(MozButtonactivetext) | BIT_FOR(MozButtondisabledface) | + BIT_FOR(Buttonborder) | + // Used by select elements. + BIT_FOR(MozCombobox) | BIT_FOR(MozComboboxtext) | + BIT_FOR(Threedlightshadow) | + // For symmetry with the above. + BIT_FOR(Threeddarkshadow) | + // Used by fieldset borders. + BIT_FOR(Threedface) | + // Used by input / textarea. + BIT_FOR(Field) | BIT_FOR(Fieldtext) | + // Used by disabled form controls. + BIT_FOR(MozDisabledfield) | BIT_FOR(Graytext) | + // Some pages expect these to return windows-like colors, see bug 1773795. + // Also, per spec these should match Canvas/CanvasText, see + // https://drafts.csswg.org/css-color-4/#window + BIT_FOR(Window) | BIT_FOR(Windowtext)}; +#undef BIT_FOR + +static bool ShouldUseStandinsForNativeColorForNonNativeTheme( + const dom::Document& aDoc, LookAndFeel::ColorID aColor, + const PreferenceSheet::Prefs& aPrefs) { + const bool shouldUseStandinsForColor = [&] { + if (sNonNativeThemeStandinColors[size_t(aColor)]) { + return true; + } + // There are platforms where we want the content-exposed accent color to be + // the windows blue rather than the system accent color, for now. + return !StaticPrefs::widget_non_native_theme_use_theme_accent() && + (aColor == LookAndFeel::ColorID::Accentcolor || + aColor == LookAndFeel::ColorID::Accentcolortext); + }(); + + return shouldUseStandinsForColor && aDoc.ShouldAvoidNativeTheme() && + !aPrefs.NonNativeThemeShouldBeHighContrast(); +} + +ColorScheme LookAndFeel::sChromeColorScheme; +ColorScheme LookAndFeel::sContentColorScheme; +bool LookAndFeel::sColorSchemeInitialized; +bool LookAndFeel::sGlobalThemeChanged; + +bool LookAndFeel::IsDarkColor(nscolor aColor) { + // Given https://www.w3.org/TR/WCAG20/#contrast-ratiodef, this is the + // threshold that tells us whether contrast is better against white or black. + // + // Contrast ratio against black is: (L + 0.05) / 0.05 + // Contrast ratio against white is: 1.05 / (L + 0.05) + // + // So the intersection is: + // + // (L + 0.05) / 0.05 = 1.05 / (L + 0.05) + // + // And the solution to that equation is: + // + // sqrt(1.05 * 0.05) - 0.05 + // + // So we consider a color dark if the contrast is below this threshold, and + // it's at least half-opaque. + constexpr float kThreshold = 0.179129; + return NS_GET_A(aColor) > 127 && + RelativeLuminanceUtils::Compute(aColor) < kThreshold; +} + +auto LookAndFeel::ColorSchemeSettingForChrome() -> ChromeColorSchemeSetting { + switch (StaticPrefs::browser_theme_toolbar_theme()) { + case 0: // Dark + return ChromeColorSchemeSetting::Dark; + case 1: // Light + return ChromeColorSchemeSetting::Light; + default: + return ChromeColorSchemeSetting::System; + } +} + +ColorScheme LookAndFeel::ThemeDerivedColorSchemeForContent() { + switch (StaticPrefs::browser_theme_content_theme()) { + case 0: // Dark + return ColorScheme::Dark; + case 1: // Light + return ColorScheme::Light; + default: + return SystemColorScheme(); + } +} + +void LookAndFeel::RecomputeColorSchemes() { + sColorSchemeInitialized = true; + + sChromeColorScheme = [] { + switch (ColorSchemeSettingForChrome()) { + case ChromeColorSchemeSetting::Light: + return ColorScheme::Light; + case ChromeColorSchemeSetting::Dark: + return ColorScheme::Dark; + case ChromeColorSchemeSetting::System: + break; + } + return SystemColorScheme(); + }(); + + sContentColorScheme = [] { + switch (StaticPrefs::layout_css_prefers_color_scheme_content_override()) { + case 0: + return ColorScheme::Dark; + case 1: + return ColorScheme::Light; + default: + return ThemeDerivedColorSchemeForContent(); + } + }(); +} + +ColorScheme LookAndFeel::ColorSchemeForStyle( + const dom::Document& aDoc, const StyleColorSchemeFlags& aFlags, + ColorSchemeMode aMode) { + using Choice = PreferenceSheet::Prefs::ColorSchemeChoice; + + const auto& prefs = PreferenceSheet::PrefsFor(aDoc); + switch (prefs.mColorSchemeChoice) { + case Choice::Standard: + break; + case Choice::UserPreferred: + return aDoc.PreferredColorScheme(); + case Choice::Light: + return ColorScheme::Light; + case Choice::Dark: + return ColorScheme::Dark; + } + + StyleColorSchemeFlags style(aFlags); + if (!style) { + style.bits = aDoc.GetColorSchemeBits(); + } + const bool supportsDark = bool(style & StyleColorSchemeFlags::DARK); + const bool supportsLight = bool(style & StyleColorSchemeFlags::LIGHT); + if (supportsLight && supportsDark) { + // Both color-schemes are explicitly supported, use the preferred one. + return aDoc.PreferredColorScheme(); + } + if (supportsDark || supportsLight) { + // One color-scheme is explicitly supported and one isn't, so use the one + // the content supports. + return supportsDark ? ColorScheme::Dark : ColorScheme::Light; + } + // No value specified. Chrome docs always supports both, so use the preferred + // color-scheme. + if (aMode == ColorSchemeMode::Preferred || + nsContentUtils::IsChromeDoc(&aDoc)) { + return aDoc.PreferredColorScheme(); + } + // Default content to light. + return ColorScheme::Light; +} + +LookAndFeel::ColorScheme LookAndFeel::ColorSchemeForFrame( + const nsIFrame* aFrame, ColorSchemeMode aMode) { + return ColorSchemeForStyle(*aFrame->PresContext()->Document(), + aFrame->StyleUI()->mColorScheme.bits, aMode); +} + +// static +Maybe<nscolor> LookAndFeel::GetColor(ColorID aId, ColorScheme aScheme, + UseStandins aUseStandins) { + nscolor result; + nsresult rv = nsLookAndFeel::GetInstance()->GetColorValue( + aId, aScheme, aUseStandins, result); + if (NS_FAILED(rv)) { + return Nothing(); + } + return Some(result); +} + +// Returns whether there is a CSS color name for this color. +static bool ColorIsCSSAccessible(LookAndFeel::ColorID aId) { + using ColorID = LookAndFeel::ColorID; + + switch (aId) { + case ColorID::TextSelectDisabledBackground: + case ColorID::TextSelectAttentionBackground: + case ColorID::TextSelectAttentionForeground: + case ColorID::TextHighlightBackground: + case ColorID::TextHighlightForeground: + case ColorID::ThemedScrollbar: + case ColorID::ThemedScrollbarInactive: + case ColorID::ThemedScrollbarThumb: + case ColorID::ThemedScrollbarThumbActive: + case ColorID::ThemedScrollbarThumbInactive: + case ColorID::ThemedScrollbarThumbHover: + case ColorID::IMERawInputBackground: + case ColorID::IMERawInputForeground: + case ColorID::IMERawInputUnderline: + case ColorID::IMESelectedRawTextBackground: + case ColorID::IMESelectedRawTextForeground: + case ColorID::IMESelectedRawTextUnderline: + case ColorID::IMEConvertedTextBackground: + case ColorID::IMEConvertedTextForeground: + case ColorID::IMEConvertedTextUnderline: + case ColorID::IMESelectedConvertedTextBackground: + case ColorID::IMESelectedConvertedTextForeground: + case ColorID::IMESelectedConvertedTextUnderline: + case ColorID::SpellCheckerUnderline: + return false; + default: + break; + } + + return true; +} + +LookAndFeel::UseStandins LookAndFeel::ShouldUseStandins( + const dom::Document& aDoc, ColorID aId) { + const auto& prefs = PreferenceSheet::PrefsFor(aDoc); + if (ShouldUseStandinsForNativeColorForNonNativeTheme(aDoc, aId, prefs)) { + return UseStandins::Yes; + } + if (prefs.mUseStandins && ColorIsCSSAccessible(aId)) { + return UseStandins::Yes; + } + return UseStandins::No; +} + +Maybe<nscolor> LookAndFeel::GetColor(ColorID aId, const nsIFrame* aFrame) { + const auto* doc = aFrame->PresContext()->Document(); + return GetColor(aId, ColorSchemeForFrame(aFrame), + ShouldUseStandins(*doc, aId)); +} + +// static +nsresult LookAndFeel::GetInt(IntID aID, int32_t* aResult) { + return nsLookAndFeel::GetInstance()->GetIntValue(aID, *aResult); +} + +// static +nsresult LookAndFeel::GetFloat(FloatID aID, float* aResult) { + return nsLookAndFeel::GetInstance()->GetFloatValue(aID, *aResult); +} + +// static +bool LookAndFeel::GetFont(FontID aID, nsString& aName, gfxFontStyle& aStyle) { + return nsLookAndFeel::GetInstance()->GetFontValue(aID, aName, aStyle); +} + +// static +char16_t LookAndFeel::GetPasswordCharacter() { + return nsLookAndFeel::GetInstance()->GetPasswordCharacterImpl(); +} + +// static +bool LookAndFeel::GetEchoPassword() { + if (StaticPrefs::editor_password_mask_delay() >= 0) { + return StaticPrefs::editor_password_mask_delay() > 0; + } + return nsLookAndFeel::GetInstance()->GetEchoPasswordImpl(); +} + +// static +uint32_t LookAndFeel::GetPasswordMaskDelay() { + int32_t delay = StaticPrefs::editor_password_mask_delay(); + if (delay < 0) { + return nsLookAndFeel::GetInstance()->GetPasswordMaskDelayImpl(); + } + return delay; +} + +bool LookAndFeel::DrawInTitlebar() { + switch (StaticPrefs::browser_tabs_inTitlebar()) { + case 0: + return false; + case 1: + return true; + default: + break; + } + return nsLookAndFeel::GetInstance()->GetDefaultDrawInTitlebar(); +} + +void LookAndFeel::GetThemeInfo(nsACString& aOut) { + nsLookAndFeel::GetInstance()->GetThemeInfo(aOut); +} + +uint32_t LookAndFeel::GetMenuAccessKey() { + return StaticPrefs::ui_key_menuAccessKey(); +} + +Modifiers LookAndFeel::GetMenuAccessKeyModifiers() { + switch (GetMenuAccessKey()) { + case dom::KeyboardEvent_Binding::DOM_VK_SHIFT: + return MODIFIER_SHIFT; + case dom::KeyboardEvent_Binding::DOM_VK_CONTROL: + return MODIFIER_CONTROL; + case dom::KeyboardEvent_Binding::DOM_VK_ALT: + return MODIFIER_ALT; + case dom::KeyboardEvent_Binding::DOM_VK_META: + return MODIFIER_META; + case dom::KeyboardEvent_Binding::DOM_VK_WIN: + return MODIFIER_OS; + default: + return 0; + } +} + +// static +void LookAndFeel::Refresh() { + nsLookAndFeel::GetInstance()->RefreshImpl(); + widget::Theme::LookAndFeelChanged(); +} + +// static +void LookAndFeel::NativeInit() { nsLookAndFeel::GetInstance()->NativeInit(); } + +// static +void LookAndFeel::SetData(widget::FullLookAndFeel&& aTables) { + nsLookAndFeel::GetInstance()->SetDataImpl(std::move(aTables)); +} + +} // namespace mozilla |