| /* |
| * Copyright (C) 2016-2022 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
| * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
| * THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #import "config.h" |
| #import "RenderThemeCocoa.h" |
| |
| #import "ApplePayLogoSystemImage.h" |
| #import "FontCacheCoreText.h" |
| #import "GraphicsContextCG.h" |
| #import "HTMLInputElement.h" |
| #import "ImageBuffer.h" |
| #import "RenderText.h" |
| #import "UserAgentScripts.h" |
| #import "UserAgentStyleSheets.h" |
| #import <algorithm> |
| #import <pal/spi/cf/CoreTextSPI.h> |
| #import <wtf/Language.h> |
| |
| #if ENABLE(VIDEO) |
| #import "LocalizedStrings.h" |
| #import <wtf/BlockObjCExceptions.h> |
| #endif |
| |
| #if PLATFORM(MAC) |
| #import <AppKit/NSFont.h> |
| #else |
| #import <UIKit/UIFont.h> |
| #import <pal/ios/UIKitSoftLink.h> |
| #endif |
| |
| #if ENABLE(APPLE_PAY) |
| #import <pal/cocoa/PassKitSoftLink.h> |
| #endif |
| |
| @interface WebCoreRenderThemeBundle : NSObject |
| @end |
| |
| @implementation WebCoreRenderThemeBundle |
| @end |
| |
| namespace WebCore { |
| |
| RenderThemeCocoa& RenderThemeCocoa::singleton() |
| { |
| return static_cast<RenderThemeCocoa&>(RenderTheme::singleton()); |
| } |
| |
| void RenderThemeCocoa::purgeCaches() |
| { |
| #if ENABLE(VIDEO) && ENABLE(MODERN_MEDIA_CONTROLS) |
| m_mediaControlsLocalizedStringsScript.clearImplIfNotShared(); |
| m_mediaControlsScript.clearImplIfNotShared(); |
| m_mediaControlsStyleSheet.clearImplIfNotShared(); |
| #endif // ENABLE(VIDEO) && ENABLE(MODERN_MEDIA_CONTROLS) |
| |
| RenderTheme::purgeCaches(); |
| } |
| |
| bool RenderThemeCocoa::shouldHaveCapsLockIndicator(const HTMLInputElement& element) const |
| { |
| return element.isPasswordField(); |
| } |
| |
| #if ENABLE(APPLE_PAY) |
| |
| static const auto applePayButtonMinimumWidth = 140; |
| static const auto applePayButtonPlainMinimumWidth = 100; |
| static const auto applePayButtonMinimumHeight = 30; |
| |
| void RenderThemeCocoa::adjustApplePayButtonStyle(RenderStyle& style, const Element*) const |
| { |
| if (style.applePayButtonType() == ApplePayButtonType::Plain) |
| style.setMinWidth(Length(applePayButtonPlainMinimumWidth, LengthType::Fixed)); |
| else |
| style.setMinWidth(Length(applePayButtonMinimumWidth, LengthType::Fixed)); |
| style.setMinHeight(Length(applePayButtonMinimumHeight, LengthType::Fixed)); |
| |
| if (!style.hasExplicitlySetBorderRadius()) { |
| auto cornerRadius = PKApplePayButtonDefaultCornerRadius; |
| style.setBorderRadius({ { cornerRadius, LengthType::Fixed }, { cornerRadius, LengthType::Fixed } }); |
| } |
| } |
| |
| bool RenderThemeCocoa::paintApplePayButton(const RenderObject& renderer, const PaintInfo& paintInfo, const IntRect& paintRect) |
| { |
| auto& style = renderer.style(); |
| auto largestCornerRadius = std::max<CGFloat>({ |
| floatValueForLength(style.borderTopLeftRadius().height, paintRect.height()), |
| floatValueForLength(style.borderTopLeftRadius().width, paintRect.width()), |
| floatValueForLength(style.borderTopRightRadius().height, paintRect.height()), |
| floatValueForLength(style.borderTopRightRadius().width, paintRect.width()), |
| floatValueForLength(style.borderBottomLeftRadius().height, paintRect.height()), |
| floatValueForLength(style.borderBottomLeftRadius().width, paintRect.width()), |
| floatValueForLength(style.borderBottomRightRadius().height, paintRect.height()), |
| floatValueForLength(style.borderBottomRightRadius().width, paintRect.width()) |
| }); |
| String locale = style.computedLocale(); |
| if (locale.isEmpty()) |
| locale = defaultLanguage(ShouldMinimizeLanguages::No); |
| paintInfo.context().drawSystemImage(ApplePayButtonSystemImage::create(style.applePayButtonType(), style.applePayButtonStyle(), locale, largestCornerRadius), paintRect); |
| return false; |
| } |
| |
| #endif // ENABLE(APPLE_PAY) |
| |
| #if ENABLE(VIDEO) && ENABLE(MODERN_MEDIA_CONTROLS) |
| |
| String RenderThemeCocoa::mediaControlsStyleSheet() |
| { |
| if (m_mediaControlsStyleSheet.isEmpty()) |
| m_mediaControlsStyleSheet = StringImpl::createWithoutCopying(ModernMediaControlsUserAgentStyleSheet, sizeof(ModernMediaControlsUserAgentStyleSheet)); |
| return m_mediaControlsStyleSheet; |
| } |
| |
| Vector<String, 2> RenderThemeCocoa::mediaControlsScripts() |
| { |
| // FIXME: Localized strings are not worth having a script. We should make it JSON data etc. instead. |
| if (m_mediaControlsLocalizedStringsScript.isEmpty()) { |
| NSBundle *bundle = [NSBundle bundleForClass:[WebCoreRenderThemeBundle class]]; |
| m_mediaControlsLocalizedStringsScript = [NSString stringWithContentsOfFile:[bundle pathForResource:@"modern-media-controls-localized-strings" ofType:@"js"] encoding:NSUTF8StringEncoding error:nil]; |
| } |
| |
| if (m_mediaControlsScript.isEmpty()) |
| m_mediaControlsScript = StringImpl::createWithoutCopying(ModernMediaControlsJavaScript, sizeof(ModernMediaControlsJavaScript)); |
| |
| return { |
| m_mediaControlsLocalizedStringsScript, |
| m_mediaControlsScript, |
| }; |
| } |
| |
| String RenderThemeCocoa::mediaControlsBase64StringForIconNameAndType(const String& iconName, const String& iconType) |
| { |
| NSString *directory = @"modern-media-controls/images"; |
| NSBundle *bundle = [NSBundle bundleForClass:[WebCoreRenderThemeBundle class]]; |
| return [[NSData dataWithContentsOfFile:[bundle pathForResource:iconName ofType:iconType inDirectory:directory]] base64EncodedStringWithOptions:0]; |
| } |
| |
| String RenderThemeCocoa::mediaControlsFormattedStringForDuration(const double durationInSeconds) |
| { |
| if (!std::isfinite(durationInSeconds)) |
| return WEB_UI_STRING("indefinite time", "accessibility help text for an indefinite media controller time value"); |
| |
| BEGIN_BLOCK_OBJC_EXCEPTIONS |
| if (!m_durationFormatter) { |
| m_durationFormatter = adoptNS([NSDateComponentsFormatter new]); |
| m_durationFormatter.get().unitsStyle = NSDateComponentsFormatterUnitsStyleFull; |
| m_durationFormatter.get().allowedUnits = NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond; |
| m_durationFormatter.get().formattingContext = NSFormattingContextStandalone; |
| m_durationFormatter.get().maximumUnitCount = 2; |
| } |
| return [m_durationFormatter stringFromTimeInterval:durationInSeconds]; |
| END_BLOCK_OBJC_EXCEPTIONS |
| } |
| |
| #endif // ENABLE(VIDEO) && ENABLE(MODERN_MEDIA_CONTROLS) |
| |
| static inline FontSelectionValue cssWeightOfSystemFont(CTFontRef font) |
| { |
| auto resultRef = adoptCF(static_cast<CFNumberRef>(CTFontCopyAttribute(font, kCTFontCSSWeightAttribute))); |
| float result = 0; |
| if (resultRef && CFNumberGetValue(resultRef.get(), kCFNumberFloatType, &result)) |
| return FontSelectionValue(result); |
| |
| auto traits = adoptCF(CTFontCopyTraits(font)); |
| resultRef = static_cast<CFNumberRef>(CFDictionaryGetValue(traits.get(), kCTFontWeightTrait)); |
| CFNumberGetValue(resultRef.get(), kCFNumberFloatType, &result); |
| // These numbers were experimentally gathered from weights of the system font. |
| static constexpr float weightThresholds[] = { -0.6, -0.365, -0.115, 0.130, 0.235, 0.350, 0.5, 0.7 }; |
| for (unsigned i = 0; i < WTF_ARRAY_LENGTH(weightThresholds); ++i) { |
| if (result < weightThresholds[i]) |
| return FontSelectionValue((static_cast<int>(i) + 1) * 100); |
| } |
| return FontSelectionValue(900); |
| } |
| |
| } |