/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* 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 "nsLookAndFeel.h" #include "nsCocoaFeatures.h" #include "nsIServiceManager.h" #include "nsNativeThemeColors.h" #include "nsStyleConsts.h" #include "nsCocoaFeatures.h" #include "gfxFont.h" #include "gfxFontConstants.h" #include "gfxPlatformMac.h" #include "mozilla/gfx/2D.h" #include "mozilla/widget/WidgetMessageUtils.h" #import // This must be included last: #include "nsObjCExceptions.h" enum { mozNSScrollerStyleLegacy = 0, mozNSScrollerStyleOverlay = 1 }; typedef int32_t NSInteger; // sigh typedef NSInteger mozNSScrollerStyle; @interface NSScroller(AvailableSinceLion) + (mozNSScrollerStyle)preferredScrollerStyle; @end nsLookAndFeel::nsLookAndFeel() : nsXPLookAndFeel() , mUseOverlayScrollbars(-1) , mUseOverlayScrollbarsCached(false) , mAllowOverlayScrollbarsOverlap(-1) , mAllowOverlayScrollbarsOverlapCached(false) { } nsLookAndFeel::~nsLookAndFeel() { } static nscolor GetColorFromNSColor(NSColor* aColor) { NSColor* deviceColor = [aColor colorUsingColorSpaceName:NSDeviceRGBColorSpace]; return NS_RGB((unsigned int)([deviceColor redComponent] * 255.0), (unsigned int)([deviceColor greenComponent] * 255.0), (unsigned int)([deviceColor blueComponent] * 255.0)); } nsresult nsLookAndFeel::NativeGetColor(ColorID aID, nscolor &aColor) { NS_OBJC_BEGIN_TRY_ABORT_BLOCK_NSRESULT; nsresult res = NS_OK; switch (aID) { case eColorID_WindowBackground: aColor = NS_RGB(0xff,0xff,0xff); return res; case eColorID_WindowForeground: aColor = NS_RGB(0x00,0x00,0x00); return res; case eColorID_WidgetBackground: aColor = NS_RGB(0xdd,0xdd,0xdd); return res; case eColorID_WidgetForeground: aColor = NS_RGB(0x00,0x00,0x00); return res; case eColorID_WidgetSelectBackground: aColor = NS_RGB(0x80,0x80,0x80); return res; case eColorID_WidgetSelectForeground: aColor = NS_RGB(0x00,0x00,0x80); return res; case eColorID_Widget3DHighlight: aColor = NS_RGB(0xa0,0xa0,0xa0); return res; case eColorID_Widget3DShadow: aColor = NS_RGB(0x40,0x40,0x40); return res; case eColorID_TextBackground: aColor = NS_RGB(0xff,0xff,0xff); return res; case eColorID_TextForeground: aColor = NS_RGB(0x00,0x00,0x00); return res; case eColorID_TextSelectBackground: aColor = GetColorFromNSColor([NSColor selectedTextBackgroundColor]); return res; case eColorID_highlight: // CSS2 color aColor = GetColorFromNSColor([NSColor alternateSelectedControlColor]); return res; case eColorID__moz_menuhover: aColor = GetColorFromNSColor([NSColor alternateSelectedControlColor]); return res; case eColorID_TextSelectForeground: GetColor(eColorID_TextSelectBackground, aColor); if (aColor == 0x000000) aColor = NS_RGB(0xff,0xff,0xff); else aColor = NS_DONT_CHANGE_COLOR; return res; case eColorID_highlighttext: // CSS2 color case eColorID__moz_menuhovertext: aColor = GetColorFromNSColor([NSColor alternateSelectedControlTextColor]); return res; case eColorID_IMESelectedRawTextBackground: case eColorID_IMESelectedConvertedTextBackground: case eColorID_IMERawInputBackground: case eColorID_IMEConvertedTextBackground: aColor = NS_TRANSPARENT; return res; case eColorID_IMESelectedRawTextForeground: case eColorID_IMESelectedConvertedTextForeground: case eColorID_IMERawInputForeground: case eColorID_IMEConvertedTextForeground: aColor = NS_SAME_AS_FOREGROUND_COLOR; return res; case eColorID_IMERawInputUnderline: case eColorID_IMEConvertedTextUnderline: aColor = NS_40PERCENT_FOREGROUND_COLOR; return res; case eColorID_IMESelectedRawTextUnderline: case eColorID_IMESelectedConvertedTextUnderline: aColor = NS_SAME_AS_FOREGROUND_COLOR; return res; case eColorID_SpellCheckerUnderline: aColor = NS_RGB(0xff, 0, 0); return res; // // css2 system colors http://www.w3.org/TR/REC-CSS2/ui.html#system-colors // // It's really hard to effectively map these to the Appearance Manager properly, // since they are modeled word for word after the win32 system colors and don't have any // real counterparts in the Mac world. I'm sure we'll be tweaking these for // years to come. // // Thanks to mpt26@student.canterbury.ac.nz for the hardcoded values that form the defaults // if querying the Appearance Manager fails ;) // case eColorID__moz_mac_buttonactivetext: case eColorID__moz_mac_defaultbuttontext: #ifdef __LP64__ if (nsCocoaFeatures::OnYosemiteOrLater()) { aColor = NS_RGB(0xFF,0xFF,0xFF); return res; } // Otherwise fall through and return the regular button text: #endif case eColorID_buttontext: case eColorID__moz_buttonhovertext: aColor = GetColorFromNSColor([NSColor controlTextColor]); return res; case eColorID_captiontext: case eColorID_menutext: case eColorID_infotext: case eColorID__moz_menubartext: aColor = GetColorFromNSColor([NSColor textColor]); return res; case eColorID_windowtext: aColor = GetColorFromNSColor([NSColor windowFrameTextColor]); return res; case eColorID_activecaption: aColor = GetColorFromNSColor([NSColor gridColor]); return res; case eColorID_activeborder: aColor = NS_RGB(0x00,0x00,0x00); return res; case eColorID_appworkspace: aColor = NS_RGB(0xFF,0xFF,0xFF); return res; case eColorID_background: aColor = NS_RGB(0x63,0x63,0xCE); return res; case eColorID_buttonface: case eColorID__moz_buttonhoverface: aColor = NS_RGB(0xF0,0xF0,0xF0); return res; case eColorID_buttonhighlight: aColor = NS_RGB(0xFF,0xFF,0xFF); return res; case eColorID_buttonshadow: aColor = NS_RGB(0xDC,0xDC,0xDC); return res; case eColorID_graytext: aColor = GetColorFromNSColor([NSColor disabledControlTextColor]); return res; case eColorID_inactiveborder: aColor = GetColorFromNSColor([NSColor controlBackgroundColor]); return res; case eColorID_inactivecaption: aColor = GetColorFromNSColor([NSColor controlBackgroundColor]); return res; case eColorID_inactivecaptiontext: aColor = NS_RGB(0x45,0x45,0x45); return res; case eColorID_scrollbar: aColor = GetColorFromNSColor([NSColor scrollBarColor]); return res; case eColorID_threeddarkshadow: aColor = NS_RGB(0xDC,0xDC,0xDC); return res; case eColorID_threedshadow: aColor = NS_RGB(0xE0,0xE0,0xE0); return res; case eColorID_threedface: aColor = NS_RGB(0xF0,0xF0,0xF0); return res; case eColorID_threedhighlight: aColor = GetColorFromNSColor([NSColor highlightColor]); return res; case eColorID_threedlightshadow: aColor = NS_RGB(0xDA,0xDA,0xDA); return res; case eColorID_menu: aColor = GetColorFromNSColor([NSColor alternateSelectedControlTextColor]); return res; case eColorID_infobackground: aColor = NS_RGB(0xFF,0xFF,0xC7); return res; case eColorID_windowframe: aColor = GetColorFromNSColor([NSColor gridColor]); return res; case eColorID_window: case eColorID__moz_field: case eColorID__moz_combobox: aColor = NS_RGB(0xff,0xff,0xff); return res; case eColorID__moz_fieldtext: case eColorID__moz_comboboxtext: aColor = GetColorFromNSColor([NSColor controlTextColor]); return res; case eColorID__moz_dialog: aColor = GetColorFromNSColor([NSColor controlHighlightColor]); return res; case eColorID__moz_dialogtext: case eColorID__moz_cellhighlighttext: case eColorID__moz_html_cellhighlighttext: aColor = GetColorFromNSColor([NSColor controlTextColor]); return res; case eColorID__moz_dragtargetzone: aColor = GetColorFromNSColor([NSColor selectedControlColor]); return res; case eColorID__moz_mac_chrome_active: case eColorID__moz_mac_chrome_inactive: { int grey = NativeGreyColorAsInt(headerEndGrey, (aID == eColorID__moz_mac_chrome_active)); aColor = NS_RGB(grey, grey, grey); } return res; case eColorID__moz_mac_focusring: aColor = GetColorFromNSColor([NSColor keyboardFocusIndicatorColor]); return res; case eColorID__moz_mac_menushadow: aColor = NS_RGB(0xA3,0xA3,0xA3); return res; case eColorID__moz_mac_menutextdisable: aColor = NS_RGB(0x98,0x98,0x98); return res; case eColorID__moz_mac_menutextselect: aColor = GetColorFromNSColor([NSColor selectedMenuItemTextColor]); return res; case eColorID__moz_mac_disabledtoolbartext: aColor = GetColorFromNSColor([NSColor disabledControlTextColor]); return res; case eColorID__moz_mac_menuselect: aColor = GetColorFromNSColor([NSColor alternateSelectedControlColor]); return res; case eColorID__moz_buttondefault: aColor = NS_RGB(0xDC,0xDC,0xDC); return res; case eColorID__moz_cellhighlight: case eColorID__moz_html_cellhighlight: case eColorID__moz_mac_secondaryhighlight: // For inactive list selection aColor = GetColorFromNSColor([NSColor secondarySelectedControlColor]); return res; case eColorID__moz_eventreerow: // Background color of even list rows. aColor = GetColorFromNSColor([[NSColor controlAlternatingRowBackgroundColors] objectAtIndex:0]); return res; case eColorID__moz_oddtreerow: // Background color of odd list rows. aColor = GetColorFromNSColor([[NSColor controlAlternatingRowBackgroundColors] objectAtIndex:1]); return res; case eColorID__moz_nativehyperlinktext: // There appears to be no available system defined color. HARDCODING to the appropriate color. aColor = NS_RGB(0x14,0x4F,0xAE); return res; default: NS_WARNING("Someone asked nsILookAndFeel for a color I don't know about"); aColor = NS_RGB(0xff,0xff,0xff); res = NS_ERROR_FAILURE; return res; } // unreachable? res = NS_ERROR_FAILURE; return res; NS_OBJC_END_TRY_ABORT_BLOCK_NSRESULT; } nsresult nsLookAndFeel::GetIntImpl(IntID aID, int32_t &aResult) { NS_OBJC_BEGIN_TRY_ABORT_BLOCK_NSRESULT; nsresult res = nsXPLookAndFeel::GetIntImpl(aID, aResult); if (NS_SUCCEEDED(res)) return res; res = NS_OK; switch (aID) { case eIntID_CaretBlinkTime: aResult = 567; return res; case eIntID_CaretWidth: aResult = 1; return res; case eIntID_ShowCaretDuringSelection: aResult = 0; return res; case eIntID_SelectTextfieldsOnKeyFocus: // Select textfield content when focused by kbd // used by EventStateManager::sTextfieldSelectModel aResult = 1; return res; case eIntID_SubmenuDelay: aResult = 200; return res; case eIntID_TooltipDelay: aResult = 500; return res; case eIntID_MenusCanOverlapOSBar: // xul popups are not allowed to overlap the menubar. aResult = 0; return res; case eIntID_SkipNavigatingDisabledMenuItem: aResult = 1; return res; case eIntID_DragThresholdX: case eIntID_DragThresholdY: aResult = 4; return res; case eIntID_ScrollArrowStyle: #ifdef __LP64__ if (nsCocoaFeatures::OnLionOrLater()) { // OS X Lion's scrollbars have no arrows aResult = eScrollArrow_None; } else { #else { #endif // This can get called during startup and can leak. NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init]; NSString *buttonPlacement = [[NSUserDefaults standardUserDefaults] objectForKey:@"AppleScrollBarVariant"]; if ([buttonPlacement isEqualToString:@"Single"]) { aResult = eScrollArrowStyle_Single; } else if ([buttonPlacement isEqualToString:@"DoubleMin"]) { aResult = eScrollArrowStyle_BothAtTop; } else if ([buttonPlacement isEqualToString:@"DoubleBoth"]) { aResult = eScrollArrowStyle_BothAtEachEnd; } else { aResult = eScrollArrowStyle_BothAtBottom; // The default is BothAtBottom. } [pool drain]; } return res; case eIntID_ScrollSliderStyle: aResult = eScrollThumbStyle_Proportional; return res; case eIntID_UseOverlayScrollbars: aResult = 0; #ifdef __LP64__ if (!mUseOverlayScrollbarsCached) { mUseOverlayScrollbars = SystemWantsOverlayScrollbars() ? 1 : 0; mUseOverlayScrollbarsCached = true; } aResult = mUseOverlayScrollbars; #endif return res; case eIntID_AllowOverlayScrollbarsOverlap: aResult = 0; #ifdef __LP64__ if (!mAllowOverlayScrollbarsOverlapCached) { mAllowOverlayScrollbarsOverlap = AllowOverlayScrollbarsOverlap() ? 1 : 0; mAllowOverlayScrollbarsOverlapCached = true; } aResult = mAllowOverlayScrollbarsOverlap; #endif return res; case eIntID_ScrollbarDisplayOnMouseMove: aResult = 0; return res; case eIntID_ScrollbarFadeBeginDelay: aResult = 450; return res; case eIntID_ScrollbarFadeDuration: aResult = 200; return res; case eIntID_TreeOpenDelay: aResult = 1000; return res; case eIntID_TreeCloseDelay: aResult = 1000; return res; case eIntID_TreeLazyScrollDelay: aResult = 150; return res; case eIntID_TreeScrollDelay: aResult = 100; return res; case eIntID_TreeScrollLinesMax: aResult = 3; return res; case eIntID_DWMCompositor: case eIntID_WindowsClassic: case eIntID_WindowsDefaultTheme: case eIntID_TouchEnabled: case eIntID_WindowsThemeIdentifier: case eIntID_OperatingSystemVersionIdentifier: aResult = 0; res = NS_ERROR_NOT_IMPLEMENTED; return res; case eIntID_MacGraphiteTheme: aResult = [NSColor currentControlTint] == NSGraphiteControlTint; return res; case eIntID_MacLionTheme: aResult = nsCocoaFeatures::OnLionOrLater(); return res; case eIntID_MacYosemiteTheme: aResult = nsCocoaFeatures::OnYosemiteOrLater(); return res; case eIntID_AlertNotificationOrigin: aResult = NS_ALERT_TOP; return res; case eIntID_TabFocusModel: { fprintf(stderr, "TabFocusModel\n"); // we should probably cache this CFPropertyListRef fullKeyboardAccessProperty; fullKeyboardAccessProperty = ::CFPreferencesCopyValue(CFSTR("AppleKeyboardUIMode"), kCFPreferencesAnyApplication, kCFPreferencesCurrentUser, kCFPreferencesAnyHost); aResult = 1; // default to just textboxes if (fullKeyboardAccessProperty) { int32_t fullKeyboardAccessPrefVal; if (::CFNumberGetValue((CFNumberRef) fullKeyboardAccessProperty, kCFNumberIntType, &fullKeyboardAccessPrefVal)) { // the second bit means "Full keyboard access" is on if (fullKeyboardAccessPrefVal & (1 << 1)) aResult = 7; // everything that can be focused } ::CFRelease(fullKeyboardAccessProperty); } } return res; case eIntID_ScrollToClick: { aResult = [[NSUserDefaults standardUserDefaults] boolForKey:@"AppleScrollerPagingBehavior"]; } return res; case eIntID_ChosenMenuItemsShouldBlink: aResult = 1; return res; case eIntID_IMERawInputUnderlineStyle: case eIntID_IMEConvertedTextUnderlineStyle: case eIntID_IMESelectedRawTextUnderlineStyle: case eIntID_IMESelectedConvertedTextUnderline: aResult = NS_STYLE_TEXT_DECORATION_STYLE_SOLID; return res; case eIntID_SpellCheckerUnderlineStyle: aResult = NS_STYLE_TEXT_DECORATION_STYLE_DOTTED; return res; case eIntID_ScrollbarButtonAutoRepeatBehavior: aResult = 0; return res; case eIntID_SwipeAnimationEnabled: aResult = 0; #ifdef __LP64__ if ([NSEvent respondsToSelector:@selector( isSwipeTrackingFromScrollEventsEnabled)]) { aResult = [NSEvent isSwipeTrackingFromScrollEventsEnabled] ? 1 : 0; } #endif return res; case eIntID_ColorPickerAvailable: aResult = 1; return res; case eIntID_ContextMenuOffsetVertical: aResult = -6; return res; case eIntID_ContextMenuOffsetHorizontal: aResult = 1; return res; default: aResult = 0; res = NS_ERROR_FAILURE; } return res; NS_OBJC_END_TRY_ABORT_BLOCK_NSRESULT; } nsresult nsLookAndFeel::GetFloatImpl(FloatID aID, float &aResult) { nsresult res = nsXPLookAndFeel::GetFloatImpl(aID, aResult); if (NS_SUCCEEDED(res)) return res; res = NS_OK; switch (aID) { case eFloatID_IMEUnderlineRelativeSize: aResult = 2.0f; return res; case eFloatID_SpellCheckerUnderlineRelativeSize: aResult = 2.0f; return res; default: aResult = -1.0; res = NS_ERROR_FAILURE; } return res; } bool nsLookAndFeel::UseOverlayScrollbars() { #ifdef __LP64__ return GetInt(eIntID_UseOverlayScrollbars) != 0; #else return false; // Unpossible before 10.7. #endif } bool nsLookAndFeel::SystemWantsOverlayScrollbars() { #ifdef __LP64__ return ([NSScroller respondsToSelector:@selector(preferredScrollerStyle)] && [NSScroller preferredScrollerStyle] == mozNSScrollerStyleOverlay); #else return false; // Unpossible before 10.7. #endif } bool nsLookAndFeel::AllowOverlayScrollbarsOverlap() { #ifdef __LP64__ return (UseOverlayScrollbars() && nsCocoaFeatures::OnMountainLionOrLater()); #else return false; // Unpossible before 10.7. #endif } bool nsLookAndFeel::GetFontImpl(FontID aID, nsString &aFontName, gfxFontStyle &aFontStyle, float aDevPixPerCSSPixel) { NS_OBJC_BEGIN_TRY_ABORT_BLOCK_RETURN; // hack for now if (aID == eFont_Window || aID == eFont_Document) { aFontStyle.style = NS_FONT_STYLE_NORMAL; aFontStyle.weight = NS_FONT_WEIGHT_NORMAL; aFontStyle.stretch = NS_FONT_STRETCH_NORMAL; aFontStyle.size = 14 * aDevPixPerCSSPixel; aFontStyle.systemFont = true; aFontName.AssignLiteral("sans-serif"); return true; } gfxPlatformMac::LookupSystemFont(aID, aFontName, aFontStyle, aDevPixPerCSSPixel); return true; NS_OBJC_END_TRY_ABORT_BLOCK_RETURN(false); } nsTArray nsLookAndFeel::GetIntCacheImpl() { nsTArray lookAndFeelIntCache = nsXPLookAndFeel::GetIntCacheImpl(); // Useless before 10.7. #ifdef __LP64__ LookAndFeelInt useOverlayScrollbars; useOverlayScrollbars.id = eIntID_UseOverlayScrollbars; useOverlayScrollbars.value = GetInt(eIntID_UseOverlayScrollbars); lookAndFeelIntCache.AppendElement(useOverlayScrollbars); LookAndFeelInt allowOverlayScrollbarsOverlap; allowOverlayScrollbarsOverlap.id = eIntID_AllowOverlayScrollbarsOverlap; allowOverlayScrollbarsOverlap.value = GetInt(eIntID_AllowOverlayScrollbarsOverlap); lookAndFeelIntCache.AppendElement(allowOverlayScrollbarsOverlap); #endif return lookAndFeelIntCache; } void nsLookAndFeel::SetIntCacheImpl(const nsTArray& aLookAndFeelIntCache) { // Useless before 10.7. #ifdef __LP64__ for (auto entry : aLookAndFeelIntCache) { switch(entry.id) { case eIntID_UseOverlayScrollbars: mUseOverlayScrollbars = entry.value; mUseOverlayScrollbarsCached = true; return res; case eIntID_AllowOverlayScrollbarsOverlap: mAllowOverlayScrollbarsOverlap = entry.value; mAllowOverlayScrollbarsOverlapCached = true; return res; } } #endif } void nsLookAndFeel::RefreshImpl() { // Useless before 10.7. #ifdef __LP64__ // We should only clear the cache if we're in the main browser process. // Otherwise, we should wait for the parent to inform us of new values // to cache via LookAndFeel::SetIntCache. if (XRE_IsParentProcess()) { mUseOverlayScrollbarsCached = false; mAllowOverlayScrollbarsOverlapCached = false; } #endif }