Index: third_party/WebKit/Source/core/css/CSSMatrix.cpp |
diff --git a/third_party/WebKit/Source/core/css/CSSMatrix.cpp b/third_party/WebKit/Source/core/css/CSSMatrix.cpp |
new file mode 100644 |
index 0000000000000000000000000000000000000000..92af03334253d3ad62c8a6bded0454aefa996737 |
--- /dev/null |
+++ b/third_party/WebKit/Source/core/css/CSSMatrix.cpp |
@@ -0,0 +1,205 @@ |
+/* |
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved. |
+ * |
+ * Redistribution and use in source and binary forms, with or without |
+ * modification, are permitted provided that the following conditions |
+ * are met: |
+ * 1. Redistributions of source code must retain the above copyright |
+ * notice, this list of conditions and the following disclaimer. |
+ * 2. Redistributions in binary form must reproduce the above copyright |
+ * notice, this list of conditions and the following disclaimer in the |
+ * documentation and/or other materials provided with the distribution. |
+ * |
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY |
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR |
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
+ */ |
+ |
+#include "core/css/CSSMatrix.h" |
+ |
+#include "bindings/core/v8/ExceptionState.h" |
+#include "core/CSSPropertyNames.h" |
+#include "core/CSSValueKeywords.h" |
+#include "core/css/CSSIdentifierValue.h" |
+#include "core/css/CSSToLengthConversionData.h" |
+#include "core/css/StylePropertySet.h" |
+#include "core/css/parser/CSSParser.h" |
+#include "core/css/resolver/TransformBuilder.h" |
+#include "core/dom/ExceptionCode.h" |
+#include "core/frame/UseCounter.h" |
+#include "core/layout/api/LayoutViewItem.h" |
+#include "core/style/ComputedStyle.h" |
+#include "platform/wtf/MathExtras.h" |
+ |
+namespace blink { |
+ |
+CSSMatrix* CSSMatrix::Create(ExecutionContext* execution_context, |
+ const String& s, |
+ ExceptionState& exception_state) { |
+ return new CSSMatrix(s, exception_state); |
+} |
+ |
+CSSMatrix::CSSMatrix(const TransformationMatrix& m) |
+ : matrix_(TransformationMatrix::Create(m)) {} |
+ |
+CSSMatrix::CSSMatrix(const String& s, ExceptionState& exception_state) |
+ : matrix_(TransformationMatrix::Create()) { |
+ setMatrixValue(s, exception_state); |
+} |
+ |
+static inline PassRefPtr<ComputedStyle> CreateInitialStyle() { |
+ RefPtr<ComputedStyle> initial_style = ComputedStyle::Create(); |
+ initial_style->GetFont().Update(nullptr); |
+ return initial_style; |
+} |
+ |
+void CSSMatrix::setMatrixValue(const String& string, |
+ ExceptionState& exception_state) { |
+ if (string.IsEmpty()) |
+ return; |
+ |
+ if (const CSSValue* value = |
+ CSSParser::ParseSingleValue(CSSPropertyTransform, string)) { |
+ // Check for a "none" transform. In these cases we can use the default |
+ // identity matrix. |
+ if (value->IsIdentifierValue() && |
+ (ToCSSIdentifierValue(value))->GetValueID() == CSSValueNone) |
+ return; |
+ |
+ DEFINE_STATIC_REF(ComputedStyle, initial_style, CreateInitialStyle()); |
+ TransformOperations operations = |
+ TransformBuilder::CreateTransformOperations( |
+ *value, CSSToLengthConversionData(initial_style, initial_style, |
+ LayoutViewItem(nullptr), 1.0f)); |
+ |
+ // Convert transform operations to a TransformationMatrix. This can fail |
+ // if a param has a percentage ('%') |
+ if (operations.DependsOnBoxSize()) { |
+ exception_state.ThrowDOMException(kSyntaxError, |
+ "The transformation depends on the box " |
+ "size, which is not supported."); |
+ } |
+ matrix_ = TransformationMatrix::Create(); |
+ operations.Apply(FloatSize(0, 0), *matrix_); |
+ } else { // There is something there but parsing failed. |
+ exception_state.ThrowDOMException(kSyntaxError, |
+ "Failed to parse '" + string + "'."); |
+ } |
+} |
+ |
+// Perform a concatenation of the matrices (this * secondMatrix) |
+CSSMatrix* CSSMatrix::multiply(CSSMatrix* second_matrix) const { |
+ if (!second_matrix) |
+ return nullptr; |
+ |
+ return CSSMatrix::Create( |
+ TransformationMatrix(*matrix_).Multiply(*second_matrix->matrix_)); |
+} |
+ |
+CSSMatrix* CSSMatrix::inverse(ExceptionState& exception_state) const { |
+ if (!matrix_->IsInvertible()) { |
+ exception_state.ThrowDOMException(kNotSupportedError, |
+ "The matrix is not invertable."); |
+ return nullptr; |
+ } |
+ |
+ return CSSMatrix::Create(matrix_->Inverse()); |
+} |
+ |
+CSSMatrix* CSSMatrix::translate(double x, double y, double z) const { |
+ if (std::isnan(x)) |
+ x = 0; |
+ if (std::isnan(y)) |
+ y = 0; |
+ if (std::isnan(z)) |
+ z = 0; |
+ return CSSMatrix::Create(TransformationMatrix(*matrix_).Translate3d(x, y, z)); |
+} |
+ |
+CSSMatrix* CSSMatrix::scale(double scale_x, |
+ double scale_y, |
+ double scale_z) const { |
+ if (std::isnan(scale_x)) |
+ scale_x = 1; |
+ if (std::isnan(scale_y)) |
+ scale_y = scale_x; |
+ if (std::isnan(scale_z)) |
+ scale_z = 1; |
+ return CSSMatrix::Create( |
+ TransformationMatrix(*matrix_).Scale3d(scale_x, scale_y, scale_z)); |
+} |
+ |
+CSSMatrix* CSSMatrix::rotate(double rot_x, double rot_y, double rot_z) const { |
+ if (std::isnan(rot_x)) |
+ rot_x = 0; |
+ |
+ if (std::isnan(rot_y) && std::isnan(rot_z)) { |
+ rot_z = rot_x; |
+ rot_x = 0; |
+ rot_y = 0; |
+ } |
+ |
+ if (std::isnan(rot_y)) |
+ rot_y = 0; |
+ if (std::isnan(rot_z)) |
+ rot_z = 0; |
+ return CSSMatrix::Create( |
+ TransformationMatrix(*matrix_).Rotate3d(rot_x, rot_y, rot_z)); |
+} |
+ |
+CSSMatrix* CSSMatrix::rotateAxisAngle(double x, |
+ double y, |
+ double z, |
+ double angle) const { |
+ if (std::isnan(x)) |
+ x = 0; |
+ if (std::isnan(y)) |
+ y = 0; |
+ if (std::isnan(z)) |
+ z = 0; |
+ if (std::isnan(angle)) |
+ angle = 0; |
+ if (!x && !y && !z) |
+ z = 1; |
+ return CSSMatrix::Create( |
+ TransformationMatrix(*matrix_).Rotate3d(x, y, z, angle)); |
+} |
+ |
+CSSMatrix* CSSMatrix::skewX(double angle) const { |
+ if (std::isnan(angle)) |
+ angle = 0; |
+ return CSSMatrix::Create(TransformationMatrix(*matrix_).SkewX(angle)); |
+} |
+ |
+CSSMatrix* CSSMatrix::skewY(double angle) const { |
+ if (std::isnan(angle)) |
+ angle = 0; |
+ return CSSMatrix::Create(TransformationMatrix(*matrix_).SkewY(angle)); |
+} |
+ |
+String CSSMatrix::toString() const { |
+ // FIXME - Need to ensure valid CSS floating point values |
+ // (https://bugs.webkit.org/show_bug.cgi?id=20674) |
+ if (matrix_->IsAffine()) { |
+ return String::Format("matrix(%g, %g, %g, %g, %g, %g)", matrix_->A(), |
+ matrix_->B(), matrix_->C(), matrix_->D(), |
+ matrix_->E(), matrix_->F()); |
+ } |
+ return String::Format( |
+ "matrix3d(%g, %g, %g, %g, %g, %g, %g, %g, %g, %g, %g, %g, %g, %g, %g, " |
+ "%g)", |
+ matrix_->M11(), matrix_->M12(), matrix_->M13(), matrix_->M14(), |
+ matrix_->M21(), matrix_->M22(), matrix_->M23(), matrix_->M24(), |
+ matrix_->M31(), matrix_->M32(), matrix_->M33(), matrix_->M34(), |
+ matrix_->M41(), matrix_->M42(), matrix_->M43(), matrix_->M44()); |
+} |
+ |
+} // namespace blink |