This source file includes following definitions.
- m_angle
- notifyChange
- add
- calculateAnimatedValue
- calculateDistance
- m_orientType
- m_orientType
- clone
- cloneForAnimation
- value
- setValue
- stringToAngleType
- valueAsString
- parseValue
- setValueAsString
- newValueSpecifiedUnits
- convertToSpecifiedUnits
- add
- calculateAnimatedValue
- calculateDistance
- orientTypeChanged
#include "config.h"
#include "core/svg/SVGAngle.h"
#include "bindings/v8/ExceptionState.h"
#include "bindings/v8/ExceptionStatePlaceholder.h"
#include "core/dom/ExceptionCode.h"
#include "core/svg/SVGAnimationElement.h"
#include "core/svg/SVGParserUtilities.h"
#include "wtf/MathExtras.h"
#include "wtf/text/WTFString.h"
namespace WebCore {
template<> const SVGEnumerationStringEntries& getStaticStringEntries<SVGMarkerOrientType>()
{
DEFINE_STATIC_LOCAL(SVGEnumerationStringEntries, entries, ());
if (entries.isEmpty()) {
entries.append(std::make_pair(SVGMarkerOrientUnknown, emptyString()));
entries.append(std::make_pair(SVGMarkerOrientAuto, "auto"));
entries.append(std::make_pair(SVGMarkerOrientAngle, "angle"));
}
return entries;
}
SVGMarkerOrientEnumeration::SVGMarkerOrientEnumeration(SVGAngle* angle)
: SVGEnumeration<SVGMarkerOrientType>(SVGMarkerOrientAngle)
, m_angle(angle)
{
}
SVGMarkerOrientEnumeration::~SVGMarkerOrientEnumeration()
{
}
void SVGMarkerOrientEnumeration::notifyChange()
{
ASSERT(m_angle);
m_angle->orientTypeChanged();
}
void SVGMarkerOrientEnumeration::add(PassRefPtr<SVGPropertyBase>, SVGElement*)
{
ASSERT_NOT_REACHED();
}
void SVGMarkerOrientEnumeration::calculateAnimatedValue(SVGAnimationElement*, float percentage, unsigned repeatCount, PassRefPtr<SVGPropertyBase> from, PassRefPtr<SVGPropertyBase> to, PassRefPtr<SVGPropertyBase> toAtEndOfDurationValue, SVGElement* contextElement)
{
ASSERT_NOT_REACHED();
}
float SVGMarkerOrientEnumeration::calculateDistance(PassRefPtr<SVGPropertyBase> to, SVGElement* contextElement)
{
ASSERT_NOT_REACHED();
return -1.0;
}
SVGAngle::SVGAngle()
: SVGPropertyBase(classType())
, m_unitType(SVG_ANGLETYPE_UNSPECIFIED)
, m_valueInSpecifiedUnits(0)
, m_orientType(SVGMarkerOrientEnumeration::create(this))
{
}
SVGAngle::SVGAngle(SVGAngleType unitType, float valueInSpecifiedUnits, SVGMarkerOrientType orientType)
: SVGPropertyBase(classType())
, m_unitType(unitType)
, m_valueInSpecifiedUnits(valueInSpecifiedUnits)
, m_orientType(SVGMarkerOrientEnumeration::create(this))
{
m_orientType->setEnumValue(orientType);
}
SVGAngle::~SVGAngle()
{
}
PassRefPtr<SVGAngle> SVGAngle::clone() const
{
return adoptRef(new SVGAngle(m_unitType, m_valueInSpecifiedUnits, m_orientType->enumValue()));
}
PassRefPtr<SVGPropertyBase> SVGAngle::cloneForAnimation(const String& value) const
{
RefPtr<SVGAngle> point = create();
point->setValueAsString(value, IGNORE_EXCEPTION);
return point.release();
}
float SVGAngle::value() const
{
switch (m_unitType) {
case SVG_ANGLETYPE_GRAD:
return grad2deg(m_valueInSpecifiedUnits);
case SVG_ANGLETYPE_RAD:
return rad2deg(m_valueInSpecifiedUnits);
case SVG_ANGLETYPE_UNSPECIFIED:
case SVG_ANGLETYPE_UNKNOWN:
case SVG_ANGLETYPE_DEG:
return m_valueInSpecifiedUnits;
}
ASSERT_NOT_REACHED();
return 0;
}
void SVGAngle::setValue(float value)
{
switch (m_unitType) {
case SVG_ANGLETYPE_GRAD:
m_valueInSpecifiedUnits = deg2grad(value);
break;
case SVG_ANGLETYPE_RAD:
m_valueInSpecifiedUnits = deg2rad(value);
break;
case SVG_ANGLETYPE_UNSPECIFIED:
case SVG_ANGLETYPE_UNKNOWN:
case SVG_ANGLETYPE_DEG:
m_valueInSpecifiedUnits = value;
break;
}
m_orientType->setEnumValue(SVGMarkerOrientAngle);
}
template<typename CharType>
static SVGAngle::SVGAngleType stringToAngleType(const CharType*& ptr, const CharType* end)
{
if (ptr == end)
return SVGAngle::SVG_ANGLETYPE_UNSPECIFIED;
const CharType firstChar = *ptr;
++ptr;
if (ptr == end)
return SVGAngle::SVG_ANGLETYPE_UNKNOWN;
const CharType secondChar = *ptr;
++ptr;
if (ptr == end)
return SVGAngle::SVG_ANGLETYPE_UNKNOWN;
const CharType thirdChar = *ptr;
if (firstChar == 'd' && secondChar == 'e' && thirdChar == 'g')
return SVGAngle::SVG_ANGLETYPE_DEG;
if (firstChar == 'r' && secondChar == 'a' && thirdChar == 'd')
return SVGAngle::SVG_ANGLETYPE_RAD;
++ptr;
if (ptr == end)
return SVGAngle::SVG_ANGLETYPE_UNKNOWN;
const CharType fourthChar = *ptr;
if (firstChar == 'g' && secondChar == 'r' && thirdChar == 'a' && fourthChar == 'd')
return SVGAngle::SVG_ANGLETYPE_GRAD;
return SVGAngle::SVG_ANGLETYPE_UNKNOWN;
}
String SVGAngle::valueAsString() const
{
switch (m_unitType) {
case SVG_ANGLETYPE_DEG: {
DEFINE_STATIC_LOCAL(String, degString, ("deg"));
return String::number(m_valueInSpecifiedUnits) + degString;
}
case SVG_ANGLETYPE_RAD: {
DEFINE_STATIC_LOCAL(String, radString, ("rad"));
return String::number(m_valueInSpecifiedUnits) + radString;
}
case SVG_ANGLETYPE_GRAD: {
DEFINE_STATIC_LOCAL(String, gradString, ("grad"));
return String::number(m_valueInSpecifiedUnits) + gradString;
}
case SVG_ANGLETYPE_UNSPECIFIED:
case SVG_ANGLETYPE_UNKNOWN:
return String::number(m_valueInSpecifiedUnits);
}
ASSERT_NOT_REACHED();
return String();
}
template<typename CharType>
static bool parseValue(const String& value, float& valueInSpecifiedUnits, SVGAngle::SVGAngleType& unitType)
{
const CharType* ptr = value.getCharacters<CharType>();
const CharType* end = ptr + value.length();
if (!parseNumber(ptr, end, valueInSpecifiedUnits, false))
return false;
unitType = stringToAngleType(ptr, end);
if (unitType == SVGAngle::SVG_ANGLETYPE_UNKNOWN)
return false;
return true;
}
void SVGAngle::setValueAsString(const String& value, ExceptionState& exceptionState)
{
if (value.isEmpty()) {
newValueSpecifiedUnits(SVG_ANGLETYPE_UNSPECIFIED, 0);
return;
}
if (value == "auto") {
newValueSpecifiedUnits(SVG_ANGLETYPE_UNSPECIFIED, 0);
m_orientType->setEnumValue(SVGMarkerOrientAuto);
return;
}
float valueInSpecifiedUnits = 0;
SVGAngleType unitType = SVG_ANGLETYPE_UNKNOWN;
bool success = value.is8Bit() ? parseValue<LChar>(value, valueInSpecifiedUnits, unitType)
: parseValue<UChar>(value, valueInSpecifiedUnits, unitType);
if (!success) {
exceptionState.throwDOMException(SyntaxError, "The value provided ('" + value + "') is invalid.");
return;
}
m_orientType->setEnumValue(SVGMarkerOrientAngle);
m_unitType = unitType;
m_valueInSpecifiedUnits = valueInSpecifiedUnits;
}
void SVGAngle::newValueSpecifiedUnits(SVGAngleType unitType, float valueInSpecifiedUnits)
{
m_orientType->setEnumValue(SVGMarkerOrientAngle);
m_unitType = unitType;
m_valueInSpecifiedUnits = valueInSpecifiedUnits;
}
void SVGAngle::convertToSpecifiedUnits(SVGAngleType unitType, ExceptionState& exceptionState)
{
if (m_unitType == SVG_ANGLETYPE_UNKNOWN) {
exceptionState.throwDOMException(NotSupportedError, "Cannot convert from unknown or invalid units.");
return;
}
if (unitType == m_unitType)
return;
switch (m_unitType) {
case SVG_ANGLETYPE_RAD:
switch (unitType) {
case SVG_ANGLETYPE_GRAD:
m_valueInSpecifiedUnits = rad2grad(m_valueInSpecifiedUnits);
break;
case SVG_ANGLETYPE_UNSPECIFIED:
case SVG_ANGLETYPE_DEG:
m_valueInSpecifiedUnits = rad2deg(m_valueInSpecifiedUnits);
break;
case SVG_ANGLETYPE_RAD:
case SVG_ANGLETYPE_UNKNOWN:
ASSERT_NOT_REACHED();
break;
}
break;
case SVG_ANGLETYPE_GRAD:
switch (unitType) {
case SVG_ANGLETYPE_RAD:
m_valueInSpecifiedUnits = grad2rad(m_valueInSpecifiedUnits);
break;
case SVG_ANGLETYPE_UNSPECIFIED:
case SVG_ANGLETYPE_DEG:
m_valueInSpecifiedUnits = grad2deg(m_valueInSpecifiedUnits);
break;
case SVG_ANGLETYPE_GRAD:
case SVG_ANGLETYPE_UNKNOWN:
ASSERT_NOT_REACHED();
break;
}
break;
case SVG_ANGLETYPE_UNSPECIFIED:
case SVG_ANGLETYPE_DEG:
switch (unitType) {
case SVG_ANGLETYPE_RAD:
m_valueInSpecifiedUnits = deg2rad(m_valueInSpecifiedUnits);
break;
case SVG_ANGLETYPE_GRAD:
m_valueInSpecifiedUnits = deg2grad(m_valueInSpecifiedUnits);
break;
case SVG_ANGLETYPE_UNSPECIFIED:
break;
case SVG_ANGLETYPE_DEG:
case SVG_ANGLETYPE_UNKNOWN:
ASSERT_NOT_REACHED();
break;
}
break;
case SVG_ANGLETYPE_UNKNOWN:
ASSERT_NOT_REACHED();
break;
}
m_unitType = unitType;
m_orientType->setEnumValue(SVGMarkerOrientAngle);
}
void SVGAngle::add(PassRefPtr<SVGPropertyBase> other, SVGElement*)
{
RefPtr<SVGAngle> otherAngle = toSVGAngle(other);
if (orientType()->enumValue() != SVGMarkerOrientAngle || otherAngle->orientType()->enumValue() != SVGMarkerOrientAngle)
return;
setValue(value() + otherAngle->value());
}
void SVGAngle::calculateAnimatedValue(SVGAnimationElement* animationElement, float percentage, unsigned repeatCount, PassRefPtr<SVGPropertyBase> from, PassRefPtr<SVGPropertyBase> to, PassRefPtr<SVGPropertyBase> toAtEndOfDuration, SVGElement*)
{
ASSERT(animationElement);
bool isToAnimation = animationElement->animationMode() == ToAnimation;
RefPtr<SVGAngle> fromAngle = isToAnimation ? this : toSVGAngle(from);
RefPtr<SVGAngle> toAngle = toSVGAngle(to);
RefPtr<SVGAngle> toAtEndOfDurationAngle = toSVGAngle(toAtEndOfDuration);
SVGMarkerOrientType fromOrientType = fromAngle->orientType()->enumValue();
SVGMarkerOrientType toOrientType = toAngle->orientType()->enumValue();
if (fromOrientType != toOrientType) {
if (fromOrientType == SVGMarkerOrientAngle) {
if (toOrientType == SVGMarkerOrientAuto) {
if (percentage < 0.5f) {
newValueSpecifiedUnits(fromAngle->unitType(), fromAngle->valueInSpecifiedUnits());
return;
}
orientType()->setEnumValue(SVGMarkerOrientAuto);
return;
}
m_valueInSpecifiedUnits = 0;
orientType()->setEnumValue(SVGMarkerOrientUnknown);
return;
}
}
if (fromOrientType == SVGMarkerOrientAuto) {
m_valueInSpecifiedUnits = 0;
orientType()->setEnumValue(SVGMarkerOrientAuto);
return;
}
if (fromOrientType != SVGMarkerOrientAngle) {
m_valueInSpecifiedUnits = 0;
orientType()->setEnumValue(SVGMarkerOrientUnknown);
return;
}
float animatedValue = value();
animationElement->animateAdditiveNumber(percentage, repeatCount, fromAngle->value(), toAngle->value(), toAtEndOfDurationAngle->value(), animatedValue);
orientType()->setEnumValue(SVGMarkerOrientAngle);
setValue(animatedValue);
}
float SVGAngle::calculateDistance(PassRefPtr<SVGPropertyBase> other, SVGElement*)
{
return fabsf(value() - toSVGAngle(other)->value());
}
void SVGAngle::orientTypeChanged()
{
if (orientType()->enumValue() == SVGMarkerOrientAuto) {
m_unitType = SVG_ANGLETYPE_UNSPECIFIED;
m_valueInSpecifiedUnits = 0;
}
}
}