2015-05-03 22:32:37 +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: */
|
2012-05-21 15:12:37 +04:00
|
|
|
/* 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/. */
|
2009-01-15 07:38:07 +03:00
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
#include "SMILValue.h"
|
|
|
|
|
2009-01-15 07:38:07 +03:00
|
|
|
#include "nsDebug.h"
|
2010-03-26 22:22:54 +03:00
|
|
|
#include <string.h>
|
2009-01-15 07:38:07 +03:00
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
namespace mozilla {
|
|
|
|
|
2010-01-29 23:18:50 +03:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Public methods
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
SMILValue::SMILValue(const SMILType* aType) : mType(SMILNullType::Singleton()) {
|
2018-06-16 13:54:44 +03:00
|
|
|
mU.mBool = false;
|
2010-01-29 23:18:50 +03:00
|
|
|
if (!aType) {
|
2019-01-23 16:48:08 +03:00
|
|
|
NS_ERROR("Trying to construct SMILValue with null mType pointer");
|
2010-01-29 23:18:50 +03:00
|
|
|
return;
|
|
|
|
}
|
2009-01-15 07:38:07 +03:00
|
|
|
|
2010-01-29 23:18:50 +03:00
|
|
|
InitAndCheckPostcondition(aType);
|
2009-01-15 07:38:07 +03:00
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
SMILValue::SMILValue(const SMILValue& aVal) : mType(SMILNullType::Singleton()) {
|
2010-03-22 21:57:36 +03:00
|
|
|
InitAndCheckPostcondition(aVal.mType);
|
2009-01-15 07:38:07 +03:00
|
|
|
mType->Assign(*this, aVal);
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
const SMILValue& SMILValue::operator=(const SMILValue& aVal) {
|
2009-01-15 07:38:07 +03:00
|
|
|
if (&aVal == this) return *this;
|
|
|
|
|
|
|
|
if (mType != aVal.mType) {
|
2010-03-22 21:57:36 +03:00
|
|
|
DestroyAndReinit(aVal.mType);
|
2009-01-15 07:38:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
mType->Assign(*this, aVal);
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2015-05-29 23:10:39 +03:00
|
|
|
// Move constructor / reassignment operator:
|
2019-04-24 23:39:47 +03:00
|
|
|
SMILValue::SMILValue(SMILValue&& aVal) noexcept
|
2015-05-29 23:10:39 +03:00
|
|
|
: mU(aVal.mU), // Copying union is only OK because we clear aVal.mType
|
|
|
|
// below.
|
|
|
|
mType(aVal.mType) {
|
|
|
|
// Leave aVal with a null type, so that it's safely destructible (and won't
|
|
|
|
// mess with anything referenced by its union, which we've copied).
|
2018-12-27 13:21:40 +03:00
|
|
|
aVal.mType = SMILNullType::Singleton();
|
2015-05-29 23:10:39 +03:00
|
|
|
}
|
|
|
|
|
2019-04-24 23:39:47 +03:00
|
|
|
SMILValue& SMILValue::operator=(SMILValue&& aVal) noexcept {
|
2015-05-29 23:10:39 +03:00
|
|
|
if (!IsNull()) {
|
|
|
|
// Clean up any data we're currently tracking.
|
|
|
|
DestroyAndCheckPostcondition();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy the union (which could include a pointer to external memory) & mType:
|
|
|
|
mU = aVal.mU;
|
|
|
|
mType = aVal.mType;
|
|
|
|
|
|
|
|
// Leave aVal with a null type, so that it's safely destructible (and won't
|
|
|
|
// mess with anything referenced by its union, which we've now copied).
|
2018-12-27 13:21:40 +03:00
|
|
|
aVal.mType = SMILNullType::Singleton();
|
2015-05-29 23:10:39 +03:00
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
bool SMILValue::operator==(const SMILValue& aVal) const {
|
2010-02-21 00:13:11 +03:00
|
|
|
if (&aVal == this) return true;
|
|
|
|
|
|
|
|
return mType == aVal.mType && mType->IsEqual(*this, aVal);
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
nsresult SMILValue::Add(const SMILValue& aValueToAdd, uint32_t aCount) {
|
2009-01-15 07:38:07 +03:00
|
|
|
if (aValueToAdd.mType != mType) {
|
2010-01-28 12:50:30 +03:00
|
|
|
NS_ERROR("Trying to add incompatible types");
|
2009-01-15 07:38:07 +03:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mType->Add(*this, aValueToAdd, aCount);
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
nsresult SMILValue::SandwichAdd(const SMILValue& aValueToAdd) {
|
2009-01-19 12:14:16 +03:00
|
|
|
if (aValueToAdd.mType != mType) {
|
2010-01-28 12:50:30 +03:00
|
|
|
NS_ERROR("Trying to add incompatible types");
|
2009-01-19 12:14:16 +03:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mType->SandwichAdd(*this, aValueToAdd);
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
nsresult SMILValue::ComputeDistance(const SMILValue& aTo,
|
|
|
|
double& aDistance) const {
|
2009-01-15 07:38:07 +03:00
|
|
|
if (aTo.mType != mType) {
|
2010-01-28 12:50:30 +03:00
|
|
|
NS_ERROR("Trying to calculate distance between incompatible types");
|
2009-01-15 07:38:07 +03:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mType->ComputeDistance(*this, aTo, aDistance);
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
nsresult SMILValue::Interpolate(const SMILValue& aEndVal, double aUnitDistance,
|
|
|
|
SMILValue& aResult) const {
|
2009-01-15 07:38:07 +03:00
|
|
|
if (aEndVal.mType != mType) {
|
2010-01-28 12:50:30 +03:00
|
|
|
NS_ERROR("Trying to interpolate between incompatible types");
|
2009-01-15 07:38:07 +03:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aResult.mType != mType) {
|
2010-01-29 23:18:50 +03:00
|
|
|
// Outparam has wrong type
|
2010-03-22 21:57:36 +03:00
|
|
|
aResult.DestroyAndReinit(mType);
|
2009-01-15 07:38:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return mType->Interpolate(*this, aEndVal, aUnitDistance, aResult);
|
|
|
|
}
|
2010-01-29 23:18:50 +03:00
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Helper methods
|
|
|
|
|
2018-12-28 13:34:35 +03:00
|
|
|
// Wrappers for SMILType::Init & ::Destroy that verify their postconditions
|
2019-01-23 16:48:08 +03:00
|
|
|
void SMILValue::InitAndCheckPostcondition(const SMILType* aNewType) {
|
2010-03-22 21:57:36 +03:00
|
|
|
aNewType->Init(*this);
|
2015-02-10 01:34:50 +03:00
|
|
|
MOZ_ASSERT(mType == aNewType,
|
2019-01-23 16:48:08 +03:00
|
|
|
"Post-condition of Init failed. SMILValue is invalid");
|
2010-01-29 23:18:50 +03:00
|
|
|
}
|
2017-07-06 15:00:35 +03:00
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
void SMILValue::DestroyAndCheckPostcondition() {
|
2010-01-29 23:18:50 +03:00
|
|
|
mType->Destroy(*this);
|
2015-02-10 01:34:50 +03:00
|
|
|
MOZ_ASSERT(IsNull(),
|
|
|
|
"Post-condition of Destroy failed. "
|
2019-01-23 16:48:08 +03:00
|
|
|
"SMILValue not null after destroying");
|
2010-01-29 23:18:50 +03:00
|
|
|
}
|
|
|
|
|
2019-01-23 16:48:08 +03:00
|
|
|
void SMILValue::DestroyAndReinit(const SMILType* aNewType) {
|
2010-01-29 23:18:50 +03:00
|
|
|
DestroyAndCheckPostcondition();
|
2010-03-22 21:57:36 +03:00
|
|
|
InitAndCheckPostcondition(aNewType);
|
2010-01-29 23:18:50 +03:00
|
|
|
}
|
2019-01-23 16:48:08 +03:00
|
|
|
|
|
|
|
} // namespace mozilla
|