forked from mirrors/gecko-dev
		
	 e7bfa879da
			
		
	
	
		e7bfa879da
		
	
	
	
	
		
			
			This will be needed to determine whether following the Firefox theme would produce light or dark colors. Differential Revision: https://phabricator.services.mozilla.com/D140221
		
			
				
	
	
		
			1395 lines
		
	
	
	
		
			44 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1395 lines
		
	
	
	
		
			44 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- 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 "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 "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_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/Telemetry.h"
 | |
| #include "mozilla/TelemetryScalarEnums.h"
 | |
| 
 | |
| #include "gfxPlatform.h"
 | |
| #include "gfxFont.h"
 | |
| 
 | |
| #include "qcms.h"
 | |
| 
 | |
| #ifdef DEBUG
 | |
| #  include "nsSize.h"
 | |
| #endif
 | |
| 
 | |
| 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 {
 | |
|   static constexpr uint32_t ChunkFor(Index aIndex) {
 | |
|     return uint32_t(aIndex) >> 5;  // >> 5 is the same as / 32.
 | |
|   }
 | |
|   static constexpr uint32_t BitFor(Index aIndex) {
 | |
|     return 1u << (uint32_t(aIndex) & 31);
 | |
|   }
 | |
|   static constexpr uint32_t kChunks = ChunkFor(kEnd) + 1;
 | |
| 
 | |
|   mozilla::EnumeratedArray<Index, kEnd, Value> mEntries;
 | |
|   uint32_t mValidity[kChunks] = {0};
 | |
| 
 | |
|  public:
 | |
|   constexpr EnumeratedCache() = default;
 | |
| 
 | |
|   bool IsValid(Index aIndex) const {
 | |
|     return mValidity[ChunkFor(aIndex)] & BitFor(aIndex);
 | |
|   }
 | |
| 
 | |
|   const Value* Get(Index aIndex) const {
 | |
|     return IsValid(aIndex) ? &mEntries[aIndex] : nullptr;
 | |
|   }
 | |
| 
 | |
|   void Insert(Index aIndex, Value aValue) {
 | |
|     mValidity[ChunkFor(aIndex)] |= BitFor(aIndex);
 | |
|     mEntries[aIndex] = aValue;
 | |
|   }
 | |
| 
 | |
|   void Remove(Index aIndex) {
 | |
|     mValidity[ChunkFor(aIndex)] &= ~BitFor(aIndex);
 | |
|     mEntries[aIndex] = Value();
 | |
|   }
 | |
| 
 | |
|   void Clear() {
 | |
|     for (auto& chunk : mValidity) {
 | |
|       chunk = 0;
 | |
|     }
 | |
|     for (auto& entry : mEntries) {
 | |
|       entry = Value();
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| static EnumeratedCache<ColorID, Maybe<nscolor>, ColorID::End> sLightColorCache;
 | |
| static EnumeratedCache<ColorID, Maybe<nscolor>, ColorID::End> sDarkColorCache;
 | |
| 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[][43] = {
 | |
|     "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.scrollSliderStyle",
 | |
|     "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.primaryPointerCapabilities",
 | |
|     "ui.allPointerCapabilities",
 | |
|     "ui.systemVerticalScrollbarWidth",
 | |
|     "ui.systemHorizontalScrollbarHeight",
 | |
|     "ui.touchDeviceSupportPresent",
 | |
|     "ui.titlebarRadius",
 | |
|     "ui.GtkMenuRadius",
 | |
| };
 | |
| 
 | |
| 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.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",
 | |
|     "ui.activeborder",
 | |
|     "ui.activecaption",
 | |
|     "ui.appworkspace",
 | |
|     "ui.background",
 | |
|     "ui.buttonface",
 | |
|     "ui.buttonhighlight",
 | |
|     "ui.buttonshadow",
 | |
|     "ui.buttontext",
 | |
|     "ui.captiontext",
 | |
|     "ui.-moz-field",
 | |
|     "ui.-moz-disabledfield",
 | |
|     "ui.-moz-fieldtext",
 | |
|     "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_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-vibrant-titlebar-light",
 | |
|     "ui.-moz-mac-vibrant-titlebar-dark",
 | |
|     "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.-moz-accent-color",
 | |
|     "ui.-moz-accent-color-foreground",
 | |
|     "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-comboboxtext",
 | |
|     "ui.-moz-combobox",
 | |
|     "ui.-moz-colheadertext",
 | |
|     "ui.-moz-colheaderhovertext",
 | |
| };
 | |
| 
 | |
| static_assert(ArrayLength(sColorPrefs) == size_t(LookAndFeel::ColorID::End),
 | |
|               "Should have a pref for each color value");
 | |
| 
 | |
| const char* nsXPLookAndFeel::GetColorPrefName(ColorID aId) {
 | |
|   return sColorPrefs[size_t(aId)];
 | |
| }
 | |
| 
 | |
| bool nsXPLookAndFeel::sInitialized = false;
 | |
| 
 | |
| nsXPLookAndFeel* nsXPLookAndFeel::sInstance = nullptr;
 | |
| bool nsXPLookAndFeel::sShutdown = false;
 | |
| 
 | |
| // 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() {
 | |
|   // Float prefs can't change our system colors or fonts.
 | |
|   LookAndFeel::NotifyChangedAllWindows(
 | |
|       widget::ThemeChangeKind::MediaQueriesOnly);
 | |
| }
 | |
| 
 | |
| static void ColorPrefChanged() {
 | |
|   // Color prefs affect style, because they by definition change system colors.
 | |
|   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();
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (const char* pref : sColorPrefs) {
 | |
|     // We use StringBeginsWith to handle .dark prefs too.
 | |
|     if (StringBeginsWith(prefName, nsDependentCString(pref))) {
 | |
|       ColorPrefChanged();
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| static constexpr struct {
 | |
|   nsLiteralCString mName;
 | |
|   widget::ThemeChangeKind mChangeKind =
 | |
|       widget::ThemeChangeKind::MediaQueriesOnly;
 | |
| } kMediaQueryPrefs[] = {
 | |
|     {"browser.display.windows.native_menus"_ns},
 | |
|     {"browser.proton.places-tooltip.enabled"_ns},
 | |
|     {"layout.css.prefers-color-scheme.content-override"_ns},
 | |
|     // Affects media queries and scrollbar sizes, so gotta relayout.
 | |
|     {"widget.gtk.overlay-scrollbars.enabled"_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},
 | |
| };
 | |
| 
 | |
| // 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();
 | |
| 
 | |
|   // 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;
 | |
|       COLOR(MozAccentColor, 53, 132, 228)
 | |
|       COLOR(MozAccentColorForeground, 0xff, 0xff, 0xff)
 | |
|       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(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(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(MozMacVibrantTitlebarLight, 0xf7, 0xf7, 0xf7)
 | |
|       COLOR(MozMacVibrantTitlebarDark, 0x28, 0x28, 0x28)
 | |
|       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:
 | |
|     case ColorID::Menu:
 | |
|       color = kWindowBackground;
 | |
|       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::Menutext:
 | |
|     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::Threedlightshadow:  // --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;
 | |
| 
 | |
|     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
 | |
| 
 | |
|   if (aUseStandins == UseStandins::Yes) {
 | |
|     aResult = GetStandinForNativeColor(aID, aScheme);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   auto& cache =
 | |
|       aScheme == ColorScheme::Light ? sLightColorCache : sDarkColorCache;
 | |
|   if (const auto* cached = cache.Get(aID)) {
 | |
|     if (cached->isNothing()) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
|     aResult = cached->value();
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   if (NS_SUCCEEDED(GetColorFromPref(aID, aScheme, aResult))) {
 | |
|     cache.Insert(aID, Some(aResult));
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   if (NS_SUCCEEDED(NativeGetColor(aID, aScheme, aResult))) {
 | |
|     if (gfxPlatform::GetCMSMode() == CMSMode::All &&
 | |
|         !IsSpecialColor(aID, aResult)) {
 | |
|       qcms_transform* transform = gfxPlatform::GetCMSInverseRGBTransform();
 | |
|       if (transform) {
 | |
|         uint8_t color[4];
 | |
|         color[0] = NS_GET_R(aResult);
 | |
|         color[1] = NS_GET_G(aResult);
 | |
|         color[2] = NS_GET_B(aResult);
 | |
|         color[3] = NS_GET_A(aResult);
 | |
|         qcms_transform_data(transform, color, color, 1);
 | |
|         aResult = NS_RGBA(color[0], color[1], color[2], color[3]);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // NOTE: Servo holds a lock and the main thread is paused, so writing to the
 | |
|     // global cache here is fine.
 | |
|     cache.Insert(aID, Some(aResult));
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   cache.Insert(aID, Nothing());
 | |
|   return NS_ERROR_FAILURE;
 | |
| }
 | |
| 
 | |
| 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(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;
 | |
|   const bool haveFont = NativeGetFont(aID, aName, aStyle);
 | |
|   font.haveFont() = haveFont;
 | |
|   if (haveFont) {
 | |
|     font = StyleToLookAndFeelFont(aName, aStyle);
 | |
|   }
 | |
|   sFontCache.Insert(aID, std::move(font));
 | |
|   return haveFont;
 | |
| }
 | |
| 
 | |
| void nsXPLookAndFeel::RefreshImpl() {
 | |
|   // Wipe out our caches.
 | |
|   sLightColorCache.Clear();
 | |
|   sDarkColorCache.Clear();
 | |
|   sFontCache.Clear();
 | |
|   sFloatCache.Clear();
 | |
|   sIntCache.Clear();
 | |
|   RecomputeColorSchemes();
 | |
| 
 | |
|   // Clear any cached FullLookAndFeel data, which is now invalid.
 | |
|   if (XRE_IsParentProcess()) {
 | |
|     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);
 | |
|         }
 | |
|       }));
 | |
| }
 | |
| 
 | |
| static bool ShouldUseStandinsForNativeColorForNonNativeTheme(
 | |
|     const dom::Document& aDoc, LookAndFeel::ColorID aColor) {
 | |
|   using ColorID = LookAndFeel::ColorID;
 | |
|   if (!aDoc.ShouldAvoidNativeTheme()) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // The native theme doesn't use native system colors backgrounds etc, except
 | |
|   // when in high-contrast mode, so spoof some of the colors with stand-ins to
 | |
|   // prevent lack of contrast.
 | |
|   switch (aColor) {
 | |
|     case ColorID::Buttonface:
 | |
|     case ColorID::Buttontext:
 | |
|     case ColorID::MozButtonhoverface:
 | |
|     case ColorID::MozButtonhovertext:
 | |
|     case ColorID::MozButtonactiveface:
 | |
|     case ColorID::MozButtonactivetext:
 | |
|     case ColorID::MozButtondisabledface:
 | |
| 
 | |
|     case ColorID::Threedlightshadow:
 | |
|     case ColorID::Threeddarkshadow:
 | |
|     case ColorID::Threedface:
 | |
| 
 | |
|     case ColorID::MozCombobox:
 | |
|     case ColorID::MozComboboxtext:
 | |
| 
 | |
|     case ColorID::Field:
 | |
|     case ColorID::MozDisabledfield:
 | |
|     case ColorID::Fieldtext:
 | |
| 
 | |
|     case ColorID::Graytext:
 | |
| 
 | |
|       return !PreferenceSheet::PrefsFor(aDoc)
 | |
|                   .NonNativeThemeShouldBeHighContrast();
 | |
| 
 | |
|     default:
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| ColorScheme LookAndFeel::sChromeColorScheme;
 | |
| ColorScheme LookAndFeel::sContentColorScheme;
 | |
| bool LookAndFeel::sColorSchemeInitialized;
 | |
| bool LookAndFeel::sGlobalThemeChanged;
 | |
| 
 | |
| 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 ColorSchemeForChrome();
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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;
 | |
|       case 2:
 | |
|         return SystemColorScheme();
 | |
|       default:
 | |
|         break;  // Use the browser theme.
 | |
|     }
 | |
| 
 | |
|     return ThemeDerivedColorSchemeForContent();
 | |
|   }();
 | |
| }
 | |
| 
 | |
| ColorScheme LookAndFeel::ColorSchemeForStyle(
 | |
|     const dom::Document& aDoc, const StyleColorSchemeFlags& aFlags) {
 | |
|   if (PreferenceSheet::MayForceColors()) {
 | |
|     auto& prefs = PreferenceSheet::PrefsFor(aDoc);
 | |
|     if (!prefs.mUseDocumentColors) {
 | |
|       // When forcing colors, we can use our preferred color-scheme. Do this
 | |
|       // only if we're using system colors, as dark preference colors are not
 | |
|       // exposed on the UI.
 | |
|       //
 | |
|       // Also, use light if we're using a high-contrast-theme on Windows, since
 | |
|       // Windows overrides the light colors with HCM colors when HCM is active.
 | |
| #ifdef XP_WIN
 | |
|       if (prefs.mUseAccessibilityTheme) {
 | |
|         return ColorScheme::Light;
 | |
|       }
 | |
| #endif
 | |
|       if (StaticPrefs::browser_display_use_system_colors()) {
 | |
|         return aDoc.PreferredColorScheme();
 | |
|       }
 | |
|       return ColorScheme::Light;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   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 (nsContentUtils::IsChromeDoc(&aDoc)) {
 | |
|     return aDoc.PreferredColorScheme();
 | |
|   }
 | |
|   // Default content to light.
 | |
|   return ColorScheme::Light;
 | |
| }
 | |
| 
 | |
| LookAndFeel::ColorScheme LookAndFeel::ColorSchemeForFrame(
 | |
|     const nsIFrame* aFrame) {
 | |
|   return ColorSchemeForStyle(*aFrame->PresContext()->Document(),
 | |
|                              aFrame->StyleUI()->mColorScheme.bits);
 | |
| }
 | |
| 
 | |
| // 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) {
 | |
|   if (ShouldUseStandinsForNativeColorForNonNativeTheme(aDoc, aId)) {
 | |
|     return UseStandins::Yes;
 | |
|   }
 | |
|   if (nsContentUtils::UseStandinsForNativeColors() &&
 | |
|       ColorIsCSSAccessible(aId) && !nsContentUtils::IsChromeDoc(&aDoc)) {
 | |
|     return UseStandins::Yes;
 | |
|   }
 | |
|   if (aDoc.IsStaticDocument() &&
 | |
|       !PreferenceSheet::ContentPrefs().mUseDocumentColors) {
 | |
|     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);
 | |
| }
 | |
| 
 | |
| // 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
 |