From 69f6554c4c5266b2267a2191f297458a88ba1b79 Mon Sep 17 00:00:00 2001 From: kcz Date: Fri, 6 Dec 2024 00:14:28 -0500 Subject: [PATCH] avm2: Support matrix3D in transform copy reference: https://docs.ruffle.rs/en_US/FlashPlatform/reference/actionscript/3/flash/display/DisplayObject.html#transform --- .../globals/flash/display/display_object.rs | 39 ++++++++++++------- core/src/avm2/globals/flash/geom/transform.rs | 2 +- 2 files changed, 27 insertions(+), 14 deletions(-) diff --git a/core/src/avm2/globals/flash/display/display_object.rs b/core/src/avm2/globals/flash/display/display_object.rs index 92f8737b269e3..91e4a2e36cc29 100644 --- a/core/src/avm2/globals/flash/display/display_object.rs +++ b/core/src/avm2/globals/flash/display/display_object.rs @@ -750,25 +750,37 @@ pub fn set_transform<'gc>( ) -> Result, Error<'gc>> { let transform = args.get_object(activation, 0, "transform")?; - // FIXME - consider 3D matrix and pixel bounds - let matrix = transform - .get_public_property("matrix", activation)? - .as_object(); - - let Some(matrix) = matrix else { - // FP seems to not do anything when setting to a Transform with a null matrix, - // but we don't actually support setting the matrix to null anyway - // (see the comment in `flash::geom::transform::set_matrix`) - return Ok(Value::Undefined); + // FIXME - consider pixel bounds + let (matrix, has_matrix3d) = { + if let Some(matrix3d) = transform + .get_public_property("matrix3D", activation)? + .as_object() + { + let matrix3d = crate::avm2::globals::flash::geom::transform::object_to_matrix3d( + matrix3d, activation, + )?; + // FIXME: 3D transformation is unsupported now. + let matrix = Matrix::from(matrix3d); + (matrix, true) + } else if let Some(matrix) = transform + .get_public_property("matrix", activation)? + .as_object() + { + let matrix = + crate::avm2::globals::flash::geom::transform::object_to_matrix(matrix, activation)?; + (matrix, false) + } else { + // FP seems to not do anything when setting to a Transform with a null matrix, + // but we don't actually support setting the matrix to null anyway + // (see the comment in `flash::geom::transform::set_matrix`) + return Ok(Value::Undefined); + } }; let color_transform = transform .get_public_property("colorTransform", activation)? .as_object() .expect("colorTransform should be non-null"); - - let matrix = - crate::avm2::globals::flash::geom::transform::object_to_matrix(matrix, activation)?; let color_transform = crate::avm2::globals::flash::geom::transform::object_to_color_transform( color_transform, activation, @@ -778,6 +790,7 @@ pub fn set_transform<'gc>( let mut write = dobj.base_mut(activation.context.gc_context); write.set_matrix(matrix); write.set_color_transform(color_transform); + write.set_has_matrix3d_stub(has_matrix3d); drop(write); if let Some(parent) = dobj.parent() { // Self-transform changes are automatically handled, diff --git a/core/src/avm2/globals/flash/geom/transform.rs b/core/src/avm2/globals/flash/geom/transform.rs index 4cd9501cac264..48c4717b25cb6 100644 --- a/core/src/avm2/globals/flash/geom/transform.rs +++ b/core/src/avm2/globals/flash/geom/transform.rs @@ -203,7 +203,7 @@ fn matrix3d_to_object<'gc>( Ok(object.into()) } -fn object_to_matrix3d<'gc>( +pub fn object_to_matrix3d<'gc>( object: Object<'gc>, activation: &mut Activation<'_, 'gc>, ) -> Result> {