2011-06-22  Dirk Schulze  <krit@webkit.org>

        Reviewed by Nikolas Zimmermann.

        Animate viewBox attribute in SVG
        https://bugs.webkit.org/show_bug.cgi?id=20057

        Test SVGRect animation with from-to and from-by animations.

        * svg/animations/script-tests/svgrect-animation-1.js: Added.
        (sample1):
        (sample2):
        (sample3):
        (executeTest):
        * svg/animations/script-tests/svgrect-animation-2.js: Added.
        (sample1):
        (sample2):
        (sample3):
        (executeTest):
        * svg/animations/svgrect-animation-1-expected.txt: Added.
        * svg/animations/svgrect-animation-1.html: Added.
        * svg/animations/svgrect-animation-2-expected.txt: Added.
        * svg/animations/svgrect-animation-2.html: Added.
2011-06-22  Dirk Schulze  <krit@webkit.org>

        Reviewed by Nikolas Zimmermann.

        Animate viewBox attribute in SVG
        https://bugs.webkit.org/show_bug.cgi?id=20057

        Follow up of "SVGAnimation should use direct unit animation for SVGLength": https://bugs.webkit.org/show_bug.cgi?id=61368
        This patch continues the conversion to the new concept of SVGAnimatorFactory with SVGRect. With the new animator we support
        interpolation between SVGRects. A feature that can be used to get smooth zooming animations for SVGs.
        
        Added parseRect() to SVGParserUtilities for parsing Strings to FloatRects.

        Tests: svg/animations/svgrect-animation-1.html
               svg/animations/svgrect-animation-2.html

        * CMakeLists.txt: Added new file SVGAnimatedRect.cpp to build system.
        * GNUmakefile.list.am: Ditto.
        * WebCore.gypi: Ditto.
        * WebCore.pro: Ditto.
        * WebCore.xcodeproj/project.pbxproj: Ditto.
        * platform/graphics/FloatRect.h: Added new operators + and +=.
        (WebCore::operator+=):
        (WebCore::operator+):
        * svg/SVGAllInOne.cpp: Added SVGAnimatedRect.cpp for Win build.
        * svg/SVGAnimateElement.cpp: Handle AnimatedRect explicitly.
        (WebCore::SVGAnimateElement::determineAnimatedAttributeType): Ditto.
        (WebCore::SVGAnimateElement::calculateAnimatedValue): Ditto.
        (WebCore::SVGAnimateElement::calculateFromAndToValues): Ditto.
        (WebCore::SVGAnimateElement::calculateFromAndByValues): Ditto.
        (WebCore::SVGAnimateElement::resetToBaseValue): Ditto.
        (WebCore::SVGAnimateElement::applyResultsToTarget): Ditto.
        (WebCore::SVGAnimateElement::calculateDistance): Ditto.
        * svg/SVGAnimatedRect.cpp: Added. The new Animator for SVGRect.
        (WebCore::SVGAnimatedRectAnimator::SVGAnimatedRectAnimator):
        (WebCore::SVGAnimatedRectAnimator::constructFromString):
        (WebCore::SVGAnimatedRectAnimator::calculateFromAndToValues):
        (WebCore::SVGAnimatedRectAnimator::calculateFromAndByValues):
        (WebCore::SVGAnimatedRectAnimator::calculateAnimatedValue):
        (WebCore::SVGAnimatedRectAnimator::calculateDistance):
        * svg/SVGAnimatedRect.h:
        (WebCore::SVGAnimatedRectAnimator::~SVGAnimatedRectAnimator):
        * svg/SVGAnimatedType.cpp: Support for FloatRect.
        (WebCore::SVGAnimatedType::~SVGAnimatedType):
        (WebCore::SVGAnimatedType::createRect):
        (WebCore::SVGAnimatedType::rect):
        (WebCore::SVGAnimatedType::valueAsString):
        (WebCore::SVGAnimatedType::setValueAsString):
        * svg/SVGAnimatedType.h:
        * svg/SVGAnimatorFactory.h:
        (WebCore::SVGAnimatorFactory::create):
        * svg/SVGParserUtilities.cpp: Parse String as FloatRect.
        (WebCore::parseRect):
        * svg/SVGParserUtilities.h:


git-svn-id: http://svn.webkit.org/repository/webkit/trunk@89431 268f45cc-cd09-0410-ab3c-d52691b4dbfc
diff --git a/Source/WebCore/svg/SVGAllInOne.cpp b/Source/WebCore/svg/SVGAllInOne.cpp
index fa31e6d..5e96e6b 100644
--- a/Source/WebCore/svg/SVGAllInOne.cpp
+++ b/Source/WebCore/svg/SVGAllInOne.cpp
@@ -33,6 +33,7 @@
 #include "SVGAnimatedAngle.cpp"
 #include "SVGAnimatedLength.cpp"
 #include "SVGAnimatedNumber.cpp"
+#include "SVGAnimatedRect.cpp"
 #include "SVGAnimatedType.cpp"
 #include "SVGAnimateElement.cpp"
 #include "SVGAnimateMotionElement.cpp"
diff --git a/Source/WebCore/svg/SVGAnimateElement.cpp b/Source/WebCore/svg/SVGAnimateElement.cpp
index a033dbe..d7545ee 100644
--- a/Source/WebCore/svg/SVGAnimateElement.cpp
+++ b/Source/WebCore/svg/SVGAnimateElement.cpp
@@ -120,7 +120,6 @@
     case AnimatedNumberOptionalNumber:
     case AnimatedLengthList:
     case AnimatedPreserveAspectRatio:
-    case AnimatedRect:
     case AnimatedString:
         return AnimatedString;
     case AnimatedLength:
@@ -132,6 +131,8 @@
         return AnimatedPath;
     case AnimatedPoints:
         return AnimatedPoints;
+    case AnimatedRect:
+        return AnimatedRect;
     case AnimatedColor:
         return AnimatedColor;
     case AnimatedTransformList:
@@ -250,7 +251,8 @@
     }
     case AnimatedAngle:
     case AnimatedLength:
-    case AnimatedNumber: {
+    case AnimatedNumber:
+    case AnimatedRect: {
         ASSERT(m_animator);
         ASSERT(results->m_animatedType);
         // Target element might have changed.
@@ -337,6 +339,7 @@
     case AnimatedAngle:
     case AnimatedLength:
     case AnimatedNumber:
+    case AnimatedRect:
         ensureAnimator()->calculateFromAndToValues(m_fromType, m_toType, fromString, toString);
         return true;
     default:
@@ -379,6 +382,7 @@
     case AnimatedAngle:
     case AnimatedLength:
     case AnimatedNumber:
+    case AnimatedRect:
         ensureAnimator()->calculateFromAndByValues(m_fromType, m_toType, fromString, byString);
         return true;
     default:
@@ -414,7 +418,8 @@
         return;
     case AnimatedAngle:
     case AnimatedLength:
-    case AnimatedNumber: {
+    case AnimatedNumber:
+    case AnimatedRect: {
         if (!m_animatedType)
             m_animatedType = ensureAnimator()->constructFromString(baseString);
         else
@@ -453,6 +458,7 @@
     case AnimatedAngle:
     case AnimatedLength:
     case AnimatedNumber:
+    case AnimatedRect:
         valueToApply = m_animatedType->valueAsString();
         break;
     default:
@@ -481,6 +487,7 @@
     case AnimatedAngle:
     case AnimatedLength:
     case AnimatedNumber:
+    case AnimatedRect:
         return ensureAnimator()->calculateDistance(this, fromString, toString);
     default:
         break;
diff --git a/Source/WebCore/svg/SVGAnimatedRect.cpp b/Source/WebCore/svg/SVGAnimatedRect.cpp
new file mode 100644
index 0000000..9ef95be
--- /dev/null
+++ b/Source/WebCore/svg/SVGAnimatedRect.cpp
@@ -0,0 +1,104 @@
+/*
+ * Copyright (C) Research In Motion Limited 2011. All rights reserved.
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public License
+ * along with this library; see the file COPYING.LIB.  If not, write to
+ * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ * Boston, MA 02110-1301, USA.
+ */
+
+#include "config.h"
+
+#if ENABLE(SVG) && ENABLE(SVG_ANIMATION)
+#include "SVGAnimatedRect.h"
+
+#include "SVGParserUtilities.h"
+
+namespace WebCore {
+
+SVGAnimatedRectAnimator::SVGAnimatedRectAnimator(SVGElement* contextElement, const QualifiedName&)
+    : SVGAnimatedTypeAnimator(AnimatedRect, contextElement)
+{
+}
+
+PassOwnPtr<SVGAnimatedType> SVGAnimatedRectAnimator::constructFromString(const String& string)
+{
+    OwnPtr<SVGAnimatedType> animatedType = SVGAnimatedType::createRect(new FloatRect);
+    parseRect(string, animatedType->rect());
+    return animatedType.release();
+}
+
+void SVGAnimatedRectAnimator::calculateFromAndToValues(OwnPtr<SVGAnimatedType>& from, OwnPtr<SVGAnimatedType>& to, const String& fromString, const String& toString)
+{
+    from = constructFromString(fromString);
+    to = constructFromString(toString);
+}
+
+void SVGAnimatedRectAnimator::calculateFromAndByValues(OwnPtr<SVGAnimatedType>& from, OwnPtr<SVGAnimatedType>& to, const String& fromString, const String& byString)
+{
+    ASSERT(m_contextElement);
+    
+    from = constructFromString(fromString);
+    to = constructFromString(byString);
+
+    to->rect() += from->rect();
+}
+
+void SVGAnimatedRectAnimator::calculateAnimatedValue(SVGSMILElement* smilElement, float percentage, unsigned repeatCount,
+                                                       OwnPtr<SVGAnimatedType>& from, OwnPtr<SVGAnimatedType>& to, OwnPtr<SVGAnimatedType>& animated,
+                                                       bool, bool)
+{
+    ASSERT(smilElement);
+    ASSERT(m_contextElement);
+
+    SVGAnimateElement* animationElement = static_cast<SVGAnimateElement*>(smilElement);
+    AnimationMode animationMode = animationElement->animationMode();
+    // To animation uses contributions from the lower priority animations as the base value.
+    FloatRect& animatedRect = animated->rect();
+    if (animationMode == ToAnimation)
+        from->rect() = animatedRect;
+    
+    const FloatRect& fromRect = from->rect();
+    const FloatRect& toRect = to->rect();
+    FloatRect newRect;    
+    if (animationElement->calcMode() == CalcModeDiscrete)
+        newRect = percentage < 0.5 ? fromRect : toRect;
+    else
+        newRect = FloatRect((toRect.x() - fromRect.x()) * percentage + fromRect.x(),
+                            (toRect.y() - fromRect.y()) * percentage + fromRect.y(),
+                            (toRect.width() - fromRect.width()) * percentage + fromRect.width(),
+                            (toRect.height() - fromRect.height()) * percentage + fromRect.height());
+    
+    // FIXME: This is not correct for values animation. Right now we transform values-animation to multiple from-to-animations and
+    // accumulate every single value to the previous one. But accumulation should just take into account after a complete cycle
+    // of values-animaiton. See example at: http://www.w3.org/TR/2001/REC-smil-animation-20010904/#RepeatingAnim
+    if (animationElement->isAccumulated() && repeatCount) {
+        newRect += toRect;
+        newRect.scale(repeatCount);
+    }
+    
+    if (animationElement->isAdditive() && animationMode != ToAnimation)
+        animatedRect += newRect;
+    else
+        animatedRect = newRect;
+}
+
+float SVGAnimatedRectAnimator::calculateDistance(SVGSMILElement*, const String&, const String&)
+{
+    // FIXME: Distance calculation is not possible for SVGRect right now. We need the distance of for every single value.
+    return -1;
+}
+    
+}
+
+#endif // ENABLE(SVG) && ENABLE(SVG_ANIMATION)
diff --git a/Source/WebCore/svg/SVGAnimatedRect.h b/Source/WebCore/svg/SVGAnimatedRect.h
index af9f94b..02e8652 100644
--- a/Source/WebCore/svg/SVGAnimatedRect.h
+++ b/Source/WebCore/svg/SVGAnimatedRect.h
@@ -21,6 +21,7 @@
 #define SVGAnimatedRect_h
 
 #if ENABLE(SVG)
+#include "SVGAnimateElement.h"
 #include "SVGAnimatedPropertyMacros.h"
 #include "SVGAnimatedPropertyTearOff.h"
 #include "SVGRect.h"
@@ -36,7 +37,26 @@
 #define DEFINE_ANIMATED_RECT(OwnerType, DOMAttribute, UpperProperty, LowerProperty) \
 DEFINE_ANIMATED_PROPERTY(OwnerType, DOMAttribute, DOMAttribute.localName(), SVGAnimatedRect, FloatRect, UpperProperty, LowerProperty)
 
+#if ENABLE(SVG_ANIMATION)
+class SVGAnimatedRectAnimator : public SVGAnimatedTypeAnimator {
+    
+public:
+    SVGAnimatedRectAnimator(SVGElement* contextElement, const QualifiedName&);
+    virtual ~SVGAnimatedRectAnimator() { }
+    
+    virtual PassOwnPtr<SVGAnimatedType> constructFromString(const String&);
+    
+    virtual void calculateFromAndToValues(OwnPtr<SVGAnimatedType>& fromValue, OwnPtr<SVGAnimatedType>& toValue, const String& fromString, const String& toString);
+    virtual void calculateFromAndByValues(OwnPtr<SVGAnimatedType>& fromValue, OwnPtr<SVGAnimatedType>& toValue, const String& fromString, const String& byString);
+    virtual void calculateAnimatedValue(SVGSMILElement*, float percentage, unsigned repeatCount,
+                                        OwnPtr<SVGAnimatedType>& fromValue, OwnPtr<SVGAnimatedType>& toValue, OwnPtr<SVGAnimatedType>& animatedValue,
+                                        bool fromPropertyInherits, bool toPropertyInherits);
+    virtual float calculateDistance(SVGSMILElement*, const String& fromString, const String& toString);
+    
+    static bool parseSVGRect(const String&, FloatRect&);
+};
 } // namespace WebCore
 
+#endif // ENABLE(SVG_ANIMATION)
 #endif // ENABLE(SVG)
 #endif
diff --git a/Source/WebCore/svg/SVGAnimatedType.cpp b/Source/WebCore/svg/SVGAnimatedType.cpp
index e2fd825..ddfe261 100644
--- a/Source/WebCore/svg/SVGAnimatedType.cpp
+++ b/Source/WebCore/svg/SVGAnimatedType.cpp
@@ -22,6 +22,7 @@
 #if ENABLE(SVG) && ENABLE(SVG_ANIMATION)
 #include "SVGAnimatedType.h"
 
+#include "FloatRect.h"
 #include "SVGAngle.h"
 #include "SVGLength.h"
 #include "SVGParserUtilities.h"
@@ -45,6 +46,9 @@
     case AnimatedNumber:
         delete m_data.number;
         break;
+    case AnimatedRect:
+        delete m_data.rect;
+        break;
     default:
         ASSERT_NOT_REACHED();
         break;
@@ -75,6 +79,14 @@
     return animatedType.release();
 }
 
+PassOwnPtr<SVGAnimatedType> SVGAnimatedType::createRect(FloatRect* rect)
+{
+    ASSERT(rect);
+    OwnPtr<SVGAnimatedType> animatedType = adoptPtr(new SVGAnimatedType(AnimatedRect));
+    animatedType->m_data.rect = rect;
+    return animatedType.release();
+}
+
 SVGAngle& SVGAnimatedType::angle()
 {
     ASSERT(m_type == AnimatedAngle);
@@ -93,6 +105,12 @@
     return *m_data.number;
 }
 
+FloatRect& SVGAnimatedType::rect()
+{
+    ASSERT(m_type == AnimatedRect);
+    return *m_data.rect;
+}
+
 String SVGAnimatedType::valueAsString()
 {
     switch (m_type) {
@@ -105,6 +123,10 @@
     case AnimatedNumber:
         ASSERT(m_data.number);
         return String::number(*m_data.number);
+    case AnimatedRect:
+        ASSERT(m_data.rect);
+        return String::number(m_data.rect->x()) + ' ' + String::number(m_data.rect->y()) + ' '
+             + String::number(m_data.rect->width()) + ' ' + String::number(m_data.rect->height());
     default:
         break;
     }
@@ -128,6 +150,10 @@
         ASSERT(m_data.number);
         parseNumberFromString(value, *m_data.number);
         break;
+    case AnimatedRect:
+        ASSERT(m_data.rect);
+        parseRect(value, *m_data.rect);
+        break;
     default:
         ASSERT_NOT_REACHED();
         break;
diff --git a/Source/WebCore/svg/SVGAnimatedType.h b/Source/WebCore/svg/SVGAnimatedType.h
index 7ef93d1..e8d6add 100644
--- a/Source/WebCore/svg/SVGAnimatedType.h
+++ b/Source/WebCore/svg/SVGAnimatedType.h
@@ -25,6 +25,7 @@
 
 namespace WebCore {
 
+class FloatRect;
 class SVGAngle;
 class SVGLength;
 
@@ -36,12 +37,14 @@
     static PassOwnPtr<SVGAnimatedType> createAngle(SVGAngle*);
     static PassOwnPtr<SVGAnimatedType> createLength(SVGLength*);
     static PassOwnPtr<SVGAnimatedType> createNumber(float*);
+    static PassOwnPtr<SVGAnimatedType> createRect(FloatRect*);
     
     AnimatedAttributeType type() const { return m_type; }
 
     SVGAngle& angle();
     SVGLength& length();
     float& number();
+    FloatRect& rect();
 
     String valueAsString();
     bool setValueAsString(const QualifiedName&, const String&);
@@ -61,6 +64,7 @@
         SVGAngle* angle;
         SVGLength* length;
         float* number;
+        FloatRect* rect;
     } m_data;
 };
     
diff --git a/Source/WebCore/svg/SVGAnimatorFactory.h b/Source/WebCore/svg/SVGAnimatorFactory.h
index f880717..62bb770 100644
--- a/Source/WebCore/svg/SVGAnimatorFactory.h
+++ b/Source/WebCore/svg/SVGAnimatorFactory.h
@@ -24,6 +24,7 @@
 #include "SVGAnimatedAngle.h"
 #include "SVGAnimatedLength.h"
 #include "SVGAnimatedNumber.h"
+#include "SVGAnimatedRect.h"
 
 namespace WebCore {
     
@@ -39,6 +40,8 @@
             return adoptPtr(new SVGAnimatedLengthAnimator(contextElement, attributeName));
         case AnimatedNumber:
             return adoptPtr(new SVGAnimatedNumberAnimator(contextElement, attributeName));
+        case AnimatedRect:
+            return adoptPtr(new SVGAnimatedRectAnimator(contextElement, attributeName));
         default:
             ASSERT_NOT_REACHED();
             return adoptPtr(new SVGAnimatedLengthAnimator(contextElement, attributeName));
diff --git a/Source/WebCore/svg/SVGParserUtilities.cpp b/Source/WebCore/svg/SVGParserUtilities.cpp
index c09f8fe..a858dad 100644
--- a/Source/WebCore/svg/SVGParserUtilities.cpp
+++ b/Source/WebCore/svg/SVGParserUtilities.cpp
@@ -26,7 +26,7 @@
 #include "SVGParserUtilities.h"
 
 #include "Document.h"
-#include "FloatPoint.h"
+#include "FloatRect.h"
 #include "SVGPointList.h"
 
 #include <limits>
@@ -189,6 +189,21 @@
     return cur == end;
 }
 
+bool parseRect(const String& string, FloatRect& rect)
+{
+    const UChar* ptr = string.characters();
+    const UChar* end = ptr + string.length();
+    skipOptionalSpaces(ptr, end);
+    
+    float x = 0;
+    float y = 0;
+    float width = 0;
+    float height = 0;
+    bool valid = parseNumber(ptr, end, x) && parseNumber(ptr, end, y) && parseNumber(ptr, end, width) && parseNumber(ptr, end, height, false);
+    rect = FloatRect(x, y, width, height);
+    return valid;
+}
+
 bool pointsListFromSVGData(SVGPointList& pointsList, const String& points)
 {
     if (points.isEmpty())
diff --git a/Source/WebCore/svg/SVGParserUtilities.h b/Source/WebCore/svg/SVGParserUtilities.h
index 921da6a..48406bb 100644
--- a/Source/WebCore/svg/SVGParserUtilities.h
+++ b/Source/WebCore/svg/SVGParserUtilities.h
@@ -30,12 +30,14 @@
 
 namespace WebCore {
 
+class FloatRect;
 class SVGPointList;
 
 bool parseNumber(const UChar*& ptr, const UChar* end, float& number, bool skip = true);
 bool parseNumberFromString(const String&, float& number, bool skip = true);
 bool parseNumberOptionalNumber(const String& s, float& h, float& v);
 bool parseArcFlag(const UChar*& ptr, const UChar* end, bool& flag);
+bool parseRect(const String&, FloatRect&);
 
 // SVG allows several different whitespace characters:
 // http://www.w3.org/TR/SVG/paths.html#PathDataBNF