mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 17:39:27 +00:00 
			
		
		
		
	This was almost a no-op, except we intern JS exception messages. So the bulk of this patch is porting exception messages to UTF-16.
		
			
				
	
	
		
			1000 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1000 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | ||
|  * Copyright (c) 2023, Luke Wilde <lukew@serenityos.org>
 | ||
|  * Copyright (c) 2023, Bastiaan van der Plaat <bastiaan.v.d.plaat@gmail.com>
 | ||
|  * Copyright (c) 2024, Kenneth Myhra <kennethmyhra@serenityos.org>
 | ||
|  *
 | ||
|  * SPDX-License-Identifier: BSD-2-Clause
 | ||
|  */
 | ||
| 
 | ||
| #include <LibJS/Runtime/TypedArray.h>
 | ||
| #include <LibWeb/Bindings/DOMMatrixReadOnlyPrototype.h>
 | ||
| #include <LibWeb/CSS/ComputedProperties.h>
 | ||
| #include <LibWeb/CSS/Parser/Parser.h>
 | ||
| #include <LibWeb/CSS/StyleValues/KeywordStyleValue.h>
 | ||
| #include <LibWeb/CSS/StyleValues/ShorthandStyleValue.h>
 | ||
| #include <LibWeb/Geometry/DOMMatrix.h>
 | ||
| #include <LibWeb/Geometry/DOMMatrixReadOnly.h>
 | ||
| #include <LibWeb/Geometry/DOMPoint.h>
 | ||
| #include <LibWeb/HTML/StructuredSerialize.h>
 | ||
| #include <LibWeb/HTML/Window.h>
 | ||
| #include <LibWeb/WebIDL/Buffers.h>
 | ||
| #include <LibWeb/WebIDL/ExceptionOr.h>
 | ||
| 
 | ||
| namespace Web::Geometry {
 | ||
| 
 | ||
| GC_DEFINE_ALLOCATOR(DOMMatrixReadOnly);
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-dommatrixreadonly
 | ||
| WebIDL::ExceptionOr<GC::Ref<DOMMatrixReadOnly>> DOMMatrixReadOnly::construct_impl(JS::Realm& realm, Optional<Variant<String, Vector<double>>> const& init)
 | ||
| {
 | ||
|     auto& vm = realm.vm();
 | ||
| 
 | ||
|     // -> If init is omitted
 | ||
|     if (!init.has_value()) {
 | ||
|         // Return the result of invoking create a 2d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with the sequence [1, 0, 0, 1, 0, 0].
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm, 1, 0, 0, 1, 0, 0);
 | ||
|     }
 | ||
| 
 | ||
|     auto const& init_value = init.value();
 | ||
| 
 | ||
|     // -> If init is a DOMString
 | ||
|     if (init_value.has<String>()) {
 | ||
|         // 1. If current global object is not a Window object, then throw a TypeError exception.
 | ||
|         if (!is<HTML::Window>(realm.global_object()))
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "This can only be used in a Window context"_string };
 | ||
| 
 | ||
|         // 2. Parse init into an abstract matrix, and let matrix and 2dTransform be the result. If the result is failure, then throw a "SyntaxError" DOMException.
 | ||
|         auto result = TRY(parse_dom_matrix_init_string(realm, init_value.get<String>()));
 | ||
|         auto& m = result.matrix;
 | ||
| 
 | ||
|         // If 2dTransform is true
 | ||
|         if (result.is_2d_transform) {
 | ||
|             // Return the result of invoking create a 2d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers, the values being the elements m11, m12, m21, m22, m41 and m42 of matrix.
 | ||
|             return realm.create<DOMMatrixReadOnly>(realm, m[0, 0], m[1, 0], m[0, 1], m[1, 1], m[0, 3], m[1, 3]);
 | ||
|         }
 | ||
| 
 | ||
|         // Otherwise, return the result of invoking create a 3d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers, the values being the 16 elements of matrix.
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm,
 | ||
|             m[0, 0], m[1, 0], m[2, 0], m[3, 0],
 | ||
|             m[0, 1], m[1, 1], m[2, 1], m[3, 1],
 | ||
|             m[0, 2], m[1, 2], m[2, 2], m[3, 2],
 | ||
|             m[0, 3], m[1, 3], m[2, 3], m[3, 3]);
 | ||
|     }
 | ||
| 
 | ||
|     auto const& double_sequence = init_value.get<Vector<double>>();
 | ||
| 
 | ||
|     // -> If init is a sequence with 6 elements
 | ||
|     if (double_sequence.size() == 6) {
 | ||
|         // Return the result of invoking create a 2d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with the sequence init.
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm, double_sequence[0], double_sequence[1], double_sequence[2], double_sequence[3], double_sequence[4], double_sequence[5]);
 | ||
|     }
 | ||
| 
 | ||
|     // -> If init is a sequence with 16 elements
 | ||
|     if (double_sequence.size() == 16) {
 | ||
|         // Return the result of invoking create a 3d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with the sequence init.
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm,
 | ||
|             double_sequence[0], double_sequence[1], double_sequence[2], double_sequence[3],
 | ||
|             double_sequence[4], double_sequence[5], double_sequence[6], double_sequence[7],
 | ||
|             double_sequence[8], double_sequence[9], double_sequence[10], double_sequence[11],
 | ||
|             double_sequence[12], double_sequence[13], double_sequence[14], double_sequence[15]);
 | ||
|     }
 | ||
| 
 | ||
|     // -> Otherwise, throw a TypeError exception.
 | ||
|     return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, TRY_OR_THROW_OOM(vm, String::formatted("Sequence must contain exactly 6 or 16 elements, got {} element(s)", double_sequence.size())) };
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#create-a-dommatrixreadonly-from-the-2d-dictionary
 | ||
| WebIDL::ExceptionOr<GC::Ref<DOMMatrixReadOnly>> DOMMatrixReadOnly::create_from_dom_matrix_2d_init(JS::Realm& realm, DOMMatrix2DInit& init)
 | ||
| {
 | ||
|     // 1. Validate and fixup (2D) other.
 | ||
|     TRY(validate_and_fixup_dom_matrix_2d_init(init));
 | ||
| 
 | ||
|     // These should all have values after calling `validate_and_fixup_dom_matrix_2d_init`
 | ||
|     VERIFY(init.m11.has_value());
 | ||
|     VERIFY(init.m12.has_value());
 | ||
|     VERIFY(init.m21.has_value());
 | ||
|     VERIFY(init.m22.has_value());
 | ||
|     VERIFY(init.m41.has_value());
 | ||
|     VERIFY(init.m42.has_value());
 | ||
| 
 | ||
|     // 2. Return the result of invoking create a 2d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers,
 | ||
|     //    the values being the 6 elements m11, m12, m21, m22, m41 and m42 of other in the given order.
 | ||
|     return realm.create<DOMMatrixReadOnly>(realm, init.m11.value(), init.m12.value(), init.m21.value(), init.m22.value(), init.m41.value(), init.m42.value());
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#create-a-dommatrixreadonly-from-the-dictionary
 | ||
| WebIDL::ExceptionOr<GC::Ref<DOMMatrixReadOnly>> DOMMatrixReadOnly::create_from_dom_matrix_init(JS::Realm& realm, DOMMatrixInit& init)
 | ||
| {
 | ||
|     // 1. Validate and fixup other.
 | ||
|     TRY(validate_and_fixup_dom_matrix_init(init));
 | ||
| 
 | ||
|     // 2. If the is2D dictionary member of other is true.
 | ||
|     if (init.is2d.has_value() && init.is2d.value()) {
 | ||
|         // Return the result of invoking create a 2d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers, the values being the 6 elements m11, m12, m21, m22, m41 and m42 of other in the given order.
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm, init.m11.value(), init.m12.value(), init.m21.value(), init.m22.value(), init.m41.value(), init.m42.value());
 | ||
|     }
 | ||
| 
 | ||
|     // Otherwise, Return the result of invoking create a 3d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers, the values being the 16 elements m11, m12, m13, ..., m44 of other in the given order.
 | ||
|     return realm.create<DOMMatrixReadOnly>(realm, init.m11.value(), init.m12.value(), init.m13, init.m14,
 | ||
|         init.m21.value(), init.m22.value(), init.m23, init.m24,
 | ||
|         init.m31, init.m32, init.m33, init.m34,
 | ||
|         init.m41.value(), init.m42.value(), init.m43, init.m44);
 | ||
| }
 | ||
| 
 | ||
| GC::Ref<DOMMatrixReadOnly> DOMMatrixReadOnly::create(JS::Realm& realm)
 | ||
| {
 | ||
|     return realm.create<DOMMatrixReadOnly>(realm);
 | ||
| }
 | ||
| 
 | ||
| DOMMatrixReadOnly::DOMMatrixReadOnly(JS::Realm& realm, double m11, double m12, double m21, double m22, double m41, double m42)
 | ||
|     : Bindings::PlatformObject(realm)
 | ||
| {
 | ||
|     initialize_from_create_2d_matrix(m11, m12, m21, m22, m41, m42);
 | ||
| }
 | ||
| 
 | ||
| DOMMatrixReadOnly::DOMMatrixReadOnly(JS::Realm& realm, double m11, double m12, double m13, double m14, double m21, double m22, double m23, double m24, double m31, double m32, double m33, double m34, double m41, double m42, double m43, double m44)
 | ||
|     : Bindings::PlatformObject(realm)
 | ||
| {
 | ||
|     initialize_from_create_3d_matrix(m11, m12, m13, m14, m21, m22, m23, m24, m31, m32, m33, m34, m41, m42, m43, m44);
 | ||
| }
 | ||
| 
 | ||
| DOMMatrixReadOnly::DOMMatrixReadOnly(JS::Realm& realm, DOMMatrixReadOnly const& other)
 | ||
|     : Bindings::PlatformObject(realm)
 | ||
|     , m_matrix(other.m_matrix)
 | ||
|     , m_is_2d(other.m_is_2d)
 | ||
| {
 | ||
| }
 | ||
| 
 | ||
| DOMMatrixReadOnly::DOMMatrixReadOnly(JS::Realm& realm)
 | ||
|     : Bindings::PlatformObject(realm)
 | ||
| {
 | ||
| }
 | ||
| 
 | ||
| DOMMatrixReadOnly::~DOMMatrixReadOnly() = default;
 | ||
| 
 | ||
| void DOMMatrixReadOnly::initialize(JS::Realm& realm)
 | ||
| {
 | ||
|     WEB_SET_PROTOTYPE_FOR_INTERFACE(DOMMatrixReadOnly);
 | ||
|     Base::initialize(realm);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#create-a-2d-matrix
 | ||
| void DOMMatrixReadOnly::initialize_from_create_2d_matrix(double m11, double m12, double m21, double m22, double m41, double m42)
 | ||
| {
 | ||
|     // NOTE: The matrix used in the spec is column-major (https://drafts.fxtf.org/geometry/#4x4-abstract-matrix) but Gfx::Matrix4x4 is row-major so we need to transpose the values.
 | ||
| 
 | ||
|     // 1. Let matrix be a new instance of type.
 | ||
|     // 2. Set m11 element, m12 element, m21 element, m22 element, m41 element and m42 element to the values of init in order starting with the first value.
 | ||
|     m_matrix[0, 0] = m11;
 | ||
|     m_matrix[1, 0] = m12;
 | ||
|     m_matrix[0, 1] = m21;
 | ||
|     m_matrix[1, 1] = m22;
 | ||
|     m_matrix[0, 3] = m41;
 | ||
|     m_matrix[1, 3] = m42;
 | ||
| 
 | ||
|     // 3. Set m13 element, m14 element, m23 element, m24 element, m31 element, m32 element, m34 element, and m43 element to 0.
 | ||
|     m_matrix[2, 0] = 0.0;
 | ||
|     m_matrix[3, 0] = 0.0;
 | ||
|     m_matrix[2, 1] = 0.0;
 | ||
|     m_matrix[3, 1] = 0.0;
 | ||
|     m_matrix[0, 2] = 0.0;
 | ||
|     m_matrix[1, 2] = 0.0;
 | ||
|     m_matrix[3, 2] = 0.0;
 | ||
|     m_matrix[2, 3] = 0.0;
 | ||
| 
 | ||
|     // 4. Set m33 element and m44 element to 1.
 | ||
|     m_matrix[2, 2] = 1.0;
 | ||
|     m_matrix[3, 3] = 1.0;
 | ||
| 
 | ||
|     // 5. Set is 2D to true.
 | ||
|     m_is_2d = true;
 | ||
| 
 | ||
|     // 6. Return matrix
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#create-a-3d-matrix
 | ||
| void DOMMatrixReadOnly::initialize_from_create_3d_matrix(double m11, double m12, double m13, double m14, double m21, double m22, double m23, double m24, double m31, double m32, double m33, double m34, double m41, double m42, double m43, double m44)
 | ||
| {
 | ||
|     // NOTE: The matrix used in the spec is column-major (https://drafts.fxtf.org/geometry/#4x4-abstract-matrix) but Gfx::Matrix4x4 is row-major so we need to transpose the values.
 | ||
| 
 | ||
|     // 1. Let matrix be a new instance of type.
 | ||
|     // 2. Set m11 element to m44 element to the values of init in column-major order.
 | ||
|     m_matrix[0, 0] = m11;
 | ||
|     m_matrix[1, 0] = m12;
 | ||
|     m_matrix[2, 0] = m13;
 | ||
|     m_matrix[3, 0] = m14;
 | ||
|     m_matrix[0, 1] = m21;
 | ||
|     m_matrix[1, 1] = m22;
 | ||
|     m_matrix[2, 1] = m23;
 | ||
|     m_matrix[3, 1] = m24;
 | ||
|     m_matrix[0, 2] = m31;
 | ||
|     m_matrix[1, 2] = m32;
 | ||
|     m_matrix[2, 2] = m33;
 | ||
|     m_matrix[3, 2] = m34;
 | ||
|     m_matrix[0, 3] = m41;
 | ||
|     m_matrix[1, 3] = m42;
 | ||
|     m_matrix[2, 3] = m43;
 | ||
|     m_matrix[3, 3] = m44;
 | ||
| 
 | ||
|     // 3. Set is 2D to false.
 | ||
|     m_is_2d = false;
 | ||
| 
 | ||
|     // 4. Return matrix
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-frommatrix
 | ||
| WebIDL::ExceptionOr<GC::Ref<DOMMatrixReadOnly>> DOMMatrixReadOnly::from_matrix(JS::VM& vm, DOMMatrixInit& other)
 | ||
| {
 | ||
|     return create_from_dom_matrix_init(*vm.current_realm(), other);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-fromfloat32array
 | ||
| WebIDL::ExceptionOr<GC::Ref<DOMMatrixReadOnly>> DOMMatrixReadOnly::from_float32_array(JS::VM& vm, GC::Root<WebIDL::BufferSource> const& array32)
 | ||
| {
 | ||
|     if (!is<JS::Float32Array>(*array32))
 | ||
|         return vm.throw_completion<JS::TypeError>(JS::ErrorType::NotAnObjectOfType, "Float32Array");
 | ||
| 
 | ||
|     auto& realm = *vm.current_realm();
 | ||
|     auto& float32_array = static_cast<JS::Float32Array&>(*array32->raw_object());
 | ||
|     ReadonlySpan<float> elements = float32_array.data();
 | ||
| 
 | ||
|     // If array32 has 6 elements, return the result of invoking create a 2d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers taking the values from array32 in the provided order.
 | ||
|     if (elements.size() == 6)
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm, elements.at(0), elements.at(1), elements.at(2), elements.at(3), elements.at(4), elements.at(5));
 | ||
| 
 | ||
|     // If array32 has 16 elements, return the result of invoking create a 3d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers taking the values from array32 in the provided order.
 | ||
|     if (elements.size() == 16)
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm, elements.at(0), elements.at(1), elements.at(2), elements.at(3),
 | ||
|             elements.at(4), elements.at(5), elements.at(6), elements.at(7),
 | ||
|             elements.at(8), elements.at(9), elements.at(10), elements.at(11),
 | ||
|             elements.at(12), elements.at(13), elements.at(14), elements.at(15));
 | ||
| 
 | ||
|     // Otherwise, throw a TypeError exception.
 | ||
|     return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Expected a Float32Array argument with 6 or 16 elements"_string };
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-fromfloat64array
 | ||
| WebIDL::ExceptionOr<GC::Ref<DOMMatrixReadOnly>> DOMMatrixReadOnly::from_float64_array(JS::VM& vm, GC::Root<WebIDL::BufferSource> const& array64)
 | ||
| {
 | ||
|     if (!is<JS::Float64Array>(*array64))
 | ||
|         return vm.throw_completion<JS::TypeError>(JS::ErrorType::NotAnObjectOfType, "Float64Array");
 | ||
| 
 | ||
|     auto& realm = *vm.current_realm();
 | ||
|     auto& float64_array = static_cast<JS::Float64Array&>(*array64->raw_object());
 | ||
|     ReadonlySpan<double> elements = float64_array.data();
 | ||
| 
 | ||
|     // If array64 has 6 elements, return the result of invoking create a 2d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers taking the values from array64 in the provided order.
 | ||
|     if (elements.size() == 6)
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm, elements.at(0), elements.at(1), elements.at(2), elements.at(3), elements.at(4), elements.at(5));
 | ||
| 
 | ||
|     // If array64 has 16 elements, return the result of invoking create a 3d matrix of type DOMMatrixReadOnly or DOMMatrix as appropriate, with a sequence of numbers taking the values from array64 in the provided order.
 | ||
|     if (elements.size() == 16)
 | ||
|         return realm.create<DOMMatrixReadOnly>(realm, elements.at(0), elements.at(1), elements.at(2), elements.at(3),
 | ||
|             elements.at(4), elements.at(5), elements.at(6), elements.at(7),
 | ||
|             elements.at(8), elements.at(9), elements.at(10), elements.at(11),
 | ||
|             elements.at(12), elements.at(13), elements.at(14), elements.at(15));
 | ||
| 
 | ||
|     // Otherwise, throw a TypeError exception.
 | ||
|     return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Expected a Float64Array argument with 6 or 16 elements"_string };
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-isidentity
 | ||
| bool DOMMatrixReadOnly::is_identity() const
 | ||
| {
 | ||
|     // The isIdentity attribute must return true if
 | ||
|     // m12 element, m13 element, m14 element,
 | ||
|     // m21 element, m23 element, m24 element,
 | ||
|     // m31 element, m32 element, m34 element
 | ||
|     // m41 element, m42 element, m43 element
 | ||
|     // are 0 or -0 and
 | ||
|     // m11 element, m22 element, m33 element, m44 element are 1.
 | ||
|     // Otherwise it must return false.
 | ||
|     if (m12() != 0.0 && m12() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m13() != 0.0 && m13() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m14() != 0.0 && m14() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m21() != 0.0 && m21() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m23() != 0.0 && m24() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m31() != 0.0 && m32() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m34() != 0.0 && m34() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m41() != 0.0 && m42() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m43() != 0.0 && m43() != -0.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m11() != 1.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m22() != 1.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m33() != 1.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     if (m44() != 1.0)
 | ||
|         return false;
 | ||
| 
 | ||
|     return true;
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-translate
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::translate(Optional<double> const& tx, Optional<double> const& ty, Optional<double> const& tz) const
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a translateSelf() transformation on result with the arguments tx, ty, tz.
 | ||
|     // 3. Return result.
 | ||
|     return result->translate_self(tx, ty, tz);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-scale
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::scale(Optional<double> scale_x, Optional<double> scale_y, Optional<double> scale_z, Optional<double> origin_x, Optional<double> origin_y, Optional<double> origin_z)
 | ||
| {
 | ||
|     // 1. If scaleY is missing, set scaleY to the value of scaleX.
 | ||
|     if (!scale_y.has_value())
 | ||
|         scale_y = scale_x;
 | ||
| 
 | ||
|     // 2. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 3. Perform a scaleSelf() transformation on result with the arguments scaleX, scaleY, scaleZ, originX, originY, originZ.
 | ||
|     // 4. Return result.
 | ||
|     return result->scale_self(scale_x, scale_y, scale_z, origin_x, origin_y, origin_z);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-scalenonuniform
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::scale_non_uniform(Optional<double> scale_x, Optional<double> scale_y)
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a scaleSelf() transformation on result with the arguments scaleX, scaleY, 1, 0, 0, 0.
 | ||
|     // 3. Return result.
 | ||
|     return result->scale_self(scale_x, scale_y, 1, 0, 0, 0);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-scale3d
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::scale3d(Optional<double> scale, Optional<double> origin_x, Optional<double> origin_y, Optional<double> origin_z)
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a scale3dSelf() transformation on result with the arguments scale, originX, originY, originZ.
 | ||
|     // 3. Return result.
 | ||
|     return result->scale3d_self(scale, origin_x, origin_y, origin_z);
 | ||
| }
 | ||
| 
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::rotate(Optional<double> rot_x, Optional<double> rot_y, Optional<double> rot_z)
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a rotateSelf() transformation on result with the arguments rotX, rotY, rotZ.
 | ||
|     // 3. Return result.
 | ||
|     return result->rotate_self(rot_x, rot_y, rot_z);
 | ||
| }
 | ||
| 
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::rotate_from_vector(Optional<double> x, Optional<double> y)
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a rotateFromVectorSelf() transformation on result with the arguments x, y.
 | ||
|     // 3. Return result.
 | ||
|     return result->rotate_from_vector_self(x, y);
 | ||
| }
 | ||
| 
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::rotate_axis_angle(Optional<double> x, Optional<double> y, Optional<double> z, Optional<double> angle)
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a rotateAxisAngleSelf() transformation on result with the arguments x, y, z, angle.
 | ||
|     // 3. Return result.
 | ||
|     return result->rotate_axis_angle_self(x, y, z, angle);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-skewx
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::skew_x(double sx) const
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a skewXSelf() transformation on result with the argument sx.
 | ||
|     // 3. Return result.
 | ||
|     return result->skew_x_self(sx);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-skewy
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::skew_y(double sy) const
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a skewYSelf() transformation on result with the argument sy.
 | ||
|     // 3. Return result.
 | ||
|     return result->skew_y_self(sy);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-multiply
 | ||
| WebIDL::ExceptionOr<GC::Ref<DOMMatrix>> DOMMatrixReadOnly::multiply(DOMMatrixInit other)
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a multiplySelf() transformation on result with the argument other.
 | ||
|     // 3. Return result.
 | ||
|     return result->multiply_self(other);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-flipx
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::flip_x()
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Post-multiply result with new DOMMatrix([-1, 0, 0, 1, 0, 0]).
 | ||
|     Gfx::DoubleMatrix4x4 flip_matrix = {
 | ||
|         -1, 0, 0, 0,
 | ||
|         0, 1, 0, 0,
 | ||
|         0, 0, 1, 0,
 | ||
|         0, 0, 0, 1
 | ||
|     };
 | ||
|     result->m_matrix = result->m_matrix * flip_matrix;
 | ||
| 
 | ||
|     // 3. Return result.
 | ||
|     return result;
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-flipy
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::flip_y()
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Post-multiply result with new DOMMatrix([1, 0, 0, -1, 0, 0]).
 | ||
|     Gfx::DoubleMatrix4x4 flip_matrix = {
 | ||
|         1, 0, 0, 0,
 | ||
|         0, -1, 0, 0,
 | ||
|         0, 0, 1, 0,
 | ||
|         0, 0, 0, 1
 | ||
|     };
 | ||
|     result->m_matrix = result->m_matrix * flip_matrix;
 | ||
| 
 | ||
|     // 3. Return result.
 | ||
|     return result;
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-inverse
 | ||
| GC::Ref<DOMMatrix> DOMMatrixReadOnly::inverse() const
 | ||
| {
 | ||
|     // 1. Let result be the resulting matrix initialized to the values of the current matrix.
 | ||
|     auto result = DOMMatrix::create_from_dom_matrix_read_only(realm(), *this);
 | ||
| 
 | ||
|     // 2. Perform a invertSelf() transformation on result.
 | ||
|     // 3. Return result.
 | ||
|     // The current matrix is not modified.
 | ||
|     return result->invert_self();
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-transformpoint
 | ||
| GC::Ref<DOMPoint> DOMMatrixReadOnly::transform_point(DOMPointInit const& point) const
 | ||
| {
 | ||
|     // Let pointObject be the result of invoking create a DOMPoint from the dictionary point.
 | ||
|     auto point_object = DOMPoint::from_point(realm().vm(), point);
 | ||
| 
 | ||
|     // Return the result of invoking transform a point with a matrix, given pointObject and the current matrix. The passed argument does not get modified.
 | ||
|     return transform_point(point_object);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#transform-a-point-with-a-matrix
 | ||
| GC::Ref<DOMPoint> DOMMatrixReadOnly::transform_point(DOMPointReadOnly const& point) const
 | ||
| {
 | ||
|     // 1. Let x be point’s x coordinate.
 | ||
|     // 2. Let y be point’s y coordinate.
 | ||
|     // 3. Let z be point’s z coordinate.
 | ||
|     // 4. Let w be point’s w perspective.
 | ||
|     // 5. Let pointVector be a new column vector with the elements being x, y, z, and w, respectively.
 | ||
|     Vector4<double> point_vector { point.x(), point.y(), point.z(), point.w() };
 | ||
| 
 | ||
|     // 6. Set pointVector to pointVector pre-multiplied by matrix.
 | ||
|     // This is really a post multiply because of the transposed m_matrix.
 | ||
|     point_vector = m_matrix * point_vector;
 | ||
| 
 | ||
|     // 7. Let transformedPoint be a new DOMPoint object.
 | ||
|     // 8. Set transformedPoint’s x coordinate to pointVector’s first element.
 | ||
|     // 9. Set transformedPoint’s y coordinate to pointVector’s second element.
 | ||
|     // 10. Set transformedPoint’s z coordinate to pointVector’s third element.
 | ||
|     // 11. Set transformedPoint’s w perspective to pointVector’s fourth element.
 | ||
|     // 12. Return transformedPoint.
 | ||
|     return DOMPoint::construct_impl(realm(), point_vector.x(), point_vector.y(), point_vector.z(), point_vector.w());
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-tofloat32array
 | ||
| GC::Ref<JS::Float32Array> DOMMatrixReadOnly::to_float32_array() const
 | ||
| {
 | ||
|     // Returns the serialized 16 elements m11 to m44 of the current matrix in column-major order as Float32Array.
 | ||
|     float elements[16] = { static_cast<float>(m11()), static_cast<float>(m12()), static_cast<float>(m13()), static_cast<float>(m14()),
 | ||
|         static_cast<float>(m21()), static_cast<float>(m22()), static_cast<float>(m23()), static_cast<float>(m24()),
 | ||
|         static_cast<float>(m31()), static_cast<float>(m32()), static_cast<float>(m33()), static_cast<float>(m34()),
 | ||
|         static_cast<float>(m41()), static_cast<float>(m42()), static_cast<float>(m43()), static_cast<float>(m44()) };
 | ||
|     auto bytes = MUST(ByteBuffer::copy(elements, sizeof(elements)));
 | ||
|     auto array_buffer = JS::ArrayBuffer::create(realm(), move(bytes));
 | ||
|     return JS::Float32Array::create(realm(), sizeof(elements) / sizeof(float), array_buffer);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-tofloat64array
 | ||
| GC::Ref<JS::Float64Array> DOMMatrixReadOnly::to_float64_array() const
 | ||
| {
 | ||
|     // Returns the serialized 16 elements m11 to m44 of the current matrix in column-major order as Float64Array.
 | ||
|     double elements[16] = { m11(), m12(), m13(), m14(),
 | ||
|         m21(), m22(), m23(), m24(),
 | ||
|         m31(), m32(), m33(), m34(),
 | ||
|         m41(), m42(), m43(), m44() };
 | ||
|     auto bytes = MUST(ByteBuffer::copy(elements, sizeof(elements)));
 | ||
|     auto array_buffer = JS::ArrayBuffer::create(realm(), move(bytes));
 | ||
|     return JS::Float64Array::create(realm(), sizeof(elements) / sizeof(double), array_buffer);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#dommatrixreadonly-stringification-behavior
 | ||
| WebIDL::ExceptionOr<String> DOMMatrixReadOnly::to_string() const
 | ||
| {
 | ||
|     auto& vm = this->vm();
 | ||
| 
 | ||
|     // 1. If one or more of m11 element through m44 element are a non-finite value, then throw an "InvalidStateError" DOMException.
 | ||
|     // Spec Note: The CSS syntax cannot represent NaN or Infinity values.
 | ||
|     if (!isfinite(m11()) || !isfinite(m12()) || !isfinite(m13()) || !isfinite(m14())
 | ||
|         || !isfinite(m21()) || !isfinite(m22()) || !isfinite(m23()) || !isfinite(m24())
 | ||
|         || !isfinite(m31()) || !isfinite(m32()) || !isfinite(m33()) || !isfinite(m34())
 | ||
|         || !isfinite(m41()) || !isfinite(m42()) || !isfinite(m43()) || !isfinite(m44())) {
 | ||
|         return WebIDL::InvalidStateError::create(realm(), "Cannot stringify non-finite matrix values"_utf16);
 | ||
|     }
 | ||
| 
 | ||
|     // 2. Let string be the empty string.
 | ||
|     StringBuilder builder;
 | ||
| 
 | ||
|     // 3. If is 2D is true, then:
 | ||
|     if (m_is_2d) {
 | ||
|         // 1. Append "matrix(" to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append("matrix("sv));
 | ||
| 
 | ||
|         // 2. Append ! ToString(m11 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::Value(m11()).to_string_without_side_effects()));
 | ||
| 
 | ||
|         // 3. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 4. Append ! ToString(m12 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::Value(m12()).to_string_without_side_effects()));
 | ||
| 
 | ||
|         // 5. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 6. Append ! ToString(m21 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::Value(m21()).to_string_without_side_effects()));
 | ||
| 
 | ||
|         // 7. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 8. Append ! ToString(m22 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::Value(m22()).to_string_without_side_effects()));
 | ||
| 
 | ||
|         // 9. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 10. Append ! ToString(m41 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::Value(m41()).to_string_without_side_effects()));
 | ||
| 
 | ||
|         // 11. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 12. Append ! ToString(m42 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::Value(m42()).to_string_without_side_effects()));
 | ||
| 
 | ||
|         // 13. Append ")" to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(")"sv));
 | ||
|     } else {
 | ||
|         // 1. Append "matrix3d(" to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append("matrix3d("sv));
 | ||
| 
 | ||
|         // 2. Append ! ToString(m11 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m11())));
 | ||
| 
 | ||
|         // 3. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 4. Append ! ToString(m12 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m12())));
 | ||
| 
 | ||
|         // 5. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 6. Append ! ToString(m13 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m13())));
 | ||
| 
 | ||
|         // 7. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 8. Append ! ToString(m14 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m14())));
 | ||
| 
 | ||
|         // 9. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 10. Append ! ToString(m21 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m21())));
 | ||
| 
 | ||
|         // 11. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 12. Append ! ToString(m22 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m22())));
 | ||
| 
 | ||
|         // 13. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 14. Append ! ToString(m23 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m23())));
 | ||
| 
 | ||
|         // 15. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 16. Append ! ToString(m24 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m24())));
 | ||
| 
 | ||
|         // 17. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // NOTE: The spec doesn't include the steps to append m31 to m34, but they are required as matrix3d requires 16 elements.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m31())));
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m32())));
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m33())));
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m34())));
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 18. Append ! ToString(m41 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m41())));
 | ||
| 
 | ||
|         // 19. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 20. Append ! ToString(m42 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m42())));
 | ||
| 
 | ||
|         // 21. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 22. Append ! ToString(m43 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m43())));
 | ||
| 
 | ||
|         // 23. Append ", " to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(", "sv));
 | ||
| 
 | ||
|         // 24. Append ! ToString(m44 element) to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(JS::number_to_string(m44())));
 | ||
| 
 | ||
|         // 25. Append ")" to string.
 | ||
|         TRY_OR_THROW_OOM(vm, builder.try_append(")"sv));
 | ||
|     }
 | ||
| 
 | ||
|     // 5. Return string.
 | ||
|     return TRY_OR_THROW_OOM(vm, builder.to_string());
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#structured-serialization
 | ||
| WebIDL::ExceptionOr<void> DOMMatrixReadOnly::serialization_steps(HTML::TransferDataEncoder& serialized, bool, HTML::SerializationMemory&)
 | ||
| {
 | ||
|     serialized.encode(m_is_2d);
 | ||
| 
 | ||
|     // 1. If value’s is 2D is true:
 | ||
|     if (m_is_2d) {
 | ||
|         // 1. Set serialized.[[M11]] to value’s m11 element.
 | ||
|         serialized.encode(m11());
 | ||
|         // 2. Set serialized.[[M12]] to value’s m12 element.
 | ||
|         serialized.encode(m12());
 | ||
|         // 3. Set serialized.[[M21]] to value’s m21 element.
 | ||
|         serialized.encode(m21());
 | ||
|         // 4. Set serialized.[[M22]] to value’s m22 element.
 | ||
|         serialized.encode(m22());
 | ||
|         // 5. Set serialized.[[M41]] to value’s m41 element.
 | ||
|         serialized.encode(m41());
 | ||
|         // 6. Set serialized.[[M42]] to value’s m42 element.
 | ||
|         serialized.encode(m42());
 | ||
|         // 7. Set serialized.[[Is2D]] to true.
 | ||
|         // NOTE: This is set in the beginning of the function.
 | ||
|     }
 | ||
|     // 2. Otherwise:
 | ||
|     else {
 | ||
|         // 1. Set serialized.[[M11]] to value’s m11 element.
 | ||
|         serialized.encode(m11());
 | ||
|         // 2. Set serialized.[[M12]] to value’s m12 element.
 | ||
|         serialized.encode(m12());
 | ||
|         // 3. Set serialized.[[M13]] to value’s m13 element.
 | ||
|         serialized.encode(m13());
 | ||
|         // 4. Set serialized.[[M14]] to value’s m14 element.
 | ||
|         serialized.encode(m14());
 | ||
|         // 5. Set serialized.[[M21]] to value’s m21 element.
 | ||
|         serialized.encode(m21());
 | ||
|         // 6. Set serialized.[[M22]] to value’s m22 element.
 | ||
|         serialized.encode(m22());
 | ||
|         // 7. Set serialized.[[M23]] to value’s m23 element.
 | ||
|         serialized.encode(m23());
 | ||
|         // 8. Set serialized.[[M24]] to value’s m24 element.
 | ||
|         serialized.encode(m24());
 | ||
|         // 9. Set serialized.[[M31]] to value’s m31 element.
 | ||
|         serialized.encode(m31());
 | ||
|         // 10. Set serialized.[[M32]] to value’s m32 element.
 | ||
|         serialized.encode(m32());
 | ||
|         // 11. Set serialized.[[M33]] to value’s m33 element.
 | ||
|         serialized.encode(m33());
 | ||
|         // 12. Set serialized.[[M34]] to value’s m34 element.
 | ||
|         serialized.encode(m34());
 | ||
|         // 13. Set serialized.[[M41]] to value’s m41 element.
 | ||
|         serialized.encode(m41());
 | ||
|         // 14. Set serialized.[[M42]] to value’s m42 element.
 | ||
|         serialized.encode(m42());
 | ||
|         // 15. Set serialized.[[M43]] to value’s m43 element.
 | ||
|         serialized.encode(m43());
 | ||
|         // 16. Set serialized.[[M44]] to value’s m44 element.
 | ||
|         serialized.encode(m44());
 | ||
|         // 17. Set serialized.[[Is2D]] to false.
 | ||
|         // NOTE: This is set in the beginning of the function.
 | ||
|     }
 | ||
| 
 | ||
|     return {};
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#structured-serialization
 | ||
| WebIDL::ExceptionOr<void> DOMMatrixReadOnly::deserialization_steps(HTML::TransferDataDecoder& serialized, HTML::DeserializationMemory&)
 | ||
| {
 | ||
|     bool is_2d = serialized.decode<bool>();
 | ||
| 
 | ||
|     // 1. If serialized.[[Is2D]] is true:
 | ||
|     if (is_2d) {
 | ||
|         // 1. Set value’s m11 element to serialized.[[M11]].
 | ||
|         auto m11 = serialized.decode<double>();
 | ||
|         // 2. Set value’s m12 element to serialized.[[M12]].
 | ||
|         auto m12 = serialized.decode<double>();
 | ||
|         // 3. Set value’s m13 element to 0.
 | ||
|         // 4. Set value’s m14 element to 0.
 | ||
|         // 5. Set value’s m21 element to serialized.[[M21]].
 | ||
|         auto m21 = serialized.decode<double>();
 | ||
|         // 6. Set value’s m22 element to serialized.[[M22]].
 | ||
|         auto m22 = serialized.decode<double>();
 | ||
|         // 7. Set value’s m23 element to 0.
 | ||
|         // 8. Set value’s m24 element to 0.
 | ||
|         // 9. Set value’s m31 element to 0.
 | ||
|         // 10. Set value’s m32 element to 0.
 | ||
|         // 11. Set value’s m33 element to 1.
 | ||
|         // 12. Set value’s m34 element to 0.
 | ||
|         // 13. Set value’s m41 element to serialized.[[M41]].
 | ||
|         auto m41 = serialized.decode<double>();
 | ||
|         // 14 Set value’s m42 element to serialized.[[M42]].
 | ||
|         auto m42 = serialized.decode<double>();
 | ||
|         // 15. Set value’s m43 element to 0.
 | ||
|         // 16. Set value’s m44 element to 1.
 | ||
|         // 17. Set value’s is 2D to true.
 | ||
| 
 | ||
|         initialize_from_create_2d_matrix(m11, m12, m21, m22, m41, m42);
 | ||
|     }
 | ||
|     // 2. Otherwise:
 | ||
|     else {
 | ||
|         // 1. Set value’s m11 element to serialized.[[M11]].
 | ||
|         auto m11 = serialized.decode<double>();
 | ||
|         // 2. Set value’s m12 element to serialized.[[M12]].
 | ||
|         auto m12 = serialized.decode<double>();
 | ||
|         // 3. Set value’s m13 element to serialized.[[M13]].
 | ||
|         auto m13 = serialized.decode<double>();
 | ||
|         // 4. Set value’s m14 element to serialized.[[M14]].
 | ||
|         auto m14 = serialized.decode<double>();
 | ||
|         // 5. Set value’s m21 element to serialized.[[M21]].
 | ||
|         auto m21 = serialized.decode<double>();
 | ||
|         // 6. Set value’s m22 element to serialized.[[M22]].
 | ||
|         auto m22 = serialized.decode<double>();
 | ||
|         // 7. Set value’s m23 element to serialized.[[M23]].
 | ||
|         auto m23 = serialized.decode<double>();
 | ||
|         // 8. Set value’s m24 element to serialized.[[M24]].
 | ||
|         auto m24 = serialized.decode<double>();
 | ||
|         // 9. Set value’s m31 element to serialized.[[M31]].
 | ||
|         auto m31 = serialized.decode<double>();
 | ||
|         // 10. Set value’s m32 element to serialized.[[M32]].
 | ||
|         auto m32 = serialized.decode<double>();
 | ||
|         // 11. Set value’s m33 element to serialized.[[M33]].
 | ||
|         auto m33 = serialized.decode<double>();
 | ||
|         // 12. Set value’s m34 element to serialized.[[M34]].
 | ||
|         auto m34 = serialized.decode<double>();
 | ||
|         // 13. Set value’s m41 element to serialized.[[M41]].
 | ||
|         auto m41 = serialized.decode<double>();
 | ||
|         // 14. Set value’s m42 element to serialized.[[M42]].
 | ||
|         auto m42 = serialized.decode<double>();
 | ||
|         // 15. Set value’s m43 element to serialized.[[M43]].
 | ||
|         auto m43 = serialized.decode<double>();
 | ||
|         // 16. Set value’s m44 element to serialized.[[M44]].
 | ||
|         auto m44 = serialized.decode<double>();
 | ||
|         // 17. Set value’s is 2D to false.
 | ||
| 
 | ||
|         initialize_from_create_3d_matrix(m11, m12, m13, m14, m21, m22, m23, m24, m31, m32, m33, m34, m41, m42, m43, m44);
 | ||
|     }
 | ||
| 
 | ||
|     return {};
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#matrix-validate-and-fixup-2d
 | ||
| WebIDL::ExceptionOr<void> validate_and_fixup_dom_matrix_2d_init(DOMMatrix2DInit& init)
 | ||
| {
 | ||
|     // 1. If at least one of the following conditions are true for dict, then throw a TypeError exception and abort these steps.
 | ||
|     // - a and m11 are both present and SameValueZero(a, m11) is false.
 | ||
|     if (init.a.has_value() && init.m11.has_value() && !JS::same_value_zero(JS::Value(init.a.value()), JS::Value(init.m11.value())))
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "DOMMatrix2DInit.a and DOMMatrix2DInit.m11 must have the same value if they are both present"sv };
 | ||
| 
 | ||
|     // - b and m12 are both present and SameValueZero(b, m12) is false.
 | ||
|     if (init.b.has_value() && init.m12.has_value() && !JS::same_value_zero(JS::Value(init.b.value()), JS::Value(init.m12.value())))
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "DOMMatrix2DInit.b and DOMMatrix2DInit.m12 must have the same value if they are both present"sv };
 | ||
| 
 | ||
|     // - c and m21 are both present and SameValueZero(c, m21) is false.
 | ||
|     if (init.c.has_value() && init.m21.has_value() && !JS::same_value_zero(JS::Value(init.c.value()), JS::Value(init.m21.value())))
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "DOMMatrix2DInit.c and DOMMatrix2DInit.m21 must have the same value if they are both present"sv };
 | ||
| 
 | ||
|     // - d and m22 are both present and SameValueZero(d, m22) is false.
 | ||
|     if (init.d.has_value() && init.m22.has_value() && !JS::same_value_zero(JS::Value(init.d.value()), JS::Value(init.m22.value())))
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "DOMMatrix2DInit.d and DOMMatrix2DInit.m22 must have the same value if they are both present"sv };
 | ||
| 
 | ||
|     // - e and m41 are both present and SameValueZero(e, m41) is false.
 | ||
|     if (init.e.has_value() && init.m41.has_value() && !JS::same_value_zero(JS::Value(init.e.value()), JS::Value(init.m41.value())))
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "DOMMatrix2DInit.e and DOMMatrix2DInit.m41 must have the same value if they are both present"sv };
 | ||
| 
 | ||
|     // - f and m42 are both present and SameValueZero(f, m42) is false.
 | ||
|     if (init.f.has_value() && init.m42.has_value() && !JS::same_value_zero(JS::Value(init.f.value()), JS::Value(init.m42.value())))
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "DOMMatrix2DInit.f and DOMMatrix2DInit.m42 must have the same value if they are both present"sv };
 | ||
| 
 | ||
|     // 2. If m11 is not present then set it to the value of member a, or value 1 if a is also not present.
 | ||
|     if (!init.m11.has_value())
 | ||
|         init.m11 = init.a.value_or(1.0);
 | ||
| 
 | ||
|     // 3. If m12 is not present then set it to the value of member b, or value 0 if b is also not present.
 | ||
|     if (!init.m12.has_value())
 | ||
|         init.m12 = init.b.value_or(0.0);
 | ||
| 
 | ||
|     // 4. If m21 is not present then set it to the value of member c, or value 0 if c is also not present.
 | ||
|     if (!init.m21.has_value())
 | ||
|         init.m21 = init.c.value_or(0.0);
 | ||
| 
 | ||
|     // 5. If m22 is not present then set it to the value of member d, or value 1 if d is also not present.
 | ||
|     if (!init.m22.has_value())
 | ||
|         init.m22 = init.d.value_or(1.0);
 | ||
| 
 | ||
|     // 6. If m41 is not present then set it to the value of member e, or value 0 if e is also not present.
 | ||
|     if (!init.m41.has_value())
 | ||
|         init.m41 = init.e.value_or(0.0);
 | ||
| 
 | ||
|     // 7. If m42 is not present then set it to the value of member f, or value 0 if f is also not present.
 | ||
|     if (!init.m42.has_value())
 | ||
|         init.m42 = init.f.value_or(0.0);
 | ||
| 
 | ||
|     return {};
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#matrix-validate-and-fixup
 | ||
| WebIDL::ExceptionOr<void> validate_and_fixup_dom_matrix_init(DOMMatrixInit& init)
 | ||
| {
 | ||
|     // 1. Validate and fixup (2D) dict.
 | ||
|     TRY(validate_and_fixup_dom_matrix_2d_init(init));
 | ||
| 
 | ||
|     // 2. If is2D is true and: at least one of m13, m14, m23, m24, m31, m32, m34, m43 are present with a value other than 0 or -0,
 | ||
|     //    or at least one of m33, m44 are present with a value other than 1, then throw a TypeError exception and abort these steps.
 | ||
|     if (init.is2d.has_value() && init.is2d.value()) {
 | ||
|         if ((init.m13 != 0.0 && init.m13 != -0.0)
 | ||
|             || (init.m14 != 0.0 && init.m14 != -0.0)
 | ||
|             || (init.m23 != 0.0 && init.m23 != -0.0)
 | ||
|             || (init.m24 != 0.0 && init.m24 != -0.0)
 | ||
|             || (init.m31 != 0.0 && init.m31 != -0.0)
 | ||
|             || (init.m32 != 0.0 && init.m32 != -0.0)
 | ||
|             || (init.m34 != 0.0 && init.m34 != -0.0)
 | ||
|             || (init.m43 != 0.0 && init.m43 != -0.0)
 | ||
|             || init.m33 != 1.0
 | ||
|             || init.m44 != 1.0) {
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "DOMMatrixInit.is2D is true, but the given matrix is not a 2D matrix"sv };
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // If is2D is not present and at least one of m13, m14, m23, m24, m31, m32, m34, m43 are present with a value other than 0 or -0,
 | ||
|     // or at least one of m33, m44 are present with a value other than 1, set is2D to false.
 | ||
|     if (!init.is2d.has_value()) {
 | ||
|         if ((init.m13 != 0.0 && init.m13 != -0.0)
 | ||
|             || (init.m14 != 0.0 && init.m14 != -0.0)
 | ||
|             || (init.m23 != 0.0 && init.m23 != -0.0)
 | ||
|             || (init.m24 != 0.0 && init.m24 != -0.0)
 | ||
|             || (init.m31 != 0.0 && init.m31 != -0.0)
 | ||
|             || (init.m32 != 0.0 && init.m32 != -0.0)
 | ||
|             || (init.m34 != 0.0 && init.m34 != -0.0)
 | ||
|             || (init.m43 != 0.0 && init.m43 != -0.0)
 | ||
|             || init.m33 != 1.0
 | ||
|             || init.m44 != 1.0) {
 | ||
|             init.is2d = false;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // 4. If is2D is still not present, set it to true.
 | ||
|     if (!init.is2d.has_value())
 | ||
|         init.is2d = true;
 | ||
| 
 | ||
|     return {};
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.fxtf.org/geometry/#parse-a-string-into-an-abstract-matrix
 | ||
| WebIDL::ExceptionOr<ParsedMatrix> parse_dom_matrix_init_string(JS::Realm& realm, StringView transform_list)
 | ||
| {
 | ||
|     // 1. If transformList is the empty string, set it to the string "matrix(1, 0, 0, 1, 0, 0)".
 | ||
|     if (transform_list.is_empty())
 | ||
|         transform_list = "matrix(1, 0, 0, 1, 0, 0)"sv;
 | ||
| 
 | ||
|     // 2. Parse transformList into parsedValue given the grammar for the CSS transform property.
 | ||
|     // The result will be a <transform-list>, the keyword none, or failure.
 | ||
|     // If parsedValue is failure, or any <transform-function> has <length> values without absolute length units, or any keyword other than none is used, then return failure. [CSS3-SYNTAX] [CSS3-TRANSFORMS]
 | ||
|     auto transform_style_value = parse_css_value(CSS::Parser::ParsingParams {}, transform_list, CSS::PropertyID::Transform);
 | ||
|     if (!transform_style_value || (transform_style_value->is_keyword() && transform_style_value->to_keyword() != CSS::Keyword::None))
 | ||
|         return WebIDL::SyntaxError::create(realm, "Failed to parse CSS transform string."_utf16);
 | ||
|     auto parsed_value = CSS::ComputedProperties::transformations_for_style_value(*transform_style_value);
 | ||
| 
 | ||
|     // 3. If parsedValue is none, set parsedValue to a <transform-list> containing a single identity matrix.
 | ||
|     // NOTE: parsed_value is empty on none so for loop in 6 won't modify matrix
 | ||
|     auto matrix = Gfx::FloatMatrix4x4::identity();
 | ||
| 
 | ||
|     // 4. Let 2dTransform track the 2D/3D dimension status of parsedValue.
 | ||
|     // -> If parsedValue consists of any three-dimensional transform functions, set 2dTransform to false.
 | ||
|     // -> Otherwise, set 2dTransform to true.
 | ||
|     bool is_2d_transform = true;
 | ||
|     for (auto const& transform : parsed_value) {
 | ||
|         // https://www.w3.org/TR/css-transforms-1/#two-d-transform-functions
 | ||
|         if (transform.function() != CSS::TransformFunction::Matrix
 | ||
|             && transform.function() != CSS::TransformFunction::Translate
 | ||
|             && transform.function() != CSS::TransformFunction::TranslateX
 | ||
|             && transform.function() != CSS::TransformFunction::TranslateY
 | ||
|             && transform.function() != CSS::TransformFunction::Scale
 | ||
|             && transform.function() != CSS::TransformFunction::ScaleX
 | ||
|             && transform.function() != CSS::TransformFunction::ScaleY
 | ||
|             && transform.function() != CSS::TransformFunction::Rotate
 | ||
|             && transform.function() != CSS::TransformFunction::Skew
 | ||
|             && transform.function() != CSS::TransformFunction::SkewX
 | ||
|             && transform.function() != CSS::TransformFunction::SkewY)
 | ||
|             is_2d_transform = false;
 | ||
|     }
 | ||
| 
 | ||
|     // 5. Transform all <transform-function>s to 4x4 abstract matrices by following the “Mathematical Description of Transform Functions”. [CSS3-TRANSFORMS]
 | ||
|     // 6. Let matrix be a 4x4 abstract matrix as shown in the initial figure of this section. Post-multiply all matrices from left to right and set matrix to this product.
 | ||
|     for (auto const& transform : parsed_value) {
 | ||
|         auto const& transform_matrix = transform.to_matrix({});
 | ||
|         if (transform_matrix.is_error())
 | ||
|             return WebIDL::SyntaxError::create(realm, "Failed to parse CSS transform string."_utf16);
 | ||
|         matrix = matrix * transform_matrix.value();
 | ||
|     }
 | ||
| 
 | ||
|     // 7. Return matrix and 2dTransform.
 | ||
|     Gfx::DoubleMatrix4x4 double_matrix {
 | ||
|         static_cast<double>(matrix[0, 0]), static_cast<double>(matrix[0, 1]), static_cast<double>(matrix[0, 2]), static_cast<double>(matrix[0, 3]),
 | ||
|         static_cast<double>(matrix[1, 0]), static_cast<double>(matrix[1, 1]), static_cast<double>(matrix[1, 2]), static_cast<double>(matrix[1, 3]),
 | ||
|         static_cast<double>(matrix[2, 0]), static_cast<double>(matrix[2, 1]), static_cast<double>(matrix[2, 2]), static_cast<double>(matrix[2, 3]),
 | ||
|         static_cast<double>(matrix[3, 0]), static_cast<double>(matrix[3, 1]), static_cast<double>(matrix[3, 2]), static_cast<double>(matrix[3, 3])
 | ||
|     };
 | ||
|     return ParsedMatrix { double_matrix, is_2d_transform };
 | ||
| }
 | ||
| 
 | ||
| }
 |