/* -*- 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 "mozilla/dom/KeyframeEffect.h" #include "mozilla/dom/KeyframeAnimationOptionsBinding.h" // For UnrestrictedDoubleOrKeyframeAnimationOptions #include "mozilla/dom/AnimationEffectTiming.h" #include "mozilla/dom/KeyframeEffectBinding.h" #include "nsDocument.h" // For nsDocument::IsWebAnimationsEnabled #include "nsDOMMutationObserver.h" // For nsAutoAnimationMutationBatch #include "nsStyleContext.h" namespace mozilla { namespace dom { KeyframeEffect::KeyframeEffect(nsIDocument* aDocument, const Maybe& aTarget, const TimingParams& aTiming, const KeyframeEffectParams& aOptions) : KeyframeEffectReadOnly(aDocument, aTarget, new AnimationEffectTiming(aDocument, aTiming, this), aOptions) { } JSObject* KeyframeEffect::WrapObject(JSContext* aCx, JS::Handle aGivenProto) { return KeyframeEffectBinding::Wrap(aCx, this, aGivenProto); } /* static */ already_AddRefed KeyframeEffect::Constructor( const GlobalObject& aGlobal, const Nullable& aTarget, JS::Handle aKeyframes, const UnrestrictedDoubleOrKeyframeEffectOptions& aOptions, ErrorResult& aRv) { return ConstructKeyframeEffect(aGlobal, aTarget, aKeyframes, aOptions, aRv); } /* static */ already_AddRefed KeyframeEffect::Constructor(const GlobalObject& aGlobal, KeyframeEffectReadOnly& aSource, ErrorResult& aRv) { return ConstructKeyframeEffect(aGlobal, aSource, aRv); } /* static */ already_AddRefed KeyframeEffect::Constructor( const GlobalObject& aGlobal, const Nullable& aTarget, JS::Handle aKeyframes, const UnrestrictedDoubleOrKeyframeAnimationOptions& aOptions, ErrorResult& aRv) { return ConstructKeyframeEffect(aGlobal, aTarget, aKeyframes, aOptions, aRv); } void KeyframeEffect::NotifySpecifiedTimingUpdated() { // Use the same document for a pseudo element and its parent element. // Use nullptr if we don't have mTarget, so disable the mutation batch. nsAutoAnimationMutationBatch mb(mTarget ? mTarget->mElement->OwnerDoc() : nullptr); if (mAnimation) { mAnimation->NotifyEffectTimingUpdated(); if (mAnimation->IsRelevant()) { nsNodeUtils::AnimationChanged(mAnimation); } RequestRestyle(EffectCompositor::RestyleType::Layer); } } void KeyframeEffect::SetTarget(const Nullable& aTarget) { Maybe newTarget = ConvertTarget(aTarget); if (mTarget == newTarget) { // Assign the same target, skip it. return; } if (mTarget) { UnregisterTarget(); ResetIsRunningOnCompositor(); RequestRestyle(EffectCompositor::RestyleType::Layer); nsAutoAnimationMutationBatch mb(mTarget->mElement->OwnerDoc()); if (mAnimation) { nsNodeUtils::AnimationRemoved(mAnimation); } } mTarget = newTarget; if (mTarget) { UpdateTargetRegistration(); RefPtr styleContext = GetTargetStyleContext(); if (styleContext) { UpdateProperties(styleContext); } MaybeUpdateFrameForCompositor(); RequestRestyle(EffectCompositor::RestyleType::Layer); nsAutoAnimationMutationBatch mb(mTarget->mElement->OwnerDoc()); if (mAnimation) { nsNodeUtils::AnimationAdded(mAnimation); } } // If the new target frame is also oversized we should probably record that // too so we have a more complete picture of the type of frame sizes we // encounter, hence we reset the telemetry flag here. mRecordedContentTooLarge = false; mRecordedFrameSize = false; } void KeyframeEffect::SetIterationComposite( const IterationCompositeOperation& aIterationComposite, CallerType aCallerType) { // Ignore iterationComposite if the Web Animations API is not enabled, // then the default value 'Replace' will be used. if (!nsDocument::IsWebAnimationsEnabled(aCallerType)) { return; } if (mEffectOptions.mIterationComposite == aIterationComposite) { return; } if (mAnimation && mAnimation->IsRelevant()) { nsNodeUtils::AnimationChanged(mAnimation); } mEffectOptions.mIterationComposite = aIterationComposite; RequestRestyle(EffectCompositor::RestyleType::Layer); } void KeyframeEffect::SetComposite(const CompositeOperation& aComposite) { if (mEffectOptions.mComposite == aComposite) { return; } mEffectOptions.mComposite = aComposite; if (mAnimation && mAnimation->IsRelevant()) { nsNodeUtils::AnimationChanged(mAnimation); } if (mTarget) { RefPtr styleContext = GetTargetStyleContext(); if (styleContext) { UpdateProperties(styleContext); } } } } // namespace dom } // namespace mozilla