2015-12-04 02:34:12 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
|
|
/* 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 "EffectCompositor.h"
|
|
|
|
|
|
|
|
#include "mozilla/dom/Animation.h"
|
2016-01-06 05:04:04 +03:00
|
|
|
#include "mozilla/dom/Element.h"
|
2016-09-04 10:34:21 +03:00
|
|
|
#include "mozilla/dom/KeyframeEffectReadOnly.h"
|
2016-10-05 08:26:25 +03:00
|
|
|
#include "mozilla/AnimationComparator.h"
|
2016-03-04 11:54:25 +03:00
|
|
|
#include "mozilla/AnimationPerformanceWarning.h"
|
2016-04-28 18:22:43 +03:00
|
|
|
#include "mozilla/AnimationTarget.h"
|
|
|
|
#include "mozilla/AnimationUtils.h"
|
2015-12-04 02:34:12 +03:00
|
|
|
#include "mozilla/EffectSet.h"
|
2016-01-06 05:04:04 +03:00
|
|
|
#include "mozilla/LayerAnimationInfo.h"
|
2016-02-24 10:01:12 +03:00
|
|
|
#include "mozilla/RestyleManagerHandle.h"
|
|
|
|
#include "mozilla/RestyleManagerHandleInlines.h"
|
2016-01-06 05:04:05 +03:00
|
|
|
#include "nsComputedDOMStyle.h" // nsComputedDOMStyle::GetPresShellForContent
|
2016-08-17 04:46:58 +03:00
|
|
|
#include "nsCSSPropertyIDSet.h"
|
2016-01-06 05:04:05 +03:00
|
|
|
#include "nsCSSProps.h"
|
2016-01-06 05:04:05 +03:00
|
|
|
#include "nsIPresShell.h"
|
2015-12-04 02:34:12 +03:00
|
|
|
#include "nsLayoutUtils.h"
|
2016-01-06 05:04:04 +03:00
|
|
|
#include "nsRuleNode.h" // For nsRuleNode::ComputePropertiesOverridingAnimation
|
2016-01-15 09:15:47 +03:00
|
|
|
#include "nsRuleProcessorData.h" // For ElementRuleProcessorData etc.
|
2016-01-06 05:04:04 +03:00
|
|
|
#include "nsTArray.h"
|
2016-10-05 08:42:56 +03:00
|
|
|
#include <bitset>
|
2016-10-19 07:33:14 +03:00
|
|
|
#include <initializer_list>
|
2015-12-04 02:34:12 +03:00
|
|
|
|
|
|
|
using mozilla::dom::Animation;
|
2016-01-06 05:04:04 +03:00
|
|
|
using mozilla::dom::Element;
|
2015-12-04 02:34:12 +03:00
|
|
|
using mozilla::dom::KeyframeEffectReadOnly;
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
|
2016-01-13 01:54:53 +03:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(EffectCompositor)
|
|
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(EffectCompositor)
|
|
|
|
for (auto& elementSet : tmp->mElementsToRestyle) {
|
|
|
|
elementSet.Clear();
|
|
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(EffectCompositor)
|
|
|
|
for (auto& elementSet : tmp->mElementsToRestyle) {
|
|
|
|
for (auto iter = elementSet.Iter(); !iter.Done(); iter.Next()) {
|
|
|
|
CycleCollectionNoteChild(cb, iter.Key().mElement,
|
|
|
|
"EffectCompositor::mElementsToRestyle[]",
|
|
|
|
cb.Flags());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
2016-01-13 01:54:53 +03:00
|
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(EffectCompositor, AddRef)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(EffectCompositor, Release)
|
|
|
|
|
2015-12-10 00:28:10 +03:00
|
|
|
// Helper function to factor out the common logic from
|
|
|
|
// GetAnimationsForCompositor and HasAnimationsForCompositor.
|
|
|
|
//
|
|
|
|
// Takes an optional array to fill with eligible animations.
|
|
|
|
//
|
|
|
|
// Returns true if there are eligible animations, false otherwise.
|
|
|
|
bool
|
|
|
|
FindAnimationsForCompositor(const nsIFrame* aFrame,
|
2016-08-17 04:37:48 +03:00
|
|
|
nsCSSPropertyID aProperty,
|
2015-12-10 00:28:10 +03:00
|
|
|
nsTArray<RefPtr<dom::Animation>>* aMatches /*out*/)
|
2015-12-04 02:34:12 +03:00
|
|
|
{
|
2015-12-10 00:28:10 +03:00
|
|
|
MOZ_ASSERT(!aMatches || aMatches->IsEmpty(),
|
|
|
|
"Matches array, if provided, should be empty");
|
|
|
|
|
|
|
|
EffectSet* effects = EffectSet::GetEffectSet(aFrame);
|
|
|
|
if (!effects || effects->IsEmpty()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-10-05 08:53:58 +03:00
|
|
|
// If the property will be added to the animations level of the cascade but
|
|
|
|
// there is an !important rule for that property in the cascade then the
|
|
|
|
// animation will not be applied since the !important rule overrides it.
|
|
|
|
if (effects->PropertiesWithImportantRules().HasProperty(aProperty) &&
|
|
|
|
effects->PropertiesForAnimationsLevel().HasProperty(aProperty)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-12-10 00:28:10 +03:00
|
|
|
if (aFrame->RefusedAsyncAnimation()) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-12-04 02:34:12 +03:00
|
|
|
|
2016-01-06 05:04:05 +03:00
|
|
|
// The animation cascade will almost always be up-to-date by this point
|
|
|
|
// but there are some cases such as when we are restoring the refresh driver
|
|
|
|
// from test control after seeking where it might not be the case.
|
|
|
|
//
|
|
|
|
// Those cases are probably not important but just to be safe, let's make
|
|
|
|
// sure the cascade is up to date since if it *is* up to date, this is
|
|
|
|
// basically a no-op.
|
2016-03-21 11:49:50 +03:00
|
|
|
Maybe<NonOwningAnimationTarget> pseudoElement =
|
2016-01-06 05:04:05 +03:00
|
|
|
EffectCompositor::GetAnimationElementAndPseudoForFrame(aFrame);
|
|
|
|
if (pseudoElement) {
|
2016-03-21 11:49:50 +03:00
|
|
|
EffectCompositor::MaybeUpdateCascadeResults(pseudoElement->mElement,
|
|
|
|
pseudoElement->mPseudoType,
|
2016-01-06 05:04:05 +03:00
|
|
|
aFrame->StyleContext());
|
|
|
|
}
|
|
|
|
|
2015-12-04 02:34:12 +03:00
|
|
|
if (!nsLayoutUtils::AreAsyncAnimationsEnabled()) {
|
|
|
|
if (nsLayoutUtils::IsAnimationLoggingEnabled()) {
|
|
|
|
nsCString message;
|
|
|
|
message.AppendLiteral("Performance warning: Async animations are "
|
|
|
|
"disabled");
|
|
|
|
AnimationUtils::LogAsyncAnimationFailure(message);
|
|
|
|
}
|
2015-12-10 00:28:10 +03:00
|
|
|
return false;
|
2015-12-04 02:34:12 +03:00
|
|
|
}
|
|
|
|
|
2016-07-29 08:58:32 +03:00
|
|
|
// Disable async animations if we have a rendering observer that
|
|
|
|
// depends on our content (svg masking, -moz-element etc) so that
|
|
|
|
// it gets updated correctly.
|
|
|
|
nsIContent* content = aFrame->GetContent();
|
|
|
|
while (content) {
|
|
|
|
if (content->HasRenderingObservers()) {
|
|
|
|
EffectCompositor::SetPerformanceWarning(
|
|
|
|
aFrame, aProperty,
|
|
|
|
AnimationPerformanceWarning(
|
|
|
|
AnimationPerformanceWarning::Type::HasRenderingObserver));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
content = content->GetParent();
|
|
|
|
}
|
|
|
|
|
2015-12-10 00:28:10 +03:00
|
|
|
bool foundSome = false;
|
2015-12-04 02:34:12 +03:00
|
|
|
for (KeyframeEffectReadOnly* effect : *effects) {
|
|
|
|
MOZ_ASSERT(effect && effect->GetAnimation());
|
|
|
|
Animation* animation = effect->GetAnimation();
|
|
|
|
|
2016-10-14 13:14:12 +03:00
|
|
|
if (!animation->IsPlayableOnCompositor()) {
|
2015-12-04 02:34:12 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-03-04 11:54:25 +03:00
|
|
|
AnimationPerformanceWarning::Type warningType;
|
2016-03-14 03:07:48 +03:00
|
|
|
if (aProperty == eCSSProperty_transform &&
|
|
|
|
effect->ShouldBlockAsyncTransformAnimations(aFrame,
|
|
|
|
warningType)) {
|
2015-12-10 00:28:10 +03:00
|
|
|
if (aMatches) {
|
|
|
|
aMatches->Clear();
|
|
|
|
}
|
2016-03-14 03:07:48 +03:00
|
|
|
EffectCompositor::SetPerformanceWarning(
|
|
|
|
aFrame, aProperty,
|
|
|
|
AnimationPerformanceWarning(warningType));
|
2015-12-10 00:28:10 +03:00
|
|
|
return false;
|
2015-12-04 02:34:12 +03:00
|
|
|
}
|
|
|
|
|
2016-10-13 10:54:25 +03:00
|
|
|
if (!effect->HasEffectiveAnimationOfProperty(aProperty)) {
|
2015-12-04 02:34:12 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-12-10 00:28:10 +03:00
|
|
|
if (aMatches) {
|
|
|
|
aMatches->AppendElement(animation);
|
|
|
|
}
|
|
|
|
foundSome = true;
|
2015-12-04 02:34:12 +03:00
|
|
|
}
|
|
|
|
|
2015-12-10 00:28:10 +03:00
|
|
|
MOZ_ASSERT(!foundSome || !aMatches || !aMatches->IsEmpty(),
|
|
|
|
"If return value is true, matches array should be non-empty");
|
2016-10-05 08:26:25 +03:00
|
|
|
|
|
|
|
if (aMatches && foundSome) {
|
|
|
|
aMatches->Sort(AnimationPtrComparator<RefPtr<dom::Animation>>());
|
|
|
|
}
|
2015-12-10 00:28:10 +03:00
|
|
|
return foundSome;
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:53 +03:00
|
|
|
void
|
|
|
|
EffectCompositor::RequestRestyle(dom::Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-01-13 01:54:53 +03:00
|
|
|
RestyleType aRestyleType,
|
|
|
|
CascadeLevel aCascadeLevel)
|
|
|
|
{
|
2016-01-13 01:54:54 +03:00
|
|
|
if (!mPresContext) {
|
|
|
|
// Pres context will be null after the effect compositor is disconnected.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-10-12 09:58:53 +03:00
|
|
|
// Ignore animations on orphaned elements.
|
|
|
|
if (!aElement->IsInComposedDoc()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:53 +03:00
|
|
|
auto& elementsToRestyle = mElementsToRestyle[aCascadeLevel];
|
2016-03-21 11:49:50 +03:00
|
|
|
PseudoElementHashEntry::KeyType key = { aElement, aPseudoType };
|
2016-01-13 01:54:53 +03:00
|
|
|
|
2016-01-18 22:54:00 +03:00
|
|
|
if (aRestyleType == RestyleType::Throttled) {
|
|
|
|
if (!elementsToRestyle.Contains(key)) {
|
|
|
|
elementsToRestyle.Put(key, false);
|
|
|
|
}
|
2016-01-13 01:54:54 +03:00
|
|
|
mPresContext->Document()->SetNeedStyleFlush();
|
2016-01-13 01:54:53 +03:00
|
|
|
} else {
|
2016-01-13 01:54:54 +03:00
|
|
|
// Get() returns 0 if the element is not found. It will also return
|
|
|
|
// false if the element is found but does not have a pending restyle.
|
|
|
|
bool hasPendingRestyle = elementsToRestyle.Get(key);
|
|
|
|
if (!hasPendingRestyle) {
|
|
|
|
PostRestyleForAnimation(aElement, aPseudoType, aCascadeLevel);
|
|
|
|
}
|
2016-01-13 01:54:53 +03:00
|
|
|
elementsToRestyle.Put(key, true);
|
|
|
|
}
|
2016-01-13 01:54:54 +03:00
|
|
|
|
|
|
|
if (aRestyleType == RestyleType::Layer) {
|
|
|
|
// Prompt layers to re-sync their animations.
|
2016-02-24 10:01:12 +03:00
|
|
|
MOZ_ASSERT(mPresContext->RestyleManager()->IsGecko(),
|
|
|
|
"stylo: Servo-backed style system should not be using "
|
|
|
|
"EffectCompositor");
|
|
|
|
mPresContext->RestyleManager()->AsGecko()->IncrementAnimationGeneration();
|
2016-01-13 01:54:54 +03:00
|
|
|
EffectSet* effectSet =
|
|
|
|
EffectSet::GetEffectSet(aElement, aPseudoType);
|
|
|
|
if (effectSet) {
|
|
|
|
effectSet->UpdateAnimationGeneration(mPresContext);
|
|
|
|
}
|
|
|
|
}
|
2016-01-13 01:54:53 +03:00
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:54 +03:00
|
|
|
void
|
|
|
|
EffectCompositor::PostRestyleForAnimation(dom::Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-01-13 01:54:54 +03:00
|
|
|
CascadeLevel aCascadeLevel)
|
|
|
|
{
|
|
|
|
if (!mPresContext) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dom::Element* element = GetElementToRestyle(aElement, aPseudoType);
|
|
|
|
if (!element) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsRestyleHint hint = aCascadeLevel == CascadeLevel::Transitions ?
|
|
|
|
eRestyle_CSSTransitions :
|
|
|
|
eRestyle_CSSAnimations;
|
|
|
|
mPresContext->PresShell()->RestyleForAnimation(element, hint);
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:55 +03:00
|
|
|
void
|
|
|
|
EffectCompositor::PostRestyleForThrottledAnimations()
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < kCascadeLevelCount; i++) {
|
|
|
|
CascadeLevel cascadeLevel = CascadeLevel(i);
|
|
|
|
auto& elementSet = mElementsToRestyle[cascadeLevel];
|
|
|
|
|
|
|
|
for (auto iter = elementSet.Iter(); !iter.Done(); iter.Next()) {
|
|
|
|
bool& postedRestyle = iter.Data();
|
|
|
|
if (postedRestyle) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
PostRestyleForAnimation(iter.Key().mElement,
|
|
|
|
iter.Key().mPseudoType,
|
|
|
|
cascadeLevel);
|
|
|
|
postedRestyle = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-01 03:31:51 +03:00
|
|
|
void
|
|
|
|
EffectCompositor::UpdateEffectProperties(nsStyleContext* aStyleContext,
|
|
|
|
dom::Element* aElement,
|
|
|
|
CSSPseudoElementType aPseudoType)
|
|
|
|
{
|
|
|
|
EffectSet* effectSet = EffectSet::GetEffectSet(aElement, aPseudoType);
|
|
|
|
if (!effectSet) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-07-11 11:28:15 +03:00
|
|
|
// Style context change might cause CSS cascade level,
|
|
|
|
// e.g removing !important, so we should update the cascading result.
|
|
|
|
effectSet->MarkCascadeNeedsUpdate();
|
|
|
|
|
2016-04-01 03:31:51 +03:00
|
|
|
for (KeyframeEffectReadOnly* effect : *effectSet) {
|
|
|
|
effect->UpdateProperties(aStyleContext);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:53 +03:00
|
|
|
void
|
|
|
|
EffectCompositor::MaybeUpdateAnimationRule(dom::Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-07-11 11:28:14 +03:00
|
|
|
CascadeLevel aCascadeLevel,
|
|
|
|
nsStyleContext* aStyleContext)
|
2016-01-13 01:54:53 +03:00
|
|
|
{
|
2016-01-13 01:54:55 +03:00
|
|
|
// First update cascade results since that may cause some elements to
|
|
|
|
// be marked as needing a restyle.
|
2016-07-11 11:28:14 +03:00
|
|
|
MaybeUpdateCascadeResults(aElement, aPseudoType, aStyleContext);
|
2016-01-13 01:54:55 +03:00
|
|
|
|
2016-01-13 01:54:53 +03:00
|
|
|
auto& elementsToRestyle = mElementsToRestyle[aCascadeLevel];
|
2016-03-21 11:49:50 +03:00
|
|
|
PseudoElementHashEntry::KeyType key = { aElement, aPseudoType };
|
2016-01-13 01:54:53 +03:00
|
|
|
|
2016-01-13 01:54:54 +03:00
|
|
|
if (!mPresContext || !elementsToRestyle.Contains(key)) {
|
2016-01-13 01:54:53 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:54 +03:00
|
|
|
ComposeAnimationRule(aElement, aPseudoType, aCascadeLevel,
|
2016-01-13 01:54:54 +03:00
|
|
|
mPresContext->RefreshDriver()->MostRecentRefresh());
|
2016-01-13 01:54:53 +03:00
|
|
|
|
|
|
|
elementsToRestyle.Remove(key);
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:55 +03:00
|
|
|
nsIStyleRule*
|
|
|
|
EffectCompositor::GetAnimationRule(dom::Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-07-11 11:28:14 +03:00
|
|
|
CascadeLevel aCascadeLevel,
|
|
|
|
nsStyleContext* aStyleContext)
|
2016-01-13 01:54:55 +03:00
|
|
|
{
|
2016-01-15 09:15:47 +03:00
|
|
|
// NOTE: We need to be careful about early returns in this method where
|
|
|
|
// we *don't* update mElementsToRestyle. When we get a call to
|
|
|
|
// RequestRestyle that results in a call to PostRestyleForAnimation, we
|
|
|
|
// will set a bool flag in mElementsToRestyle indicating that we've
|
|
|
|
// called PostRestyleForAnimation so we don't need to call it again
|
|
|
|
// until that restyle happens. During that restyle, if we arrive here
|
|
|
|
// and *don't* update mElementsToRestyle we'll continue to skip calling
|
|
|
|
// PostRestyleForAnimation from RequestRestyle.
|
|
|
|
|
2016-01-13 01:54:55 +03:00
|
|
|
if (!mPresContext || !mPresContext->IsDynamic()) {
|
|
|
|
// For print or print preview, ignore animations.
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-02-24 10:01:12 +03:00
|
|
|
MOZ_ASSERT(mPresContext->RestyleManager()->IsGecko(),
|
|
|
|
"stylo: Servo-backed style system should not be using "
|
|
|
|
"EffectCompositor");
|
|
|
|
if (mPresContext->RestyleManager()->AsGecko()->SkipAnimationRules()) {
|
2016-01-15 09:15:47 +03:00
|
|
|
// We don't need to worry about updating mElementsToRestyle in this case
|
|
|
|
// since this is not the animation restyle we requested when we called
|
|
|
|
// PostRestyleForAnimation (see comment at start of this method).
|
2016-01-13 01:54:55 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-07-11 11:28:14 +03:00
|
|
|
MaybeUpdateAnimationRule(aElement, aPseudoType, aCascadeLevel, aStyleContext);
|
2016-01-13 01:54:55 +03:00
|
|
|
|
2016-01-15 09:15:47 +03:00
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
auto& elementsToRestyle = mElementsToRestyle[aCascadeLevel];
|
2016-03-21 11:49:50 +03:00
|
|
|
PseudoElementHashEntry::KeyType key = { aElement, aPseudoType };
|
2016-01-15 09:15:47 +03:00
|
|
|
MOZ_ASSERT(!elementsToRestyle.Contains(key),
|
|
|
|
"Element should no longer require a restyle after its "
|
|
|
|
"animation rule has been updated");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
EffectSet* effectSet = EffectSet::GetEffectSet(aElement, aPseudoType);
|
|
|
|
if (!effectSet) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:55 +03:00
|
|
|
return effectSet->AnimationRule(aCascadeLevel);
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:54 +03:00
|
|
|
/* static */ dom::Element*
|
|
|
|
EffectCompositor::GetElementToRestyle(dom::Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType)
|
2016-01-13 01:54:54 +03:00
|
|
|
{
|
2016-02-17 01:07:00 +03:00
|
|
|
if (aPseudoType == CSSPseudoElementType::NotPseudo) {
|
2016-01-13 01:54:54 +03:00
|
|
|
return aElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame* primaryFrame = aElement->GetPrimaryFrame();
|
|
|
|
if (!primaryFrame) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
nsIFrame* pseudoFrame;
|
2016-02-17 01:07:00 +03:00
|
|
|
if (aPseudoType == CSSPseudoElementType::before) {
|
2016-01-13 01:54:54 +03:00
|
|
|
pseudoFrame = nsLayoutUtils::GetBeforeFrame(primaryFrame);
|
2016-02-17 01:07:00 +03:00
|
|
|
} else if (aPseudoType == CSSPseudoElementType::after) {
|
2016-01-13 01:54:54 +03:00
|
|
|
pseudoFrame = nsLayoutUtils::GetAfterFrame(primaryFrame);
|
|
|
|
} else {
|
|
|
|
NS_NOTREACHED("Should not try to get the element to restyle for a pseudo "
|
|
|
|
"other that :before or :after");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (!pseudoFrame) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return pseudoFrame->GetContent()->AsElement();
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:55 +03:00
|
|
|
bool
|
|
|
|
EffectCompositor::HasPendingStyleUpdates() const
|
|
|
|
{
|
|
|
|
for (auto& elementSet : mElementsToRestyle) {
|
|
|
|
if (elementSet.Count()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
EffectCompositor::HasThrottledStyleUpdates() const
|
|
|
|
{
|
|
|
|
for (auto& elementSet : mElementsToRestyle) {
|
|
|
|
for (auto iter = elementSet.ConstIter(); !iter.Done(); iter.Next()) {
|
|
|
|
if (!iter.Data()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:55 +03:00
|
|
|
void
|
|
|
|
EffectCompositor::AddStyleUpdatesTo(RestyleTracker& aTracker)
|
|
|
|
{
|
|
|
|
if (!mPresContext) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < kCascadeLevelCount; i++) {
|
|
|
|
CascadeLevel cascadeLevel = CascadeLevel(i);
|
|
|
|
auto& elementSet = mElementsToRestyle[cascadeLevel];
|
|
|
|
|
|
|
|
// Copy the list of elements to restyle to a separate array that we can
|
|
|
|
// iterate over. This is because we need to call MaybeUpdateCascadeResults
|
|
|
|
// on each element, but doing that can mutate elementSet. In this case
|
|
|
|
// it will only mutate the bool value associated with each element in the
|
|
|
|
// set but even doing that will cause assertions in PLDHashTable to fail
|
|
|
|
// if we are iterating over the hashtable at the same time.
|
2016-03-21 11:49:50 +03:00
|
|
|
nsTArray<PseudoElementHashEntry::KeyType> elementsToRestyle(
|
|
|
|
elementSet.Count());
|
2016-01-13 01:54:55 +03:00
|
|
|
for (auto iter = elementSet.Iter(); !iter.Done(); iter.Next()) {
|
2016-10-12 09:58:53 +03:00
|
|
|
// Skip animations on elements that have been orphaned since they
|
|
|
|
// requested a restyle.
|
|
|
|
if (iter.Key().mElement->IsInComposedDoc()) {
|
|
|
|
elementsToRestyle.AppendElement(iter.Key());
|
|
|
|
}
|
2016-01-13 01:54:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& pseudoElem : elementsToRestyle) {
|
2016-07-11 11:28:19 +03:00
|
|
|
MaybeUpdateCascadeResults(pseudoElem.mElement,
|
|
|
|
pseudoElem.mPseudoType,
|
|
|
|
nullptr);
|
2016-01-13 01:54:55 +03:00
|
|
|
|
|
|
|
ComposeAnimationRule(pseudoElem.mElement,
|
|
|
|
pseudoElem.mPseudoType,
|
|
|
|
cascadeLevel,
|
|
|
|
mPresContext->RefreshDriver()->MostRecentRefresh());
|
|
|
|
|
|
|
|
dom::Element* elementToRestyle =
|
|
|
|
GetElementToRestyle(pseudoElem.mElement, pseudoElem.mPseudoType);
|
|
|
|
if (elementToRestyle) {
|
|
|
|
nsRestyleHint rshint = cascadeLevel == CascadeLevel::Transitions ?
|
|
|
|
eRestyle_CSSTransitions :
|
|
|
|
eRestyle_CSSAnimations;
|
|
|
|
aTracker.AddPendingRestyle(elementToRestyle, rshint, nsChangeHint(0));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
elementSet.Clear();
|
|
|
|
// Note: mElement pointers in elementsToRestyle might now dangle
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-10 00:28:10 +03:00
|
|
|
/* static */ bool
|
|
|
|
EffectCompositor::HasAnimationsForCompositor(const nsIFrame* aFrame,
|
2016-08-17 04:37:48 +03:00
|
|
|
nsCSSPropertyID aProperty)
|
2015-12-10 00:28:10 +03:00
|
|
|
{
|
|
|
|
return FindAnimationsForCompositor(aFrame, aProperty, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ nsTArray<RefPtr<dom::Animation>>
|
|
|
|
EffectCompositor::GetAnimationsForCompositor(const nsIFrame* aFrame,
|
2016-08-17 04:37:48 +03:00
|
|
|
nsCSSPropertyID aProperty)
|
2015-12-10 00:28:10 +03:00
|
|
|
{
|
|
|
|
nsTArray<RefPtr<dom::Animation>> result;
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
bool foundSome =
|
|
|
|
#endif
|
|
|
|
FindAnimationsForCompositor(aFrame, aProperty, &result);
|
|
|
|
MOZ_ASSERT(!foundSome || !result.IsEmpty(),
|
|
|
|
"If return value is true, matches array should be non-empty");
|
|
|
|
|
2015-12-04 02:34:12 +03:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-01-13 01:54:56 +03:00
|
|
|
/* static */ void
|
|
|
|
EffectCompositor::ClearIsRunningOnCompositor(const nsIFrame *aFrame,
|
2016-08-17 04:37:48 +03:00
|
|
|
nsCSSPropertyID aProperty)
|
2016-01-13 01:54:56 +03:00
|
|
|
{
|
|
|
|
EffectSet* effects = EffectSet::GetEffectSet(aFrame);
|
|
|
|
if (!effects) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (KeyframeEffectReadOnly* effect : *effects) {
|
|
|
|
effect->SetIsRunningOnCompositor(aProperty, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-06 05:04:05 +03:00
|
|
|
/* static */ void
|
|
|
|
EffectCompositor::MaybeUpdateCascadeResults(Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-01-06 05:04:05 +03:00
|
|
|
nsStyleContext* aStyleContext)
|
|
|
|
{
|
|
|
|
EffectSet* effects = EffectSet::GetEffectSet(aElement, aPseudoType);
|
|
|
|
if (!effects || !effects->CascadeNeedsUpdate()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-07-11 11:28:14 +03:00
|
|
|
nsStyleContext* styleContext = aStyleContext;
|
|
|
|
if (!styleContext) {
|
|
|
|
dom::Element* elementToRestyle = GetElementToRestyle(aElement, aPseudoType);
|
|
|
|
if (elementToRestyle) {
|
|
|
|
nsIFrame* frame = elementToRestyle->GetPrimaryFrame();
|
|
|
|
if (frame) {
|
|
|
|
styleContext = frame->StyleContext();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
UpdateCascadeResults(*effects, aElement, aPseudoType, styleContext);
|
2016-01-06 05:04:05 +03:00
|
|
|
|
|
|
|
MOZ_ASSERT(!effects->CascadeNeedsUpdate(), "Failed to update cascade state");
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class EffectCompositeOrderComparator {
|
|
|
|
public:
|
|
|
|
bool Equals(const KeyframeEffectReadOnly* a,
|
|
|
|
const KeyframeEffectReadOnly* b) const
|
|
|
|
{
|
|
|
|
return a == b;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LessThan(const KeyframeEffectReadOnly* a,
|
|
|
|
const KeyframeEffectReadOnly* b) const
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(a->GetAnimation() && b->GetAnimation());
|
|
|
|
MOZ_ASSERT(
|
|
|
|
Equals(a, b) ||
|
|
|
|
a->GetAnimation()->HasLowerCompositeOrderThan(*b->GetAnimation()) !=
|
|
|
|
b->GetAnimation()->HasLowerCompositeOrderThan(*a->GetAnimation()));
|
|
|
|
return a->GetAnimation()->HasLowerCompositeOrderThan(*b->GetAnimation());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ void
|
|
|
|
EffectCompositor::UpdateCascadeResults(Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-01-06 05:04:05 +03:00
|
|
|
nsStyleContext* aStyleContext)
|
|
|
|
{
|
|
|
|
EffectSet* effects = EffectSet::GetEffectSet(aElement, aPseudoType);
|
|
|
|
if (!effects) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
UpdateCascadeResults(*effects, aElement, aPseudoType, aStyleContext);
|
|
|
|
}
|
|
|
|
|
2016-03-21 11:49:50 +03:00
|
|
|
/* static */ Maybe<NonOwningAnimationTarget>
|
2016-01-06 05:04:04 +03:00
|
|
|
EffectCompositor::GetAnimationElementAndPseudoForFrame(const nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
// Always return the same object to benefit from return-value optimization.
|
2016-03-21 11:49:50 +03:00
|
|
|
Maybe<NonOwningAnimationTarget> result;
|
2016-01-06 05:04:04 +03:00
|
|
|
|
2016-06-01 10:24:34 +03:00
|
|
|
CSSPseudoElementType pseudoType =
|
|
|
|
aFrame->StyleContext()->GetPseudoType();
|
|
|
|
|
|
|
|
if (pseudoType != CSSPseudoElementType::NotPseudo &&
|
|
|
|
pseudoType != CSSPseudoElementType::before &&
|
|
|
|
pseudoType != CSSPseudoElementType::after) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-01-06 05:04:04 +03:00
|
|
|
nsIContent* content = aFrame->GetContent();
|
|
|
|
if (!content) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-06-01 10:24:34 +03:00
|
|
|
if (pseudoType == CSSPseudoElementType::before ||
|
|
|
|
pseudoType == CSSPseudoElementType::after) {
|
2016-01-06 05:04:04 +03:00
|
|
|
content = content->GetParent();
|
|
|
|
if (!content) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!content->IsElement()) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-03-21 11:49:50 +03:00
|
|
|
result.emplace(content->AsElement(), pseudoType);
|
2016-01-06 05:04:04 +03:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-01-06 05:04:06 +03:00
|
|
|
/* static */ void
|
|
|
|
EffectCompositor::ComposeAnimationRule(dom::Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-01-06 05:04:06 +03:00
|
|
|
CascadeLevel aCascadeLevel,
|
2016-01-13 01:54:54 +03:00
|
|
|
TimeStamp aRefreshTime)
|
2016-01-06 05:04:06 +03:00
|
|
|
{
|
|
|
|
EffectSet* effects = EffectSet::GetEffectSet(aElement, aPseudoType);
|
|
|
|
if (!effects) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The caller is responsible for calling MaybeUpdateCascadeResults first.
|
|
|
|
MOZ_ASSERT(!effects->CascadeNeedsUpdate(),
|
|
|
|
"Animation cascade out of date when composing animation rule");
|
|
|
|
|
2016-10-05 08:42:56 +03:00
|
|
|
// Get a list of effects sorted by composite order.
|
|
|
|
nsTArray<KeyframeEffectReadOnly*> sortedEffectList(effects->Count());
|
2016-01-06 05:04:06 +03:00
|
|
|
for (KeyframeEffectReadOnly* effect : *effects) {
|
2016-10-05 08:42:56 +03:00
|
|
|
sortedEffectList.AppendElement(effect);
|
2016-01-06 05:04:06 +03:00
|
|
|
}
|
|
|
|
sortedEffectList.Sort(EffectCompositeOrderComparator());
|
|
|
|
|
|
|
|
RefPtr<AnimValuesStyleRule>& animationRule =
|
|
|
|
effects->AnimationRule(aCascadeLevel);
|
|
|
|
animationRule = nullptr;
|
|
|
|
|
2016-10-05 08:42:56 +03:00
|
|
|
// If multiple animations affect the same property, animations with higher
|
|
|
|
// composite order (priority) override or add or animations with lower
|
|
|
|
// priority except properties in propertiesToSkip.
|
|
|
|
const nsCSSPropertyIDSet& propertiesToSkip =
|
|
|
|
aCascadeLevel == CascadeLevel::Animations
|
|
|
|
? nsCSSPropertyIDSet()
|
|
|
|
: effects->PropertiesForAnimationsLevel();
|
|
|
|
for (KeyframeEffectReadOnly* effect : sortedEffectList) {
|
|
|
|
effect->GetAnimation()->ComposeStyle(animationRule, propertiesToSkip);
|
2016-01-06 05:04:06 +03:00
|
|
|
}
|
2016-01-13 01:54:54 +03:00
|
|
|
|
Bug 1240228 - Don't update an effect's timing when tweaking its animation's hold time; r=heycam
In some circumstances when composing style, we tweak the time of the animation
before telling the effect to compose style. This is to avoid visual flicker in
certain situations where the main thread progress is being synchronized with an
animation running on the compositor.
In the past, effects would store their latest sample time locally so when
tweaking the animation time, we would need to call UpdateEffect() after tweaking
the time, and then again after restoring it as otherwise the style composed by
the effect would not reflect the adjusted time.
Now, however, effect's always query their animation for the time so this is no
longer necessary. Furthermore, the actions triggered by UpdateEffect are not
desirable in this case because they can, amongst other things, cause the
associated EffectSet to be destroyed and recreated.
Specifically, Animation::UpdateEffect() calls
KeyframeEffectReadOnly::NotifyAnimationTimingUpdated() which:
* Calls UpdateTargetRegistration which can trigger EffectSet
destruction/creation which is undesirable in this case because we intend to
restore whatever changes we make to the Animation's state and deleting and
recreating the EffectSet will cause any pointers to it to dangle.
* Cause us to possibly reset the "is running on compositor" status.
This too is undesirable since we intend to restore the state of the
Animation immediately after tweaking the hold time so we don't want to
act as if any state has changed.
* Similarly for marking the cascade as possibly needing an update or
requesting a restyle.
In summary, all the actions performed by NotifyAnimationTimingUpdated are
unnecessary and undesirable in this situation where we are temporarily tweaking
an Animation's current time only to restore it immediately afterwards since the
actions are all involved with recognizing actual changes in state.
2016-01-19 02:05:08 +03:00
|
|
|
MOZ_ASSERT(effects == EffectSet::GetEffectSet(aElement, aPseudoType),
|
|
|
|
"EffectSet should not change while composing style");
|
|
|
|
|
2016-01-13 01:54:54 +03:00
|
|
|
effects->UpdateAnimationRuleRefreshTime(aCascadeLevel, aRefreshTime);
|
2016-01-06 05:04:06 +03:00
|
|
|
}
|
|
|
|
|
2016-01-06 05:04:04 +03:00
|
|
|
/* static */ void
|
|
|
|
EffectCompositor::GetOverriddenProperties(nsStyleContext* aStyleContext,
|
2016-01-06 05:04:05 +03:00
|
|
|
EffectSet& aEffectSet,
|
2016-08-17 04:46:58 +03:00
|
|
|
nsCSSPropertyIDSet&
|
2016-01-06 05:04:04 +03:00
|
|
|
aPropertiesOverridden)
|
|
|
|
{
|
2016-08-17 04:37:48 +03:00
|
|
|
AutoTArray<nsCSSPropertyID, LayerAnimationInfo::kRecords> propertiesToTrack;
|
2016-01-06 05:04:05 +03:00
|
|
|
{
|
2016-08-17 04:46:58 +03:00
|
|
|
nsCSSPropertyIDSet propertiesToTrackAsSet;
|
2016-01-06 05:04:05 +03:00
|
|
|
for (KeyframeEffectReadOnly* effect : aEffectSet) {
|
|
|
|
for (const AnimationProperty& property : effect->Properties()) {
|
|
|
|
if (nsCSSProps::PropHasFlags(property.mProperty,
|
|
|
|
CSS_PROPERTY_CAN_ANIMATE_ON_COMPOSITOR) &&
|
|
|
|
!propertiesToTrackAsSet.HasProperty(property.mProperty)) {
|
|
|
|
propertiesToTrackAsSet.AddProperty(property.mProperty);
|
|
|
|
propertiesToTrack.AppendElement(property.mProperty);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Skip iterating over the rest of the effects if we've already
|
|
|
|
// found all the compositor-animatable properties.
|
|
|
|
if (propertiesToTrack.Length() == LayerAnimationInfo::kRecords) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-01-06 05:04:04 +03:00
|
|
|
}
|
2016-01-06 05:04:05 +03:00
|
|
|
|
|
|
|
if (propertiesToTrack.IsEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-01-06 05:04:04 +03:00
|
|
|
nsRuleNode::ComputePropertiesOverridingAnimation(propertiesToTrack,
|
|
|
|
aStyleContext,
|
|
|
|
aPropertiesOverridden);
|
|
|
|
}
|
|
|
|
|
2016-01-06 05:04:05 +03:00
|
|
|
/* static */ void
|
|
|
|
EffectCompositor::UpdateCascadeResults(EffectSet& aEffectSet,
|
|
|
|
Element* aElement,
|
2016-02-17 23:37:00 +03:00
|
|
|
CSSPseudoElementType aPseudoType,
|
2016-01-06 05:04:05 +03:00
|
|
|
nsStyleContext* aStyleContext)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(EffectSet::GetEffectSet(aElement, aPseudoType) == &aEffectSet,
|
|
|
|
"Effect set should correspond to the specified (pseudo-)element");
|
|
|
|
if (aEffectSet.IsEmpty()) {
|
|
|
|
aEffectSet.MarkCascadeUpdated();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a list of effects sorted by composite order.
|
2016-10-05 08:42:56 +03:00
|
|
|
nsTArray<KeyframeEffectReadOnly*> sortedEffectList(aEffectSet.Count());
|
2016-01-06 05:04:05 +03:00
|
|
|
for (KeyframeEffectReadOnly* effect : aEffectSet) {
|
|
|
|
sortedEffectList.AppendElement(effect);
|
|
|
|
}
|
|
|
|
sortedEffectList.Sort(EffectCompositeOrderComparator());
|
|
|
|
|
|
|
|
// Get properties that override the *animations* level of the cascade.
|
|
|
|
//
|
|
|
|
// We only do this for properties that we can animate on the compositor
|
|
|
|
// since we will apply other properties on the main thread where the usual
|
|
|
|
// cascade applies.
|
2016-08-17 04:46:58 +03:00
|
|
|
nsCSSPropertyIDSet overriddenProperties;
|
2016-01-06 05:04:05 +03:00
|
|
|
if (aStyleContext) {
|
2016-01-06 05:04:05 +03:00
|
|
|
GetOverriddenProperties(aStyleContext, aEffectSet, overriddenProperties);
|
2016-01-06 05:04:05 +03:00
|
|
|
}
|
|
|
|
|
2016-10-05 08:42:56 +03:00
|
|
|
// Returns a bitset the represents which properties from
|
|
|
|
// LayerAnimationInfo::sRecords are present in |aPropertySet|.
|
|
|
|
auto compositorPropertiesInSet =
|
|
|
|
[](nsCSSPropertyIDSet& aPropertySet) ->
|
|
|
|
std::bitset<LayerAnimationInfo::kRecords> {
|
|
|
|
std::bitset<LayerAnimationInfo::kRecords> result;
|
|
|
|
for (size_t i = 0; i < LayerAnimationInfo::kRecords; i++) {
|
|
|
|
if (aPropertySet.HasProperty(
|
|
|
|
LayerAnimationInfo::sRecords[i].mProperty)) {
|
|
|
|
result.set(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
|
|
|
|
nsCSSPropertyIDSet& propertiesWithImportantRules =
|
|
|
|
aEffectSet.PropertiesWithImportantRules();
|
|
|
|
nsCSSPropertyIDSet& propertiesForAnimationsLevel =
|
|
|
|
aEffectSet.PropertiesForAnimationsLevel();
|
|
|
|
|
|
|
|
// Record which compositor-animatable properties were originally set so we can
|
|
|
|
// compare for changes later.
|
|
|
|
std::bitset<LayerAnimationInfo::kRecords>
|
|
|
|
prevCompositorPropertiesWithImportantRules =
|
|
|
|
compositorPropertiesInSet(propertiesWithImportantRules);
|
|
|
|
std::bitset<LayerAnimationInfo::kRecords>
|
|
|
|
prevCompositorPropertiesForAnimationsLevel =
|
|
|
|
compositorPropertiesInSet(propertiesForAnimationsLevel);
|
|
|
|
|
|
|
|
propertiesWithImportantRules.Empty();
|
|
|
|
propertiesForAnimationsLevel.Empty();
|
|
|
|
|
|
|
|
bool hasCompositorPropertiesForTransition = false;
|
2016-01-06 05:04:05 +03:00
|
|
|
|
2016-10-05 08:48:05 +03:00
|
|
|
for (const KeyframeEffectReadOnly* effect : sortedEffectList) {
|
2016-01-06 05:04:05 +03:00
|
|
|
MOZ_ASSERT(effect->GetAnimation(),
|
|
|
|
"Effects on a target element should have an Animation");
|
2016-10-05 08:42:56 +03:00
|
|
|
CascadeLevel cascadeLevel = effect->GetAnimation()->CascadeLevel();
|
|
|
|
|
2016-10-05 08:48:05 +03:00
|
|
|
for (const AnimationProperty& prop : effect->Properties()) {
|
2016-10-05 08:42:56 +03:00
|
|
|
if (overriddenProperties.HasProperty(prop.mProperty)) {
|
|
|
|
propertiesWithImportantRules.AddProperty(prop.mProperty);
|
|
|
|
}
|
|
|
|
if (cascadeLevel == EffectCompositor::CascadeLevel::Animations) {
|
|
|
|
propertiesForAnimationsLevel.AddProperty(prop.mProperty);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nsCSSProps::PropHasFlags(prop.mProperty,
|
|
|
|
CSS_PROPERTY_CAN_ANIMATE_ON_COMPOSITOR) &&
|
|
|
|
cascadeLevel == EffectCompositor::CascadeLevel::Transitions) {
|
|
|
|
hasCompositorPropertiesForTransition = true;
|
|
|
|
}
|
2016-01-06 05:04:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
aEffectSet.MarkCascadeUpdated();
|
|
|
|
|
|
|
|
nsPresContext* presContext = GetPresContext(aElement);
|
2016-10-05 08:42:56 +03:00
|
|
|
if (!presContext) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If properties for compositor are newly overridden by !important rules, or
|
|
|
|
// released from being overridden by !important rules, we need to update
|
|
|
|
// layers for animations level because it's a trigger to send animations to
|
|
|
|
// the compositor or pull animations back from the compositor.
|
|
|
|
if (prevCompositorPropertiesWithImportantRules !=
|
|
|
|
compositorPropertiesInSet(propertiesWithImportantRules)) {
|
|
|
|
presContext->EffectCompositor()->
|
|
|
|
RequestRestyle(aElement, aPseudoType,
|
|
|
|
EffectCompositor::RestyleType::Layer,
|
|
|
|
EffectCompositor::CascadeLevel::Animations);
|
|
|
|
}
|
|
|
|
// If we have transition properties for compositor and if the same propery
|
|
|
|
// for animations level is newly added or removed, we need to update layers
|
|
|
|
// for transitions level because composite order has been changed now.
|
|
|
|
if (hasCompositorPropertiesForTransition &&
|
|
|
|
prevCompositorPropertiesForAnimationsLevel !=
|
|
|
|
compositorPropertiesInSet(propertiesForAnimationsLevel)) {
|
|
|
|
presContext->EffectCompositor()->
|
|
|
|
RequestRestyle(aElement, aPseudoType,
|
|
|
|
EffectCompositor::RestyleType::Layer,
|
|
|
|
EffectCompositor::CascadeLevel::Transitions);
|
2016-01-06 05:04:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ nsPresContext*
|
|
|
|
EffectCompositor::GetPresContext(Element* aElement)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aElement);
|
|
|
|
nsIPresShell* shell = nsComputedDOMStyle::GetPresShellForContent(aElement);
|
|
|
|
if (!shell) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return shell->GetPresContext();
|
|
|
|
}
|
|
|
|
|
2016-03-04 00:36:36 +03:00
|
|
|
/* static */ void
|
2016-03-04 11:54:25 +03:00
|
|
|
EffectCompositor::SetPerformanceWarning(
|
|
|
|
const nsIFrame *aFrame,
|
2016-08-17 04:37:48 +03:00
|
|
|
nsCSSPropertyID aProperty,
|
2016-03-04 11:54:25 +03:00
|
|
|
const AnimationPerformanceWarning& aWarning)
|
2016-03-04 00:36:36 +03:00
|
|
|
{
|
|
|
|
EffectSet* effects = EffectSet::GetEffectSet(aFrame);
|
|
|
|
if (!effects) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (KeyframeEffectReadOnly* effect : *effects) {
|
2016-03-04 11:54:25 +03:00
|
|
|
effect->SetPerformanceWarning(aProperty, aWarning);
|
2016-03-04 00:36:36 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-15 09:15:47 +03:00
|
|
|
// ---------------------------------------------------------
|
|
|
|
//
|
|
|
|
// Nested class: AnimationStyleRuleProcessor
|
|
|
|
//
|
|
|
|
// ---------------------------------------------------------
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS(EffectCompositor::AnimationStyleRuleProcessor,
|
|
|
|
nsIStyleRuleProcessor)
|
|
|
|
|
|
|
|
nsRestyleHint
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::HasStateDependentStyle(
|
|
|
|
StateRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
return nsRestyleHint(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsRestyleHint
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::HasStateDependentStyle(
|
|
|
|
PseudoElementStateRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
return nsRestyleHint(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::HasDocumentStateDependentStyle(
|
|
|
|
StateRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsRestyleHint
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::HasAttributeDependentStyle(
|
|
|
|
AttributeRuleProcessorData* aData,
|
|
|
|
RestyleHintData& aRestyleHintDataResult)
|
|
|
|
{
|
|
|
|
return nsRestyleHint(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::MediumFeaturesChanged(
|
|
|
|
nsPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::RulesMatching(
|
|
|
|
ElementRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
nsIStyleRule *rule =
|
|
|
|
mCompositor->GetAnimationRule(aData->mElement,
|
2016-02-17 01:07:00 +03:00
|
|
|
CSSPseudoElementType::NotPseudo,
|
2016-07-11 11:28:14 +03:00
|
|
|
mCascadeLevel,
|
|
|
|
nullptr);
|
2016-01-15 09:15:47 +03:00
|
|
|
if (rule) {
|
|
|
|
aData->mRuleWalker->Forward(rule);
|
|
|
|
aData->mRuleWalker->CurrentNode()->SetIsAnimationRule();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::RulesMatching(
|
|
|
|
PseudoElementRuleProcessorData* aData)
|
|
|
|
{
|
2016-02-17 01:07:00 +03:00
|
|
|
if (aData->mPseudoType != CSSPseudoElementType::before &&
|
|
|
|
aData->mPseudoType != CSSPseudoElementType::after) {
|
2016-01-15 09:15:47 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIStyleRule *rule =
|
|
|
|
mCompositor->GetAnimationRule(aData->mElement,
|
|
|
|
aData->mPseudoType,
|
2016-07-11 11:28:14 +03:00
|
|
|
mCascadeLevel,
|
|
|
|
nullptr);
|
2016-01-15 09:15:47 +03:00
|
|
|
if (rule) {
|
|
|
|
aData->mRuleWalker->Forward(rule);
|
|
|
|
aData->mRuleWalker->CurrentNode()->SetIsAnimationRule();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::RulesMatching(
|
|
|
|
AnonBoxRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MOZ_XUL
|
|
|
|
void
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::RulesMatching(
|
|
|
|
XULTreeRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
size_t
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::SizeOfExcludingThis(
|
|
|
|
MallocSizeOf aMallocSizeOf) const
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
EffectCompositor::AnimationStyleRuleProcessor::SizeOfIncludingThis(
|
|
|
|
MallocSizeOf aMallocSizeOf) const
|
|
|
|
{
|
|
|
|
return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
|
|
|
|
}
|
|
|
|
|
2015-12-04 02:34:12 +03:00
|
|
|
} // namespace mozilla
|