2023-02-27 13:04:51 -05:00
/*
* Copyright ( c ) 2023 , Luke Wilde < lukew @ serenityos . org >
2023-09-04 12:22:55 -04:00
* Copyright ( c ) 2023 , Bastiaan van der Plaat < bastiaan . v . d . plaat @ gmail . com >
2024-03-09 16:07:22 -05:00
* Copyright ( c ) 2024 , Kenneth Myhra < kennethmyhra @ serenityos . org >
2023-02-27 13:04:51 -05:00
*
* SPDX - License - Identifier : BSD - 2 - Clause
*/
2023-09-25 12:44:10 -04:00
# include <LibJS/Runtime/TypedArray.h>
2024-04-26 20:09:58 -04:00
# include <LibWeb/Bindings/DOMMatrixReadOnlyPrototype.h>
2024-12-20 05:32:17 -05:00
# include <LibWeb/CSS/ComputedProperties.h>
2024-01-06 12:05:21 -05:00
# include <LibWeb/CSS/Parser/Parser.h>
2024-12-26 16:32:16 -05:00
# include <LibWeb/CSS/StyleValues/CSSKeywordValue.h>
2024-01-06 12:05:21 -05:00
# include <LibWeb/CSS/StyleValues/ShorthandStyleValue.h>
2023-02-27 13:04:51 -05:00
# include <LibWeb/Geometry/DOMMatrix.h>
# include <LibWeb/Geometry/DOMMatrixReadOnly.h>
2023-08-15 07:16:45 -04:00
# include <LibWeb/Geometry/DOMPoint.h>
2024-03-09 16:07:22 -05:00
# include <LibWeb/HTML/StructuredSerialize.h>
2024-01-06 12:05:21 -05:00
# include <LibWeb/HTML/Window.h>
2023-11-23 02:07:25 -05:00
# include <LibWeb/WebIDL/Buffers.h>
2023-02-27 13:04:51 -05:00
# include <LibWeb/WebIDL/ExceptionOr.h>
namespace Web : : Geometry {
2024-11-14 10:01:23 -05:00
GC_DEFINE_ALLOCATOR ( DOMMatrixReadOnly ) ;
2024-04-06 13:16:04 -04:00
2024-01-06 12:05:21 -05:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-dommatrixreadonly
2024-11-14 10:01:23 -05:00
WebIDL : : ExceptionOr < GC : : Ref < DOMMatrixReadOnly > > DOMMatrixReadOnly : : construct_impl ( JS : : Realm & realm , Optional < Variant < String , Vector < double > > > const & init )
2023-02-27 13:04:51 -05:00
{
2023-03-03 13:04:58 -05:00
auto & vm = realm . vm ( ) ;
2024-01-06 12:05:21 -05:00
// -> 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].
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , 1 , 0 , 0 , 1 , 0 , 0 ) ;
2023-02-27 13:04:51 -05:00
}
2024-01-06 12:05:21 -05:00
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 * elements = result . matrix . elements ( ) ;
// 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.
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , elements [ 0 ] [ 0 ] , elements [ 1 ] [ 0 ] , elements [ 0 ] [ 1 ] , elements [ 1 ] [ 1 ] , elements [ 0 ] [ 3 ] , elements [ 1 ] [ 3 ] ) ;
2024-01-06 12:05:21 -05:00
}
// 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.
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm ,
2024-01-06 12:05:21 -05:00
elements [ 0 ] [ 0 ] , elements [ 1 ] [ 0 ] , elements [ 2 ] [ 0 ] , elements [ 3 ] [ 0 ] ,
elements [ 0 ] [ 1 ] , elements [ 1 ] [ 1 ] , elements [ 2 ] [ 1 ] , elements [ 3 ] [ 1 ] ,
elements [ 0 ] [ 2 ] , elements [ 1 ] [ 2 ] , elements [ 2 ] [ 2 ] , elements [ 3 ] [ 2 ] ,
elements [ 0 ] [ 3 ] , elements [ 1 ] [ 3 ] , elements [ 2 ] [ 3 ] , elements [ 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.
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , double_sequence [ 0 ] , double_sequence [ 1 ] , double_sequence [ 2 ] , double_sequence [ 3 ] , double_sequence [ 4 ] , double_sequence [ 5 ] ) ;
2024-01-06 12:05:21 -05:00
}
// -> 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.
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm ,
2024-01-06 12:05:21 -05:00
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 ( ) ) ) } ;
2023-02-27 13:04:51 -05:00
}
// https://drafts.fxtf.org/geometry/#create-a-dommatrixreadonly-from-the-2d-dictionary
2024-11-14 10:01:23 -05:00
WebIDL : : ExceptionOr < GC : : Ref < DOMMatrixReadOnly > > DOMMatrixReadOnly : : create_from_dom_matrix_2d_init ( JS : : Realm & realm , DOMMatrix2DInit & init )
2023-02-27 13:04:51 -05:00
{
// 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.
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , init . m11 . value ( ) , init . m12 . value ( ) , init . m21 . value ( ) , init . m22 . value ( ) , init . m41 . value ( ) , init . m42 . value ( ) ) ;
2023-02-27 13:04:51 -05:00
}
2023-09-07 15:26:21 -04:00
// https://drafts.fxtf.org/geometry/#create-a-dommatrixreadonly-from-the-dictionary
2024-11-14 10:01:23 -05:00
WebIDL : : ExceptionOr < GC : : Ref < DOMMatrixReadOnly > > DOMMatrixReadOnly : : create_from_dom_matrix_init ( JS : : Realm & realm , DOMMatrixInit & init )
2023-09-07 15:26:21 -04:00
{
// 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.
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , init . m11 . value ( ) , init . m12 . value ( ) , init . m21 . value ( ) , init . m22 . value ( ) , init . m41 . value ( ) , init . m42 . value ( ) ) ;
2023-09-07 15:26:21 -04:00
}
// 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.
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , init . m11 . value ( ) , init . m12 . value ( ) , init . m13 , init . m14 ,
2023-09-07 15:26:21 -04:00
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 ) ;
}
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrixReadOnly > DOMMatrixReadOnly : : create ( JS : : Realm & realm )
2024-03-09 16:07:22 -05:00
{
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm ) ;
2024-03-09 16:07:22 -05:00
}
2023-02-27 13:04:51 -05:00
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 )
{
}
2024-03-09 16:07:22 -05:00
DOMMatrixReadOnly : : DOMMatrixReadOnly ( JS : : Realm & realm )
: Bindings : : PlatformObject ( realm )
{
}
2023-02-27 13:04:51 -05:00
DOMMatrixReadOnly : : ~ DOMMatrixReadOnly ( ) = default ;
2023-08-07 02:41:28 -04:00
void DOMMatrixReadOnly : : initialize ( JS : : Realm & realm )
2023-02-27 13:04:51 -05:00
{
2023-08-07 02:41:28 -04:00
Base : : initialize ( realm ) ;
2024-03-16 08:13:08 -04:00
WEB_SET_PROTOTYPE_FOR_INTERFACE ( DOMMatrixReadOnly ) ;
2023-02-27 13:04:51 -05:00
}
// 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 )
{
2023-08-22 03:46:02 -04:00
// 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.
2023-02-27 13:04:51 -05:00
// 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.
auto * elements = m_matrix . elements ( ) ;
elements [ 0 ] [ 0 ] = m11 ;
2023-08-22 03:46:02 -04:00
elements [ 1 ] [ 0 ] = m12 ;
elements [ 0 ] [ 1 ] = m21 ;
2023-02-27 13:04:51 -05:00
elements [ 1 ] [ 1 ] = m22 ;
2023-08-22 03:46:02 -04:00
elements [ 0 ] [ 3 ] = m41 ;
elements [ 1 ] [ 3 ] = m42 ;
2023-02-27 13:04:51 -05:00
// 3. Set m13 element, m14 element, m23 element, m24 element, m31 element, m32 element, m34 element, and m43 element to 0.
elements [ 2 ] [ 0 ] = 0.0 ;
2023-08-22 03:46:02 -04:00
elements [ 3 ] [ 0 ] = 0.0 ;
2023-02-27 13:04:51 -05:00
elements [ 2 ] [ 1 ] = 0.0 ;
2023-08-22 03:46:02 -04:00
elements [ 3 ] [ 1 ] = 0.0 ;
elements [ 0 ] [ 2 ] = 0.0 ;
elements [ 1 ] [ 2 ] = 0.0 ;
2023-02-27 13:04:51 -05:00
elements [ 3 ] [ 2 ] = 0.0 ;
2023-08-22 03:46:02 -04:00
elements [ 2 ] [ 3 ] = 0.0 ;
2023-02-27 13:04:51 -05:00
// 4. Set m33 element and m44 element to 1.
elements [ 2 ] [ 2 ] = 1.0 ;
elements [ 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 )
{
2023-08-22 03:46:02 -04:00
// 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.
2023-02-27 13:04:51 -05:00
// 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.
auto * elements = m_matrix . elements ( ) ;
elements [ 0 ] [ 0 ] = m11 ;
2023-08-22 03:46:02 -04:00
elements [ 1 ] [ 0 ] = m12 ;
elements [ 2 ] [ 0 ] = m13 ;
elements [ 3 ] [ 0 ] = m14 ;
elements [ 0 ] [ 1 ] = m21 ;
2023-02-27 13:04:51 -05:00
elements [ 1 ] [ 1 ] = m22 ;
2023-08-22 03:46:02 -04:00
elements [ 2 ] [ 1 ] = m23 ;
elements [ 3 ] [ 1 ] = m24 ;
elements [ 0 ] [ 2 ] = m31 ;
elements [ 1 ] [ 2 ] = m32 ;
2023-02-27 13:04:51 -05:00
elements [ 2 ] [ 2 ] = m33 ;
2023-08-22 03:46:02 -04:00
elements [ 3 ] [ 2 ] = m34 ;
elements [ 0 ] [ 3 ] = m41 ;
elements [ 1 ] [ 3 ] = m42 ;
elements [ 2 ] [ 3 ] = m43 ;
2023-02-27 13:04:51 -05:00
elements [ 3 ] [ 3 ] = m44 ;
// 3. Set is 2D to false.
m_is_2d = false ;
// 4. Return matrix
}
2023-09-03 06:37:29 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-frommatrix
2024-11-14 10:01:23 -05:00
WebIDL : : ExceptionOr < GC : : Ref < DOMMatrixReadOnly > > DOMMatrixReadOnly : : from_matrix ( JS : : VM & vm , DOMMatrixInit & other )
2023-09-03 06:37:29 -04:00
{
2023-09-07 15:26:21 -04:00
return create_from_dom_matrix_init ( * vm . current_realm ( ) , other ) ;
2023-09-03 06:37:29 -04:00
}
2023-09-25 16:11:30 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-fromfloat32array
2024-11-14 10:01:23 -05:00
WebIDL : : ExceptionOr < GC : : Ref < DOMMatrixReadOnly > > DOMMatrixReadOnly : : from_float32_array ( JS : : VM & vm , GC : : Root < WebIDL : : BufferSource > const & array32 )
2023-09-25 16:11:30 -04:00
{
if ( ! is < JS : : Float32Array > ( * array32 ) )
return vm . throw_completion < JS : : TypeError > ( JS : : ErrorType : : NotAnObjectOfType , " Float32Array " ) ;
auto & realm = * vm . current_realm ( ) ;
2023-11-23 02:07:25 -05:00
auto & float32_array = static_cast < JS : : Float32Array & > ( * array32 - > raw_object ( ) ) ;
2023-09-25 16:11:30 -04:00
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 )
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , elements . at ( 0 ) , elements . at ( 1 ) , elements . at ( 2 ) , elements . at ( 3 ) , elements . at ( 4 ) , elements . at ( 5 ) ) ;
2023-09-25 16:11:30 -04:00
// 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 )
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , elements . at ( 0 ) , elements . at ( 1 ) , elements . at ( 2 ) , elements . at ( 3 ) ,
2023-09-25 16:11:30 -04:00
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
2024-11-14 10:01:23 -05:00
WebIDL : : ExceptionOr < GC : : Ref < DOMMatrixReadOnly > > DOMMatrixReadOnly : : from_float64_array ( JS : : VM & vm , GC : : Root < WebIDL : : BufferSource > const & array64 )
2023-09-25 16:11:30 -04:00
{
if ( ! is < JS : : Float64Array > ( * array64 ) )
return vm . throw_completion < JS : : TypeError > ( JS : : ErrorType : : NotAnObjectOfType , " Float64Array " ) ;
auto & realm = * vm . current_realm ( ) ;
2023-11-23 02:07:25 -05:00
auto & float64_array = static_cast < JS : : Float64Array & > ( * array64 - > raw_object ( ) ) ;
2023-09-25 16:11:30 -04:00
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 )
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , elements . at ( 0 ) , elements . at ( 1 ) , elements . at ( 2 ) , elements . at ( 3 ) , elements . at ( 4 ) , elements . at ( 5 ) ) ;
2023-09-25 16:11:30 -04:00
// 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 )
2024-11-13 11:50:17 -05:00
return realm . create < DOMMatrixReadOnly > ( realm , elements . at ( 0 ) , elements . at ( 1 ) , elements . at ( 2 ) , elements . at ( 3 ) ,
2023-09-25 16:11:30 -04:00
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 } ;
}
2023-02-27 13:04:51 -05:00
// 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 ;
}
2023-09-03 06:43:09 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-translate
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : translate ( Optional < double > const & tx , Optional < double > const & ty , Optional < double > const & tz ) const
2023-09-03 06:43:09 -04:00
{
// 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 ) ;
}
2023-09-07 15:27:34 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-scale
2024-11-14 10:01:23 -05:00
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 )
2023-09-07 15:27:34 -04:00
{
// 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
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : scale_non_uniform ( Optional < double > scale_x , Optional < double > scale_y )
2023-09-07 15:27:34 -04:00
{
// 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
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : scale3d ( Optional < double > scale , Optional < double > origin_x , Optional < double > origin_y , Optional < double > origin_z )
2023-09-07 15:27:34 -04:00
{
// 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 ) ;
}
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : rotate ( Optional < double > rot_x , Optional < double > rot_y , Optional < double > rot_z )
2023-09-07 15:27:41 -04:00
{
// 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 ) ;
}
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : rotate_from_vector ( Optional < double > x , Optional < double > y )
2023-09-07 15:27:41 -04:00
{
// 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 ) ;
}
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : rotate_axis_angle ( Optional < double > x , Optional < double > y , Optional < double > z , Optional < double > angle )
2023-09-07 15:27:41 -04:00
{
// 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 ) ;
}
2023-09-04 12:22:55 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-skewx
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : skew_x ( double sx ) const
2023-09-04 12:22:55 -04:00
{
// 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
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : skew_y ( double sy ) const
2023-09-04 12:22:55 -04:00
{
// 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 ) ;
}
2023-09-03 06:42:50 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-multiply
2024-11-14 10:01:23 -05:00
WebIDL : : ExceptionOr < GC : : Ref < DOMMatrix > > DOMMatrixReadOnly : : multiply ( DOMMatrixInit other )
2023-09-03 06:42:50 -04:00
{
// 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 ) ;
}
2023-09-04 12:23:32 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-flipx
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : flip_x ( )
2023-09-04 12:23:32 -04:00
{
// 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]).
2024-04-24 07:32:02 -04:00
Gfx : : DoubleMatrix4x4 flip_matrix = {
- 1 , 0 , 0 , 0 ,
2023-09-04 12:23:32 -04:00
0 , 1 , 0 , 0 ,
0 , 0 , 1 , 0 ,
2024-04-24 07:32:02 -04:00
0 , 0 , 0 , 1
} ;
2023-09-04 12:23:32 -04:00
result - > m_matrix = result - > m_matrix * flip_matrix ;
// 3. Return result.
return result ;
}
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-flipy
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : flip_y ( )
2023-09-04 12:23:32 -04:00
{
// 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]).
2024-04-24 07:32:02 -04:00
Gfx : : DoubleMatrix4x4 flip_matrix = {
1 , 0 , 0 , 0 ,
2023-09-04 12:23:32 -04:00
0 , - 1 , 0 , 0 ,
0 , 0 , 1 , 0 ,
2024-04-24 07:32:02 -04:00
0 , 0 , 0 , 1
} ;
2023-09-04 12:23:32 -04:00
result - > m_matrix = result - > m_matrix * flip_matrix ;
// 3. Return result.
return result ;
}
2023-02-27 13:04:51 -05:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-inverse
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMMatrix > DOMMatrixReadOnly : : inverse ( ) const
2023-02-27 13:04:51 -05:00
{
// 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 ( ) ;
}
2023-08-15 07:16:45 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-transformpoint
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMPoint > DOMMatrixReadOnly : : transform_point ( DOMPointInit const & point ) const
2023-08-15 07:16:45 -04:00
{
// 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
2024-11-14 10:01:23 -05:00
GC : : Ref < DOMPoint > DOMMatrixReadOnly : : transform_point ( DOMPointReadOnly const & point ) const
2023-08-15 07:16:45 -04:00
{
// 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.
2023-08-22 03:46:02 -04:00
point_vector = m_matrix * point_vector ;
2023-08-15 07:16:45 -04:00
// 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 ( ) ) ;
}
2023-09-25 12:44:10 -04:00
// https://drafts.fxtf.org/geometry/#dom-dommatrixreadonly-tofloat32array
2024-11-14 10:01:23 -05:00
GC : : Ref < JS : : Float32Array > DOMMatrixReadOnly : : to_float32_array ( ) const
2023-09-25 12:44:10 -04:00
{
// 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
2024-11-14 10:01:23 -05:00
GC : : Ref < JS : : Float64Array > DOMMatrixReadOnly : : to_float64_array ( ) const
2023-09-25 12:44:10 -04:00
{
// 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 ) ;
}
2023-02-27 13:04:51 -05:00
// 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 ( ) ) ) {
2024-10-12 14:56:21 -04:00
return WebIDL : : InvalidStateError : : create ( realm ( ) , " Cannot stringify non-finite matrix values " _string ) ;
2023-02-27 13:04:51 -05:00
}
// 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.
2023-08-09 02:49:02 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : Value ( m11 ( ) ) . to_string_without_side_effects ( ) ) ) ;
2023-02-27 13:04:51 -05:00
// 3. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 4. Append ! ToString(m12 element) to string.
2023-08-09 02:49:02 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : Value ( m12 ( ) ) . to_string_without_side_effects ( ) ) ) ;
2023-02-27 13:04:51 -05:00
// 5. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 6. Append ! ToString(m21 element) to string.
2023-08-09 02:49:02 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : Value ( m21 ( ) ) . to_string_without_side_effects ( ) ) ) ;
2023-02-27 13:04:51 -05:00
// 7. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 8. Append ! ToString(m22 element) to string.
2023-08-09 02:49:02 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : Value ( m22 ( ) ) . to_string_without_side_effects ( ) ) ) ;
2023-02-27 13:04:51 -05:00
// 9. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 10. Append ! ToString(m41 element) to string.
2023-08-09 02:49:02 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : Value ( m41 ( ) ) . to_string_without_side_effects ( ) ) ) ;
2023-02-27 13:04:51 -05:00
// 11. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 12. Append ! ToString(m42 element) to string.
2023-08-09 02:49:02 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : Value ( m42 ( ) ) . to_string_without_side_effects ( ) ) ) ;
2023-02-27 13:04:51 -05:00
// 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.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m11 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 3. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 4. Append ! ToString(m12 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m12 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 5. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 6. Append ! ToString(m13 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m13 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 7. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 8. Append ! ToString(m14 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m14 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 9. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 10. Append ! ToString(m21 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m21 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 11. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 12. Append ! ToString(m22 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m22 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 13. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 14. Append ! ToString(m23 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m23 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 15. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 16. Append ! ToString(m24 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m24 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 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.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m31 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m32 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m33 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m34 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 18. Append ! ToString(m41 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m41 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 19. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 20. Append ! ToString(m42 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m42 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 21. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 22. Append ! ToString(m43 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m43 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 23. Append ", " to string.
TRY_OR_THROW_OOM ( vm , builder . try_append ( " , " sv ) ) ;
// 24. Append ! ToString(m44 element) to string.
2023-08-09 02:03:12 -04:00
TRY_OR_THROW_OOM ( vm , builder . try_append ( JS : : number_to_string ( m44 ( ) ) ) ) ;
2023-02-27 13:04:51 -05:00
// 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 ( ) ) ;
}
2024-03-09 16:07:22 -05:00
// https://drafts.fxtf.org/geometry/#structured-serialization
2024-03-10 14:13:45 -04:00
WebIDL : : ExceptionOr < void > DOMMatrixReadOnly : : serialization_steps ( HTML : : SerializationRecord & serialized , bool , HTML : : SerializationMemory & )
2024-03-09 16:07:22 -05:00
{
HTML : : serialize_primitive_type ( serialized , m_is_2d ) ;
// 1. If value’ s is 2D is true:
if ( m_is_2d ) {
// 1. Set serialized.[[M11]] to value’ s m11 element.
HTML : : serialize_primitive_type ( serialized , this - > m11 ( ) ) ;
// 2. Set serialized.[[M12]] to value’ s m12 element.
HTML : : serialize_primitive_type ( serialized , this - > m12 ( ) ) ;
// 3. Set serialized.[[M21]] to value’ s m21 element.
HTML : : serialize_primitive_type ( serialized , this - > m21 ( ) ) ;
// 4. Set serialized.[[M22]] to value’ s m22 element.
HTML : : serialize_primitive_type ( serialized , this - > m22 ( ) ) ;
// 5. Set serialized.[[M41]] to value’ s m41 element.
HTML : : serialize_primitive_type ( serialized , this - > m41 ( ) ) ;
// 6. Set serialized.[[M42]] to value’ s m42 element.
HTML : : serialize_primitive_type ( serialized , this - > 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.
HTML : : serialize_primitive_type ( serialized , this - > m11 ( ) ) ;
// 2. Set serialized.[[M12]] to value’ s m12 element.
HTML : : serialize_primitive_type ( serialized , this - > m12 ( ) ) ;
// 3. Set serialized.[[M13]] to value’ s m13 element.
HTML : : serialize_primitive_type ( serialized , this - > m13 ( ) ) ;
// 4. Set serialized.[[M14]] to value’ s m14 element.
HTML : : serialize_primitive_type ( serialized , this - > m14 ( ) ) ;
// 5. Set serialized.[[M21]] to value’ s m21 element.
HTML : : serialize_primitive_type ( serialized , this - > m21 ( ) ) ;
// 6. Set serialized.[[M22]] to value’ s m22 element.
HTML : : serialize_primitive_type ( serialized , this - > m22 ( ) ) ;
// 7. Set serialized.[[M23]] to value’ s m23 element.
HTML : : serialize_primitive_type ( serialized , this - > m23 ( ) ) ;
// 8. Set serialized.[[M24]] to value’ s m24 element.
HTML : : serialize_primitive_type ( serialized , this - > m24 ( ) ) ;
// 9. Set serialized.[[M31]] to value’ s m31 element.
HTML : : serialize_primitive_type ( serialized , this - > m31 ( ) ) ;
// 10. Set serialized.[[M32]] to value’ s m32 element.
HTML : : serialize_primitive_type ( serialized , this - > m32 ( ) ) ;
// 11. Set serialized.[[M33]] to value’ s m33 element.
HTML : : serialize_primitive_type ( serialized , this - > m33 ( ) ) ;
// 12. Set serialized.[[M34]] to value’ s m34 element.
HTML : : serialize_primitive_type ( serialized , this - > m34 ( ) ) ;
// 13. Set serialized.[[M41]] to value’ s m41 element.
HTML : : serialize_primitive_type ( serialized , this - > m41 ( ) ) ;
// 14. Set serialized.[[M42]] to value’ s m42 element.
HTML : : serialize_primitive_type ( serialized , this - > m42 ( ) ) ;
// 15. Set serialized.[[M43]] to value’ s m43 element.
HTML : : serialize_primitive_type ( serialized , this - > m43 ( ) ) ;
// 16. Set serialized.[[M44]] to value’ s m44 element.
HTML : : serialize_primitive_type ( serialized , this - > 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
2024-03-10 14:48:33 -04:00
WebIDL : : ExceptionOr < void > DOMMatrixReadOnly : : deserialization_steps ( ReadonlySpan < u32 > const & record , size_t & position , HTML : : DeserializationMemory & )
2024-03-09 16:07:22 -05:00
{
bool is_2d = HTML : : deserialize_primitive_type < bool > ( record , position ) ;
// 1. If serialized.[[Is2D]] is true:
if ( is_2d ) {
// 1. Set value’ s m11 element to serialized.[[M11]].
double m11 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 2. Set value’ s m12 element to serialized.[[M12]].
double m12 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 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]].
double m21 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 6. Set value’ s m22 element to serialized.[[M22]].
double m22 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 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]].
double m41 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 14 Set value’ s m42 element to serialized.[[M42]].
double m42 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 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]].
double m11 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 2. Set value’ s m12 element to serialized.[[M12]].
double m12 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 3. Set value’ s m13 element to serialized.[[M13]].
double m13 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 4. Set value’ s m14 element to serialized.[[M14]].
double m14 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 5. Set value’ s m21 element to serialized.[[M21]].
double m21 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 6. Set value’ s m22 element to serialized.[[M22]].
double m22 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 7. Set value’ s m23 element to serialized.[[M23]].
double m23 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 8. Set value’ s m24 element to serialized.[[M24]].
double m24 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 9. Set value’ s m31 element to serialized.[[M31]].
double m31 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 10. Set value’ s m32 element to serialized.[[M32]].
double m32 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 11. Set value’ s m33 element to serialized.[[M33]].
double m33 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 12. Set value’ s m34 element to serialized.[[M34]].
double m34 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 13. Set value’ s m41 element to serialized.[[M41]].
double m41 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 14. Set value’ s m42 element to serialized.[[M42]].
double m42 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 15. Set value’ s m43 element to serialized.[[M43]].
double m43 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 16. Set value’ s m44 element to serialized.[[M44]].
double m44 = HTML : : deserialize_primitive_type < double > ( record , position ) ;
// 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 { } ;
}
2023-02-27 13:04:51 -05:00
// 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 ( ) ) ) )
2023-03-03 13:04:58 -05:00
return WebIDL : : SimpleException { WebIDL : : SimpleExceptionType : : TypeError , " DOMMatrix2DInit.a and DOMMatrix2DInit.m11 must have the same value if they are both present " sv } ;
2023-02-27 13:04:51 -05:00
// - 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 ( ) ) ) )
2023-03-03 13:04:58 -05:00
return WebIDL : : SimpleException { WebIDL : : SimpleExceptionType : : TypeError , " DOMMatrix2DInit.b and DOMMatrix2DInit.m12 must have the same value if they are both present " sv } ;
2023-02-27 13:04:51 -05:00
// - 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 ( ) ) ) )
2023-03-03 13:04:58 -05:00
return WebIDL : : SimpleException { WebIDL : : SimpleExceptionType : : TypeError , " DOMMatrix2DInit.c and DOMMatrix2DInit.m21 must have the same value if they are both present " sv } ;
2023-02-27 13:04:51 -05:00
// - 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 ( ) ) ) )
2023-03-03 13:04:58 -05:00
return WebIDL : : SimpleException { WebIDL : : SimpleExceptionType : : TypeError , " DOMMatrix2DInit.d and DOMMatrix2DInit.m22 must have the same value if they are both present " sv } ;
2023-02-27 13:04:51 -05:00
// - 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 ( ) ) ) )
2023-03-03 13:04:58 -05:00
return WebIDL : : SimpleException { WebIDL : : SimpleExceptionType : : TypeError , " DOMMatrix2DInit.e and DOMMatrix2DInit.m41 must have the same value if they are both present " sv } ;
2023-02-27 13:04:51 -05:00
// - 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 ( ) ) ) )
2023-03-03 13:04:58 -05:00
return WebIDL : : SimpleException { WebIDL : : SimpleExceptionType : : TypeError , " DOMMatrix2DInit.f and DOMMatrix2DInit.m42 must have the same value if they are both present " sv } ;
2023-02-27 13:04:51 -05:00
// 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 ) {
2023-03-03 13:04:58 -05:00
return WebIDL : : SimpleException { WebIDL : : SimpleExceptionType : : TypeError , " DOMMatrixInit.is2D is true, but the given matrix is not a 2D matrix " sv } ;
2023-02-27 13:04:51 -05:00
}
}
// 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 { } ;
}
2024-01-06 12:05:21 -05:00
// 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]
2024-12-05 06:52:43 -05:00
auto transform_style_value = parse_css_value ( CSS : : Parser : : ParsingContext { } , transform_list , CSS : : PropertyID : : Transform ) ;
2024-12-26 16:32:16 -05:00
if ( ! transform_style_value | | ( transform_style_value - > is_keyword ( ) & & transform_style_value - > to_keyword ( ) ! = CSS : : Keyword : : None ) )
2024-10-12 14:56:21 -04:00
return WebIDL : : SyntaxError : : create ( realm , " Failed to parse CSS transform string. " _string ) ;
2024-12-20 05:32:17 -05:00
auto parsed_value = CSS : : ComputedProperties : : transformations_for_style_value ( * transform_style_value ) ;
2024-01-06 12:05:21 -05:00
// 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 ( ) )
2024-10-12 14:56:21 -04:00
return WebIDL : : SyntaxError : : create ( realm , " Failed to parse CSS transform string. " _string ) ;
2024-01-06 12:05:21 -05:00
matrix = matrix * transform_matrix . value ( ) ;
}
// 7. Return matrix and 2dTransform.
auto * elements = matrix . elements ( ) ;
Gfx : : DoubleMatrix4x4 double_matrix {
static_cast < double > ( elements [ 0 ] [ 0 ] ) , static_cast < double > ( elements [ 0 ] [ 1 ] ) , static_cast < double > ( elements [ 0 ] [ 2 ] ) , static_cast < double > ( elements [ 0 ] [ 3 ] ) ,
static_cast < double > ( elements [ 1 ] [ 0 ] ) , static_cast < double > ( elements [ 1 ] [ 1 ] ) , static_cast < double > ( elements [ 1 ] [ 2 ] ) , static_cast < double > ( elements [ 1 ] [ 3 ] ) ,
static_cast < double > ( elements [ 2 ] [ 0 ] ) , static_cast < double > ( elements [ 2 ] [ 1 ] ) , static_cast < double > ( elements [ 2 ] [ 2 ] ) , static_cast < double > ( elements [ 2 ] [ 3 ] ) ,
static_cast < double > ( elements [ 3 ] [ 0 ] ) , static_cast < double > ( elements [ 3 ] [ 1 ] ) , static_cast < double > ( elements [ 3 ] [ 2 ] ) , static_cast < float > ( elements [ 3 ] [ 3 ] )
} ;
return ParsedMatrix { double_matrix , is_2d_transform } ;
}
2023-02-27 13:04:51 -05:00
}