mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
I wrote a script to do this which is attached to the bug. TBR=abarth@chromium.org BUG=435361 Review URL: https://codereview.chromium.org/736373003
273 lines
11 KiB
C++
273 lines
11 KiB
C++
/*
|
|
* Copyright (C) 1999 Lars Knoll (knoll@kde.org)
|
|
* Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011 Apple Inc. All rights reserved.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public License
|
|
* along with this library; see the file COPYING.LIB. If not, write to
|
|
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
#ifndef SKY_ENGINE_CORE_CSS_RESOLVER_STYLERESOLVER_H_
|
|
#define SKY_ENGINE_CORE_CSS_RESOLVER_STYLERESOLVER_H_
|
|
|
|
#include "sky/engine/core/css/RuleFeature.h"
|
|
#include "sky/engine/core/css/RuleSet.h"
|
|
#include "sky/engine/core/css/SelectorChecker.h"
|
|
#include "sky/engine/core/css/resolver/MatchedPropertiesCache.h"
|
|
#include "sky/engine/core/css/resolver/ScopedStyleResolver.h"
|
|
#include "sky/engine/core/css/resolver/StyleBuilder.h"
|
|
#include "sky/engine/core/css/resolver/StyleResourceLoader.h"
|
|
#include "sky/engine/platform/heap/Handle.h"
|
|
#include "sky/engine/wtf/Deque.h"
|
|
#include "sky/engine/wtf/HashMap.h"
|
|
#include "sky/engine/wtf/HashSet.h"
|
|
#include "sky/engine/wtf/ListHashSet.h"
|
|
#include "sky/engine/wtf/RefPtr.h"
|
|
#include "sky/engine/wtf/Vector.h"
|
|
|
|
namespace blink {
|
|
|
|
class AnimatableValue;
|
|
class AnimationTimeline;
|
|
class CSSAnimationUpdate;
|
|
class CSSFontSelector;
|
|
class CSSRuleList;
|
|
class CSSSelector;
|
|
class CSSStyleSheet;
|
|
class CSSValue;
|
|
class ContainerNode;
|
|
class Document;
|
|
class Element;
|
|
class ElementRuleCollector;
|
|
class Interpolation;
|
|
class KeyframeList;
|
|
class KeyframeValue;
|
|
class MediaQueryEvaluator;
|
|
class MediaQueryResult;
|
|
class RuleData;
|
|
class Settings;
|
|
class StyleKeyframe;
|
|
class StylePropertySet;
|
|
class StyleResolverStats;
|
|
class StyleRule;
|
|
class StyleRuleKeyframes;
|
|
|
|
class MatchResult;
|
|
|
|
enum StyleSharingBehavior {
|
|
AllowStyleSharing,
|
|
DisallowStyleSharing,
|
|
};
|
|
|
|
enum RuleMatchingBehavior {
|
|
MatchAllRules,
|
|
MatchAllRulesExcludingSMIL
|
|
};
|
|
|
|
const unsigned styleSharingListSize = 15;
|
|
const unsigned styleSharingMaxDepth = 32;
|
|
typedef Deque<RawPtr<Element>, styleSharingListSize> StyleSharingList;
|
|
|
|
struct CSSPropertyValue {
|
|
STACK_ALLOCATED();
|
|
public:
|
|
CSSPropertyValue(CSSPropertyID property, CSSValue* value)
|
|
: property(property), value(value) { }
|
|
// Stores value=propertySet.getPropertyCSSValue(id).get().
|
|
CSSPropertyValue(CSSPropertyID, const StylePropertySet&);
|
|
CSSPropertyID property;
|
|
RawPtr<CSSValue> value;
|
|
};
|
|
|
|
// This class selects a RenderStyle for a given element based on a collection of stylesheets.
|
|
class StyleResolver final {
|
|
WTF_MAKE_NONCOPYABLE(StyleResolver); WTF_MAKE_FAST_ALLOCATED;
|
|
public:
|
|
explicit StyleResolver(Document&);
|
|
virtual ~StyleResolver();
|
|
|
|
PassRefPtr<RenderStyle> styleForElement(Element*, RenderStyle* parentStyle = 0, StyleSharingBehavior = AllowStyleSharing,
|
|
RuleMatchingBehavior = MatchAllRules);
|
|
|
|
PassRefPtr<RenderStyle> styleForKeyframe(Element*, const RenderStyle&, RenderStyle* parentStyle, const StyleKeyframe*, const AtomicString& animationName);
|
|
static PassRefPtr<AnimatableValue> createAnimatableValueSnapshot(Element&, CSSPropertyID, CSSValue&);
|
|
static PassRefPtr<AnimatableValue> createAnimatableValueSnapshot(StyleResolverState&, CSSPropertyID, CSSValue&);
|
|
|
|
PassRefPtr<RenderStyle> defaultStyleForElement();
|
|
PassRefPtr<RenderStyle> styleForText(Text*);
|
|
|
|
static PassRefPtr<RenderStyle> styleForDocument(Document&);
|
|
|
|
// FIXME: This only has 5 callers and should be removed. Callers should be explicit about
|
|
// their dependency on Document* instead of grabbing one through StyleResolver.
|
|
Document& document() { return *m_document; }
|
|
|
|
// FIXME: It could be better to call appendAuthorStyleSheets() directly after we factor StyleResolver further.
|
|
// https://bugs.webkit.org/show_bug.cgi?id=108890
|
|
void appendAuthorStyleSheets(const Vector<RefPtr<CSSStyleSheet> >&);
|
|
void resetAuthorStyle(TreeScope&);
|
|
void finishAppendAuthorStyleSheets();
|
|
|
|
void processScopedRules(const RuleSet& authorRules, CSSStyleSheet*, unsigned sheetIndex, ContainerNode& scope);
|
|
|
|
void lazyAppendAuthorStyleSheets(unsigned firstNew, const Vector<RefPtr<CSSStyleSheet> >&);
|
|
void removePendingAuthorStyleSheets(const Vector<RefPtr<CSSStyleSheet> >&);
|
|
void appendPendingAuthorStyleSheets();
|
|
bool hasPendingAuthorStyleSheets() const { return m_pendingStyleSheets.size() > 0 || m_needCollectFeatures; }
|
|
|
|
void styleTreeResolveScopedKeyframesRules(const Element*, Vector<RawPtr<ScopedStyleResolver>, 8>&);
|
|
|
|
// These methods will give back the set of rules that matched for a given element (or a pseudo-element).
|
|
enum CSSRuleFilter {
|
|
UAAndUserCSSRules = 1 << 1,
|
|
AuthorCSSRules = 1 << 2,
|
|
EmptyCSSRules = 1 << 3,
|
|
CrossOriginCSSRules = 1 << 4,
|
|
AllButEmptyCSSRules = UAAndUserCSSRules | AuthorCSSRules | CrossOriginCSSRules,
|
|
AllCSSRules = AllButEmptyCSSRules | EmptyCSSRules,
|
|
};
|
|
PassRefPtr<CSSRuleList> cssRulesForElement(Element*, unsigned rulesToInclude = AllButEmptyCSSRules);
|
|
PassRefPtr<CSSRuleList> pseudoCSSRulesForElement(Element*, PseudoId, unsigned rulesToInclude = AllButEmptyCSSRules);
|
|
PassRefPtr<StyleRuleList> styleRulesForElement(Element*, unsigned rulesToInclude);
|
|
|
|
// |properties| is an array with |count| elements.
|
|
void applyPropertiesToStyle(const CSSPropertyValue* properties, size_t count, RenderStyle*);
|
|
|
|
void addMediaQueryResults(const MediaQueryResultList&);
|
|
MediaQueryResultList* viewportDependentMediaQueryResults() { return &m_viewportDependentMediaQueryResults; }
|
|
bool hasViewportDependentMediaQueries() const { return !m_viewportDependentMediaQueryResults.isEmpty(); }
|
|
bool mediaQueryAffectedByViewportChange() const;
|
|
|
|
// FIXME: Rename to reflect the purpose, like didChangeFontSize or something.
|
|
void invalidateMatchedPropertiesCache();
|
|
|
|
void notifyResizeForViewportUnits();
|
|
|
|
// Exposed for RenderStyle::isStyleAvilable().
|
|
static RenderStyle* styleNotYetAvailable() { return s_styleNotYetAvailable; }
|
|
|
|
RuleFeatureSet& ensureUpdatedRuleFeatureSet()
|
|
{
|
|
if (hasPendingAuthorStyleSheets())
|
|
appendPendingAuthorStyleSheets();
|
|
return m_features;
|
|
}
|
|
|
|
RuleFeatureSet& ruleFeatureSet()
|
|
{
|
|
return m_features;
|
|
}
|
|
|
|
StyleSharingList& styleSharingList();
|
|
|
|
bool hasRulesForId(const AtomicString&) const;
|
|
|
|
void addToStyleSharingList(Element&);
|
|
void clearStyleSharingList();
|
|
|
|
StyleResolverStats* stats() { return m_styleResolverStats.get(); }
|
|
StyleResolverStats* statsTotals() { return m_styleResolverStatsTotals.get(); }
|
|
enum StatsReportType { ReportDefaultStats, ReportSlowStats };
|
|
void enableStats(StatsReportType = ReportDefaultStats);
|
|
void disableStats();
|
|
void printStats();
|
|
|
|
unsigned accessCount() const { return m_accessCount; }
|
|
void didAccess() { ++m_accessCount; }
|
|
|
|
void increaseStyleSharingDepth() { ++m_styleSharingDepth; }
|
|
void decreaseStyleSharingDepth() { --m_styleSharingDepth; }
|
|
|
|
private:
|
|
// FIXME: This should probably go away, folded into FontBuilder.
|
|
void updateFont(StyleResolverState&);
|
|
|
|
void loadPendingResources(StyleResolverState&);
|
|
|
|
void appendCSSStyleSheet(CSSStyleSheet*);
|
|
|
|
void matchUARules(ElementRuleCollector&, RuleSet*);
|
|
void matchAuthorRules(Element*, ElementRuleCollector&, bool includeEmptyRules);
|
|
void matchAuthorRulesForShadowHost(Element*, ElementRuleCollector&, bool includeEmptyRules, Vector<RawPtr<ScopedStyleResolver>, 8>& resolvers, Vector<RawPtr<ScopedStyleResolver>, 8>& resolversInShadowTree);
|
|
void matchAllRules(StyleResolverState&, ElementRuleCollector&, bool includeSMILProperties);
|
|
void matchUARules(ElementRuleCollector&);
|
|
void collectFeatures();
|
|
void resetRuleFeatures();
|
|
|
|
void applyMatchedProperties(StyleResolverState&, const MatchResult&);
|
|
bool applyAnimatedProperties(StyleResolverState&, Element* animatingElement);
|
|
|
|
void resolveScopedStyles(const Element*, Vector<RawPtr<ScopedStyleResolver>, 8>&);
|
|
void collectScopedResolversForHostedShadowTrees(const Element*, Vector<RawPtr<ScopedStyleResolver>, 8>&);
|
|
|
|
enum StyleApplicationPass {
|
|
HighPriorityProperties,
|
|
LowPriorityProperties
|
|
};
|
|
template <StyleResolver::StyleApplicationPass pass>
|
|
static inline CSSPropertyID firstCSSPropertyId();
|
|
template <StyleResolver::StyleApplicationPass pass>
|
|
static inline CSSPropertyID lastCSSPropertyId();
|
|
template <StyleResolver::StyleApplicationPass pass>
|
|
static inline bool isPropertyForPass(CSSPropertyID);
|
|
template <StyleApplicationPass pass>
|
|
void applyMatchedProperties(StyleResolverState&, const MatchResult&, bool important, int startIndex, int endIndex, bool inheritedOnly);
|
|
template <StyleApplicationPass pass>
|
|
void applyProperties(StyleResolverState&, const StylePropertySet* properties, bool isImportant, bool inheritedOnly);
|
|
template <StyleApplicationPass pass>
|
|
void applyAnimatedProperties(StyleResolverState&, const HashMap<CSSPropertyID, RefPtr<Interpolation> >&);
|
|
template <StyleResolver::StyleApplicationPass pass>
|
|
void applyAllProperty(StyleResolverState&, CSSValue*);
|
|
|
|
// FIXME: This likely belongs on RuleSet.
|
|
typedef HashMap<StringImpl*, RefPtr<StyleRuleKeyframes> > KeyframesRuleMap;
|
|
KeyframesRuleMap m_keyframesRuleMap;
|
|
|
|
static RenderStyle* s_styleNotYetAvailable;
|
|
|
|
MatchedPropertiesCache m_matchedPropertiesCache;
|
|
|
|
OwnPtr<MediaQueryEvaluator> m_medium;
|
|
MediaQueryResultList m_viewportDependentMediaQueryResults;
|
|
|
|
RawPtr<Document> m_document;
|
|
|
|
ListHashSet<RawPtr<CSSStyleSheet>, 16> m_pendingStyleSheets;
|
|
|
|
// FIXME: The entire logic of collecting features on StyleResolver, as well as transferring them
|
|
// between various parts of machinery smells wrong. This needs to be better somehow.
|
|
RuleFeatureSet m_features;
|
|
OwnPtr<RuleSet> m_attributeRuleSet;
|
|
|
|
bool m_needCollectFeatures;
|
|
bool m_printMediaType;
|
|
|
|
StyleResourceLoader m_styleResourceLoader;
|
|
|
|
unsigned m_styleSharingDepth;
|
|
Vector<OwnPtr<StyleSharingList>, styleSharingMaxDepth> m_styleSharingLists;
|
|
|
|
OwnPtr<StyleResolverStats> m_styleResolverStats;
|
|
OwnPtr<StyleResolverStats> m_styleResolverStatsTotals;
|
|
unsigned m_styleResolverStatsSequence;
|
|
|
|
// Use only for Internals::updateStyleAndReturnAffectedElementCount.
|
|
unsigned m_accessCount;
|
|
};
|
|
|
|
} // namespace blink
|
|
|
|
#endif // SKY_ENGINE_CORE_CSS_RESOLVER_STYLERESOLVER_H_
|