2016-03-07 08:11:11 +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/. */
|
|
|
|
|
|
|
|
#ifndef mozilla_TimingParams_h
|
|
|
|
#define mozilla_TimingParams_h
|
|
|
|
|
2020-11-23 19:21:38 +03:00
|
|
|
#include "X11UndefineNone.h"
|
2020-06-02 13:31:46 +03:00
|
|
|
#include "nsPrintfCString.h"
|
2016-03-18 10:28:22 +03:00
|
|
|
#include "nsStringFwd.h"
|
2020-05-06 01:39:44 +03:00
|
|
|
#include "nsPrintfCString.h"
|
2016-03-07 08:11:11 +03:00
|
|
|
#include "mozilla/dom/Nullable.h"
|
|
|
|
#include "mozilla/dom/UnionTypes.h" // For OwningUnrestrictedDoubleOrString
|
|
|
|
#include "mozilla/ComputedTimingFunction.h"
|
|
|
|
#include "mozilla/Maybe.h"
|
2016-03-22 10:20:31 +03:00
|
|
|
#include "mozilla/StickyTimeDuration.h"
|
2016-03-07 08:11:11 +03:00
|
|
|
#include "mozilla/TimeStamp.h" // for TimeDuration
|
|
|
|
|
2018-05-07 05:15:16 +03:00
|
|
|
#include "mozilla/dom/AnimationEffectBinding.h" // for FillMode
|
|
|
|
// and PlaybackDirection
|
2016-03-11 11:27:34 +03:00
|
|
|
|
2016-03-07 08:11:11 +03:00
|
|
|
namespace mozilla {
|
|
|
|
|
|
|
|
namespace dom {
|
|
|
|
class UnrestrictedDoubleOrKeyframeEffectOptions;
|
|
|
|
class UnrestrictedDoubleOrKeyframeAnimationOptions;
|
|
|
|
} // namespace dom
|
|
|
|
|
|
|
|
struct TimingParams {
|
2021-12-08 04:16:29 +03:00
|
|
|
constexpr TimingParams() = default;
|
2016-03-07 08:11:11 +03:00
|
|
|
|
2021-12-08 04:16:29 +03:00
|
|
|
constexpr TimingParams(float aDuration, float aDelay, float aIterationCount,
|
|
|
|
dom::PlaybackDirection aDirection,
|
|
|
|
dom::FillMode aFillMode)
|
2017-06-22 06:12:42 +03:00
|
|
|
: mIterations(aIterationCount), mDirection(aDirection), mFill(aFillMode) {
|
|
|
|
mDuration.emplace(StickyTimeDuration::FromMilliseconds(aDuration));
|
|
|
|
mDelay = TimeDuration::FromMilliseconds(aDelay);
|
2017-06-22 06:12:43 +03:00
|
|
|
Update();
|
2017-06-22 06:12:42 +03:00
|
|
|
}
|
|
|
|
|
2021-12-08 04:16:29 +03:00
|
|
|
constexpr TimingParams(const TimeDuration& aDuration,
|
|
|
|
const TimeDuration& aDelay,
|
|
|
|
const TimeDuration& aEndDelay, float aIterations,
|
|
|
|
float aIterationStart,
|
|
|
|
dom::PlaybackDirection aDirection,
|
|
|
|
dom::FillMode aFillMode,
|
|
|
|
Maybe<ComputedTimingFunction>&& aFunction)
|
2017-06-22 06:12:42 +03:00
|
|
|
: mDelay(aDelay),
|
|
|
|
mEndDelay(aEndDelay),
|
|
|
|
mIterations(aIterations),
|
|
|
|
mIterationStart(aIterationStart),
|
|
|
|
mDirection(aDirection),
|
|
|
|
mFill(aFillMode),
|
|
|
|
mFunction(aFunction) {
|
|
|
|
mDuration.emplace(aDuration);
|
2017-06-22 06:12:43 +03:00
|
|
|
Update();
|
2017-06-22 06:12:42 +03:00
|
|
|
}
|
|
|
|
|
2017-06-22 06:12:42 +03:00
|
|
|
template <class OptionsType>
|
|
|
|
static TimingParams FromOptionsType(const OptionsType& aOptions,
|
2019-01-02 16:05:23 +03:00
|
|
|
ErrorResult& aRv);
|
2016-03-07 08:11:11 +03:00
|
|
|
static TimingParams FromOptionsUnion(
|
|
|
|
const dom::UnrestrictedDoubleOrKeyframeEffectOptions& aOptions,
|
2020-11-12 23:34:01 +03:00
|
|
|
ErrorResult& aRv);
|
2016-03-07 08:11:11 +03:00
|
|
|
static TimingParams FromOptionsUnion(
|
|
|
|
const dom::UnrestrictedDoubleOrKeyframeAnimationOptions& aOptions,
|
2020-11-12 23:34:01 +03:00
|
|
|
ErrorResult& aRv);
|
2018-05-07 06:34:47 +03:00
|
|
|
static TimingParams FromEffectTiming(const dom::EffectTiming& aEffectTiming,
|
2018-05-07 06:37:11 +03:00
|
|
|
ErrorResult& aRv);
|
|
|
|
// Returns a copy of |aSource| where each timing property in |aSource| that
|
|
|
|
// is also specified in |aEffectTiming| is replaced with the value from
|
|
|
|
// |aEffectTiming|.
|
|
|
|
//
|
|
|
|
// If any of the values in |aEffectTiming| are invalid, |aRv.Failed()| will be
|
|
|
|
// true and an unmodified copy of |aSource| will be returned.
|
|
|
|
static TimingParams MergeOptionalEffectTiming(
|
|
|
|
const TimingParams& aSource,
|
2020-11-12 23:34:01 +03:00
|
|
|
const dom::OptionalEffectTiming& aEffectTiming, ErrorResult& aRv);
|
2016-03-09 08:01:45 +03:00
|
|
|
|
|
|
|
// Range-checks and validates an UnrestrictedDoubleOrString or
|
|
|
|
// OwningUnrestrictedDoubleOrString object and converts to a
|
|
|
|
// StickyTimeDuration value or Nothing() if aDuration is "auto".
|
|
|
|
// Caller must check aRv.Failed().
|
|
|
|
template <class DoubleOrString>
|
|
|
|
static Maybe<StickyTimeDuration> ParseDuration(DoubleOrString& aDuration,
|
2016-03-11 12:04:39 +03:00
|
|
|
ErrorResult& aRv) {
|
2016-03-09 08:01:45 +03:00
|
|
|
Maybe<StickyTimeDuration> result;
|
|
|
|
if (aDuration.IsUnrestrictedDouble()) {
|
|
|
|
double durationInMs = aDuration.GetAsUnrestrictedDouble();
|
|
|
|
if (durationInMs >= 0) {
|
|
|
|
result.emplace(StickyTimeDuration::FromMilliseconds(durationInMs));
|
2016-03-18 04:07:10 +03:00
|
|
|
} else {
|
2020-03-07 02:05:16 +03:00
|
|
|
nsPrintfCString err("Duration (%g) must be nonnegative", durationInMs);
|
2020-03-07 00:08:02 +03:00
|
|
|
aRv.ThrowTypeError(err);
|
2016-03-09 08:01:45 +03:00
|
|
|
}
|
2016-03-18 04:07:10 +03:00
|
|
|
} else if (!aDuration.GetAsString().EqualsLiteral("auto")) {
|
2016-03-31 13:20:59 +03:00
|
|
|
aRv.ThrowTypeError<dom::MSG_INVALID_DURATION_ERROR>(
|
2020-03-07 00:08:02 +03:00
|
|
|
NS_ConvertUTF16toUTF8(aDuration.GetAsString()));
|
2016-03-09 08:01:45 +03:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2016-03-07 08:11:11 +03:00
|
|
|
|
2016-03-11 12:04:39 +03:00
|
|
|
static void ValidateIterationStart(double aIterationStart, ErrorResult& aRv) {
|
|
|
|
if (aIterationStart < 0) {
|
2020-03-07 02:05:16 +03:00
|
|
|
nsPrintfCString err("Iteration start (%g) must not be negative",
|
|
|
|
aIterationStart);
|
2020-03-07 00:08:02 +03:00
|
|
|
aRv.ThrowTypeError(err);
|
2016-03-11 12:04:39 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-18 04:14:49 +03:00
|
|
|
static void ValidateIterations(double aIterations, ErrorResult& aRv) {
|
2020-03-07 02:05:16 +03:00
|
|
|
if (IsNaN(aIterations)) {
|
2020-03-07 00:04:58 +03:00
|
|
|
aRv.ThrowTypeError("Iterations must not be NaN");
|
2020-03-07 02:05:16 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aIterations < 0) {
|
|
|
|
nsPrintfCString err("Iterations (%g) must not be negative", aIterations);
|
2020-03-07 00:08:02 +03:00
|
|
|
aRv.ThrowTypeError(err);
|
2016-03-18 04:14:49 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-17 17:04:35 +03:00
|
|
|
static Maybe<ComputedTimingFunction> ParseEasing(const nsACString& aEasing,
|
2016-03-18 10:28:22 +03:00
|
|
|
ErrorResult& aRv);
|
|
|
|
|
2017-06-22 06:12:43 +03:00
|
|
|
static StickyTimeDuration CalcActiveDuration(
|
|
|
|
const Maybe<StickyTimeDuration>& aDuration, double aIterations) {
|
2016-04-14 13:39:39 +03:00
|
|
|
// If either the iteration duration or iteration count is zero,
|
|
|
|
// Web Animations says that the active duration is zero. This is to
|
|
|
|
// ensure that the result is defined when the other argument is Infinity.
|
|
|
|
static const StickyTimeDuration zeroDuration;
|
2017-07-10 10:21:30 +03:00
|
|
|
if (!aDuration || aDuration->IsZero() || aIterations == 0.0) {
|
2016-04-14 13:39:39 +03:00
|
|
|
return zeroDuration;
|
|
|
|
}
|
|
|
|
|
2017-06-22 06:12:43 +03:00
|
|
|
return aDuration->MultDouble(aIterations);
|
|
|
|
}
|
|
|
|
// Return the duration of the active interval calculated by duration and
|
|
|
|
// iteration count.
|
|
|
|
StickyTimeDuration ActiveDuration() const {
|
|
|
|
MOZ_ASSERT(CalcActiveDuration(mDuration, mIterations) == mActiveDuration,
|
|
|
|
"Cached value of active duration should be up to date");
|
|
|
|
return mActiveDuration;
|
2016-04-14 13:39:39 +03:00
|
|
|
}
|
|
|
|
|
2016-04-14 13:39:42 +03:00
|
|
|
StickyTimeDuration EndTime() const {
|
2017-06-22 06:12:43 +03:00
|
|
|
MOZ_ASSERT(mEndTime == std::max(mDelay + ActiveDuration() + mEndDelay,
|
|
|
|
StickyTimeDuration()),
|
|
|
|
"Cached value of end time should be up to date");
|
|
|
|
return mEndTime;
|
2016-04-14 13:39:42 +03:00
|
|
|
}
|
|
|
|
|
2016-03-07 08:11:11 +03:00
|
|
|
bool operator==(const TimingParams& aOther) const;
|
|
|
|
bool operator!=(const TimingParams& aOther) const {
|
|
|
|
return !(*this == aOther);
|
|
|
|
}
|
2017-06-22 06:12:43 +03:00
|
|
|
|
|
|
|
void SetDuration(Maybe<StickyTimeDuration>&& aDuration) {
|
2018-05-30 22:15:35 +03:00
|
|
|
mDuration = std::move(aDuration);
|
2017-06-22 06:12:43 +03:00
|
|
|
Update();
|
2017-06-22 06:12:43 +03:00
|
|
|
}
|
2020-03-04 03:36:46 +03:00
|
|
|
void SetDuration(const Maybe<StickyTimeDuration>& aDuration) {
|
|
|
|
mDuration = aDuration;
|
|
|
|
Update();
|
|
|
|
}
|
2017-06-22 06:12:43 +03:00
|
|
|
const Maybe<StickyTimeDuration>& Duration() const { return mDuration; }
|
|
|
|
|
|
|
|
void SetDelay(const TimeDuration& aDelay) {
|
|
|
|
mDelay = aDelay;
|
2017-06-22 06:12:43 +03:00
|
|
|
Update();
|
2017-06-22 06:12:43 +03:00
|
|
|
}
|
|
|
|
const TimeDuration& Delay() const { return mDelay; }
|
|
|
|
|
|
|
|
void SetEndDelay(const TimeDuration& aEndDelay) {
|
|
|
|
mEndDelay = aEndDelay;
|
2017-06-22 06:12:43 +03:00
|
|
|
Update();
|
2017-06-22 06:12:43 +03:00
|
|
|
}
|
|
|
|
const TimeDuration& EndDelay() const { return mEndDelay; }
|
|
|
|
|
|
|
|
void SetIterations(double aIterations) {
|
|
|
|
mIterations = aIterations;
|
2017-06-22 06:12:43 +03:00
|
|
|
Update();
|
2017-06-22 06:12:43 +03:00
|
|
|
}
|
|
|
|
double Iterations() const { return mIterations; }
|
|
|
|
|
|
|
|
void SetIterationStart(double aIterationStart) {
|
|
|
|
mIterationStart = aIterationStart;
|
|
|
|
}
|
|
|
|
double IterationStart() const { return mIterationStart; }
|
|
|
|
|
|
|
|
void SetDirection(dom::PlaybackDirection aDirection) {
|
|
|
|
mDirection = aDirection;
|
|
|
|
}
|
|
|
|
dom::PlaybackDirection Direction() const { return mDirection; }
|
|
|
|
|
|
|
|
void SetFill(dom::FillMode aFill) { mFill = aFill; }
|
|
|
|
dom::FillMode Fill() const { return mFill; }
|
|
|
|
|
2017-06-22 06:12:43 +03:00
|
|
|
void SetTimingFunction(Maybe<ComputedTimingFunction>&& aFunction) {
|
2018-05-30 22:15:35 +03:00
|
|
|
mFunction = std::move(aFunction);
|
2017-06-22 06:12:43 +03:00
|
|
|
}
|
2017-06-22 06:12:43 +03:00
|
|
|
const Maybe<ComputedTimingFunction>& TimingFunction() const {
|
|
|
|
return mFunction;
|
|
|
|
}
|
2017-06-22 06:12:43 +03:00
|
|
|
|
|
|
|
private:
|
2017-06-22 06:12:43 +03:00
|
|
|
void Update() {
|
|
|
|
mActiveDuration = CalcActiveDuration(mDuration, mIterations);
|
|
|
|
|
|
|
|
mEndTime =
|
|
|
|
std::max(mDelay + mActiveDuration + mEndDelay, StickyTimeDuration());
|
|
|
|
}
|
|
|
|
|
2017-06-22 06:12:43 +03:00
|
|
|
// mDuration.isNothing() represents the "auto" value
|
|
|
|
Maybe<StickyTimeDuration> mDuration;
|
|
|
|
TimeDuration mDelay; // Initializes to zero
|
|
|
|
TimeDuration mEndDelay;
|
|
|
|
double mIterations = 1.0; // Can be NaN, negative, +/-Infinity
|
|
|
|
double mIterationStart = 0.0;
|
|
|
|
dom::PlaybackDirection mDirection = dom::PlaybackDirection::Normal;
|
|
|
|
dom::FillMode mFill = dom::FillMode::Auto;
|
|
|
|
Maybe<ComputedTimingFunction> mFunction;
|
2017-06-22 06:12:43 +03:00
|
|
|
StickyTimeDuration mActiveDuration = StickyTimeDuration();
|
|
|
|
StickyTimeDuration mEndTime = StickyTimeDuration();
|
2016-03-07 08:11:11 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace mozilla
|
|
|
|
|
|
|
|
#endif // mozilla_TimingParams_h
|