/* * Copyright (C) 2007, 2008, 2011, 2013 Apple Inc. All rights reserved. * (C) 2007, 2008 Nikolas Zimmermann * * 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. ``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 * 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. */ #include "config.h" #include "CSSFontSelector.h" #include "CachedFont.h" #include "CSSFontFace.h" #include "CSSFontFaceRule.h" #include "CSSFontFaceSource.h" #include "CSSFontFaceSrcValue.h" #include "CSSFontFamily.h" #include "CSSFontFeatureValue.h" #include "CSSPrimitiveValue.h" #include "CSSPrimitiveValueMappings.h" #include "CSSPropertyNames.h" #include "CSSSegmentedFontFace.h" #include "CSSUnicodeRangeValue.h" #include "CSSValueKeywords.h" #include "CSSValueList.h" #include "CSSValuePool.h" #include "CachedResourceLoader.h" #include "Document.h" #include "Font.h" #include "FontCache.h" #include "FontVariantBuilder.h" #include "Frame.h" #include "FrameLoader.h" #include "SVGFontFaceElement.h" #include "SVGNames.h" #include "Settings.h" #include "StyleProperties.h" #include "StyleResolver.h" #include "StyleRule.h" #include "WebKitFontFamilyNames.h" #include #include namespace WebCore { static unsigned fontSelectorId; CSSFontSelector::CSSFontSelector(Document& document) : m_document(&document) , m_beginLoadingTimer(*this, &CSSFontSelector::beginLoadTimerFired) , m_uniqueId(++fontSelectorId) , m_version(0) { // FIXME: An old comment used to say there was no need to hold a reference to m_document // because "we are guaranteed to be destroyed before the document". But there does not // seem to be any such guarantee. ASSERT(m_document); FontCache::singleton().addClient(*this); } CSSFontSelector::~CSSFontSelector() { clearDocument(); FontCache::singleton().removeClient(*this); } bool CSSFontSelector::isEmpty() const { return m_fonts.isEmpty(); } void CSSFontSelector::appendSources(CSSFontFace& fontFace, CSSValueList& srcList, Document* document, bool isInitiatingElementInUserAgentShadowTree) { for (auto& src : srcList) { // An item in the list either specifies a string (local font name) or a URL (remote font to download). CSSFontFaceSrcValue& item = downcast(src.get()); std::unique_ptr source; SVGFontFaceElement* fontFaceElement = nullptr; bool foundSVGFont = false; #if ENABLE(SVG_FONTS) foundSVGFont = item.isSVGFontFaceSrc() || item.svgFontFaceElement(); fontFaceElement = item.svgFontFaceElement(); #endif if (!item.isLocal()) { Settings* settings = document ? document->settings() : nullptr; bool allowDownloading = foundSVGFont || (settings && settings->downloadableBinaryFontsEnabled()); if (allowDownloading && item.isSupportedFormat() && document) { if (CachedFont* cachedFont = item.cachedFont(document, foundSVGFont, isInitiatingElementInUserAgentShadowTree)) source = std::make_unique(fontFace, item.resource(), cachedFont); } } else source = std::make_unique(fontFace, item.resource(), nullptr, fontFaceElement); if (source) fontFace.adoptSource(WTFMove(source)); } fontFace.sourcesPopulated(); } String CSSFontSelector::familyNameFromPrimitive(const CSSPrimitiveValue& value) { if (value.isFontFamily()) return value.fontFamily().familyName; if (!value.isValueID()) return { }; // We need to use the raw text for all the generic family types, since @font-face is a way of actually // defining what font to use for those types. switch (value.getValueID()) { case CSSValueSerif: return serifFamily; case CSSValueSansSerif: return sansSerifFamily; case CSSValueCursive: return cursiveFamily; case CSSValueFantasy: return fantasyFamily; case CSSValueMonospace: return monospaceFamily; case CSSValueWebkitPictograph: return pictographFamily; default: return { }; } } void CSSFontSelector::registerLocalFontFacesForFamily(const String& familyName) { ASSERT(!m_locallyInstalledFontFaces.contains(familyName)); Vector traitsMasks = FontCache::singleton().getTraitsInFamily(familyName); if (traitsMasks.isEmpty()) return; Vector> faces = { }; for (auto mask : traitsMasks) { Ref face = CSSFontFace::create(*this, nullptr, true); RefPtr familyList = CSSValueList::createCommaSeparated(); familyList->append(CSSValuePool::singleton().createFontFamilyValue(familyName)); face->setFamilies(*familyList); face->setTraitsMask(mask); face->adoptSource(std::make_unique(face.get(), familyName)); ASSERT(!face->allSourcesFailed()); faces.append(WTFMove(face)); } m_locallyInstalledFontFaces.add(familyName, WTFMove(faces)); } void CSSFontSelector::addFontFaceRule(const StyleRuleFontFace& fontFaceRule, bool isInitiatingElementInUserAgentShadowTree) { const StyleProperties& style = fontFaceRule.properties(); RefPtr fontFamily = style.getPropertyCSSValue(CSSPropertyFontFamily); RefPtr fontStyle = style.getPropertyCSSValue(CSSPropertyFontStyle); RefPtr fontWeight = style.getPropertyCSSValue(CSSPropertyFontWeight); RefPtr src = style.getPropertyCSSValue(CSSPropertySrc); RefPtr unicodeRange = style.getPropertyCSSValue(CSSPropertyUnicodeRange); RefPtr featureSettings = style.getPropertyCSSValue(CSSPropertyFontFeatureSettings); RefPtr variantLigatures = style.getPropertyCSSValue(CSSPropertyFontVariantLigatures); RefPtr variantPosition = style.getPropertyCSSValue(CSSPropertyFontVariantPosition); RefPtr variantCaps = style.getPropertyCSSValue(CSSPropertyFontVariantCaps); RefPtr variantNumeric = style.getPropertyCSSValue(CSSPropertyFontVariantNumeric); RefPtr variantAlternates = style.getPropertyCSSValue(CSSPropertyFontVariantAlternates); RefPtr variantEastAsian = style.getPropertyCSSValue(CSSPropertyFontVariantEastAsian); if (!is(fontFamily.get()) || !is(src.get()) || (unicodeRange && !is(*unicodeRange))) return; CSSValueList& familyList = downcast(*fontFamily); if (!familyList.length()) return; if (!fontStyle) fontStyle = CSSValuePool::singleton().createIdentifierValue(CSSValueNormal).ptr(); if (!fontWeight) fontWeight = CSSValuePool::singleton().createIdentifierValue(CSSValueNormal); CSSValueList* rangeList = downcast(unicodeRange.get()); CSSValueList& srcList = downcast(*src); if (!srcList.length()) return; Ref fontFace = CSSFontFace::create(*this); if (!fontFace->setFamilies(*fontFamily)) return; if (!fontFace->setStyle(*fontStyle)) return; if (!fontFace->setWeight(*fontWeight)) return; if (rangeList && !fontFace->setUnicodeRange(*rangeList)) return; if (variantLigatures && !fontFace->setVariantLigatures(*variantLigatures)) return; if (variantPosition && !fontFace->setVariantPosition(*variantPosition)) return; if (variantCaps && !fontFace->setVariantCaps(*variantCaps)) return; if (variantNumeric && !fontFace->setVariantNumeric(*variantNumeric)) return; if (variantAlternates && !fontFace->setVariantAlternates(*variantAlternates)) return; if (variantEastAsian && !fontFace->setVariantEastAsian(*variantEastAsian)) return; if (featureSettings && !fontFace->setFeatureSettings(*featureSettings)) return; appendSources(fontFace, srcList, m_document, isInitiatingElementInUserAgentShadowTree); if (fontFace->allSourcesFailed()) return; for (auto& item : familyList) { String familyName = familyNameFromPrimitive(downcast(item.get())); if (familyName.isEmpty()) continue; auto addResult = m_fontFaces.add(familyName, Vector>()); auto& familyFontFaces = addResult.iterator->value; if (addResult.isNewEntry) { registerLocalFontFacesForFamily(familyName); familyFontFaces = { }; } familyFontFaces.append(fontFace.copyRef()); ++m_version; } } void CSSFontSelector::registerForInvalidationCallbacks(FontSelectorClient& client) { m_clients.add(&client); } void CSSFontSelector::unregisterForInvalidationCallbacks(FontSelectorClient& client) { m_clients.remove(&client); } void CSSFontSelector::dispatchInvalidationCallbacks() { ++m_version; Vector clients; copyToVector(m_clients, clients); for (size_t i = 0; i < clients.size(); ++i) clients[i]->fontsNeedUpdate(*this); } void CSSFontSelector::fontLoaded() { dispatchInvalidationCallbacks(); } void CSSFontSelector::fontCacheInvalidated() { dispatchInvalidationCallbacks(); } static const AtomicString& resolveGenericFamily(Document* document, const FontDescription& fontDescription, const AtomicString& familyName) { if (!document || !document->frame()) return familyName; const Settings& settings = document->frame()->settings(); UScriptCode script = fontDescription.script(); if (familyName == serifFamily) return settings.serifFontFamily(script); if (familyName == sansSerifFamily) return settings.sansSerifFontFamily(script); if (familyName == cursiveFamily) return settings.cursiveFontFamily(script); if (familyName == fantasyFamily) return settings.fantasyFontFamily(script); if (familyName == monospaceFamily) return settings.fixedFontFamily(script); if (familyName == pictographFamily) return settings.pictographFontFamily(script); if (familyName == standardFamily) return settings.standardFontFamily(script); return familyName; } class FontFaceComparator { public: FontFaceComparator(FontTraitsMask desiredTraitsMaskForComparison) : m_desiredTraitsMaskForComparison(desiredTraitsMaskForComparison) { ASSERT_WITH_SECURITY_IMPLICATION(m_desiredTraitsMaskForComparison & FontWeightMask); } bool operator()(const CSSFontFace& first, const CSSFontFace& second) { FontTraitsMask firstTraitsMask = first.traitsMask(); FontTraitsMask secondTraitsMask = second.traitsMask(); bool firstHasDesiredStyle = firstTraitsMask & m_desiredTraitsMaskForComparison & FontStyleMask; bool secondHasDesiredStyle = secondTraitsMask & m_desiredTraitsMaskForComparison & FontStyleMask; if (firstHasDesiredStyle != secondHasDesiredStyle) return firstHasDesiredStyle; if ((m_desiredTraitsMaskForComparison & FontStyleItalicMask) && !first.isLocalFallback() && !second.isLocalFallback()) { // Prefer a font that has indicated that it can only support italics to a font that claims to support // all styles. The specialized font is more likely to be the one the author wants used. bool firstRequiresItalics = (firstTraitsMask & FontStyleItalicMask) && !(firstTraitsMask & FontStyleNormalMask); bool secondRequiresItalics = (secondTraitsMask & FontStyleItalicMask) && !(secondTraitsMask & FontStyleNormalMask); if (firstRequiresItalics != secondRequiresItalics) return firstRequiresItalics; } if (secondTraitsMask & m_desiredTraitsMaskForComparison & FontWeightMask) return false; if (firstTraitsMask & m_desiredTraitsMaskForComparison & FontWeightMask) return true; // http://www.w3.org/TR/2011/WD-css3-fonts-20111004/#font-matching-algorithm says : // - If the desired weight is less than 400, weights below the desired weight are checked in descending order followed by weights above the desired weight in ascending order until a match is found. // - If the desired weight is greater than 500, weights above the desired weight are checked in ascending order followed by weights below the desired weight in descending order until a match is found. // - If the desired weight is 400, 500 is checked first and then the rule for desired weights less than 400 is used. // - If the desired weight is 500, 400 is checked first and then the rule for desired weights less than 400 is used. static const unsigned fallbackRuleSets = 9; static const unsigned rulesPerSet = 8; static const FontTraitsMask weightFallbackRuleSets[fallbackRuleSets][rulesPerSet] = { { FontWeight200Mask, FontWeight300Mask, FontWeight400Mask, FontWeight500Mask, FontWeight600Mask, FontWeight700Mask, FontWeight800Mask, FontWeight900Mask }, { FontWeight100Mask, FontWeight300Mask, FontWeight400Mask, FontWeight500Mask, FontWeight600Mask, FontWeight700Mask, FontWeight800Mask, FontWeight900Mask }, { FontWeight200Mask, FontWeight100Mask, FontWeight400Mask, FontWeight500Mask, FontWeight600Mask, FontWeight700Mask, FontWeight800Mask, FontWeight900Mask }, { FontWeight500Mask, FontWeight300Mask, FontWeight200Mask, FontWeight100Mask, FontWeight600Mask, FontWeight700Mask, FontWeight800Mask, FontWeight900Mask }, { FontWeight400Mask, FontWeight300Mask, FontWeight200Mask, FontWeight100Mask, FontWeight600Mask, FontWeight700Mask, FontWeight800Mask, FontWeight900Mask }, { FontWeight700Mask, FontWeight800Mask, FontWeight900Mask, FontWeight500Mask, FontWeight400Mask, FontWeight300Mask, FontWeight200Mask, FontWeight100Mask }, { FontWeight800Mask, FontWeight900Mask, FontWeight600Mask, FontWeight500Mask, FontWeight400Mask, FontWeight300Mask, FontWeight200Mask, FontWeight100Mask }, { FontWeight900Mask, FontWeight700Mask, FontWeight600Mask, FontWeight500Mask, FontWeight400Mask, FontWeight300Mask, FontWeight200Mask, FontWeight100Mask }, { FontWeight800Mask, FontWeight700Mask, FontWeight600Mask, FontWeight500Mask, FontWeight400Mask, FontWeight300Mask, FontWeight200Mask, FontWeight100Mask } }; unsigned ruleSetIndex = 0; for (; !(m_desiredTraitsMaskForComparison & (1 << (FontWeight100Bit + ruleSetIndex))); ruleSetIndex++) { } const FontTraitsMask* weightFallbackRule = weightFallbackRuleSets[ruleSetIndex]; for (unsigned i = 0; i < rulesPerSet; ++i) { if (secondTraitsMask & weightFallbackRule[i]) return false; if (firstTraitsMask & weightFallbackRule[i]) return true; } return false; } private: FontTraitsMask m_desiredTraitsMaskForComparison; }; FontRanges CSSFontSelector::fontRangesForFamily(const FontDescription& fontDescription, const AtomicString& familyName) { // FIXME: The spec (and Firefox) says user specified generic families (sans-serif etc.) should be resolved before the @font-face lookup too. bool resolveGenericFamilyFirst = familyName == standardFamily; AtomicString familyForLookup = resolveGenericFamilyFirst ? resolveGenericFamily(m_document, fontDescription, familyName) : familyName; CSSSegmentedFontFace* face = getFontFace(fontDescription, familyForLookup); if (!face) { if (!resolveGenericFamilyFirst) familyForLookup = resolveGenericFamily(m_document, fontDescription, familyName); return FontRanges(FontCache::singleton().fontForFamily(fontDescription, familyForLookup)); } return face->fontRanges(fontDescription); } CSSSegmentedFontFace* CSSFontSelector::getFontFace(const FontDescription& fontDescription, const AtomicString& family) { auto iterator = m_fontFaces.find(family); if (iterator == m_fontFaces.end()) return nullptr; auto& familyFontFaces = iterator->value; auto& segmentedFontFaceCache = m_fonts.add(family, HashMap>()).iterator->value; FontTraitsMask traitsMask = fontDescription.traitsMask(); auto& face = segmentedFontFaceCache.add(traitsMask, nullptr).iterator->value; if (face) return face.get(); face = std::make_unique(*this); Vector, 32> candidateFontFaces; for (int i = familyFontFaces.size() - 1; i >= 0; --i) { CSSFontFace& candidate = familyFontFaces[i]; unsigned candidateTraitsMask = candidate.traitsMask(); if ((traitsMask & FontStyleNormalMask) && !(candidateTraitsMask & FontStyleNormalMask)) continue; candidateFontFaces.append(candidate); } auto localIterator = m_locallyInstalledFontFaces.find(family); if (localIterator != m_locallyInstalledFontFaces.end()) { for (auto& candidate : localIterator->value) { unsigned candidateTraitsMask = candidate->traitsMask(); if ((traitsMask & FontStyleNormalMask) && !(candidateTraitsMask & FontStyleNormalMask)) continue; candidateFontFaces.append(candidate); } } std::stable_sort(candidateFontFaces.begin(), candidateFontFaces.end(), FontFaceComparator(traitsMask)); for (auto& candidate : candidateFontFaces) face->appendFontFace(candidate.get()); return face.get(); } void CSSFontSelector::clearDocument() { if (!m_document) { ASSERT(!m_beginLoadingTimer.isActive()); ASSERT(m_fontsToBeginLoading.isEmpty()); return; } m_beginLoadingTimer.stop(); CachedResourceLoader& cachedResourceLoader = m_document->cachedResourceLoader(); for (auto& fontHandle : m_fontsToBeginLoading) { // Balances incrementRequestCount() in beginLoadingFontSoon(). cachedResourceLoader.decrementRequestCount(fontHandle.get()); } m_fontsToBeginLoading.clear(); m_document = nullptr; // FIXME: This object should outlive the Document. m_fontFaces.clear(); m_locallyInstalledFontFaces.clear(); m_fonts.clear(); m_clients.clear(); } void CSSFontSelector::beginLoadingFontSoon(CachedFont* font) { if (!m_document) return; m_fontsToBeginLoading.append(font); // Increment the request count now, in order to prevent didFinishLoad from being dispatched // after this font has been requested but before it began loading. Balanced by // decrementRequestCount() in beginLoadTimerFired() and in clearDocument(). m_document->cachedResourceLoader().incrementRequestCount(font); m_beginLoadingTimer.startOneShot(0); } void CSSFontSelector::beginLoadTimerFired() { Vector> fontsToBeginLoading; fontsToBeginLoading.swap(m_fontsToBeginLoading); // CSSFontSelector could get deleted via beginLoadIfNeeded() or loadDone() unless protected. Ref protect(*this); CachedResourceLoader& cachedResourceLoader = m_document->cachedResourceLoader(); for (auto& fontHandle : fontsToBeginLoading) { fontHandle->beginLoadIfNeeded(cachedResourceLoader); // Balances incrementRequestCount() in beginLoadingFontSoon(). cachedResourceLoader.decrementRequestCount(fontHandle.get()); } // Ensure that if the request count reaches zero, the frame loader will know about it. cachedResourceLoader.loadDone(nullptr); // New font loads may be triggered by layout after the document load is complete but before we have dispatched // didFinishLoading for the frame. Make sure the delegate is always dispatched by checking explicitly. if (m_document && m_document->frame()) m_document->frame()->loader().checkLoadComplete(); } size_t CSSFontSelector::fallbackFontCount() { if (!m_document) return 0; if (Settings* settings = m_document->settings()) return settings->fontFallbackPrefersPictographs() ? 1 : 0; return 0; } RefPtr CSSFontSelector::fallbackFontAt(const FontDescription& fontDescription, size_t index) { ASSERT_UNUSED(index, !index); if (!m_document) return nullptr; Settings* settings = m_document->settings(); if (!settings || !settings->fontFallbackPrefersPictographs()) return nullptr; return FontCache::singleton().fontForFamily(fontDescription, settings->pictographFontFamily()); } }