/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ use std::cell::Cell; use base::id::{DomPointId, DomPointIndex}; use constellation_traits::DomPoint; use dom_struct::dom_struct; use js::rust::HandleObject; use rustc_hash::FxHashMap; use crate::dom::bindings::codegen::Bindings::DOMMatrixBinding::DOMMatrixInit; use crate::dom::bindings::codegen::Bindings::DOMPointBinding::DOMPointInit; use crate::dom::bindings::codegen::Bindings::DOMPointReadOnlyBinding::DOMPointReadOnlyMethods; use crate::dom::bindings::error::Fallible; use crate::dom::bindings::reflector::{DomGlobal, Reflector, reflect_dom_object_with_proto}; use crate::dom::bindings::root::DomRoot; use crate::dom::bindings::serializable::Serializable; use crate::dom::bindings::structuredclone::StructuredData; use crate::dom::dommatrixreadonly::dommatrixinit_to_matrix; use crate::dom::globalscope::GlobalScope; use crate::dom::types::DOMPoint; use crate::script_runtime::CanGc; /// #[dom_struct] pub(crate) struct DOMPointReadOnly { reflector_: Reflector, x: Cell, y: Cell, z: Cell, w: Cell, } #[allow(non_snake_case)] impl DOMPointReadOnly { pub(crate) fn new_inherited(x: f64, y: f64, z: f64, w: f64) -> DOMPointReadOnly { DOMPointReadOnly { x: Cell::new(x), y: Cell::new(y), z: Cell::new(z), w: Cell::new(w), reflector_: Reflector::new(), } } pub(crate) fn new( global: &GlobalScope, x: f64, y: f64, z: f64, w: f64, can_gc: CanGc, ) -> DomRoot { Self::new_with_proto(global, None, x, y, z, w, can_gc) } fn new_with_proto( global: &GlobalScope, proto: Option, x: f64, y: f64, z: f64, w: f64, can_gc: CanGc, ) -> DomRoot { reflect_dom_object_with_proto( Box::new(DOMPointReadOnly::new_inherited(x, y, z, w)), global, proto, can_gc, ) } } #[allow(non_snake_case)] impl DOMPointReadOnlyMethods for DOMPointReadOnly { /// fn Constructor( global: &GlobalScope, proto: Option, can_gc: CanGc, x: f64, y: f64, z: f64, w: f64, ) -> Fallible> { Ok(DOMPointReadOnly::new_with_proto( global, proto, x, y, z, w, can_gc, )) } /// fn FromPoint(global: &GlobalScope, init: &DOMPointInit, can_gc: CanGc) -> DomRoot { Self::new(global, init.x, init.y, init.z, init.w, can_gc) } /// fn X(&self) -> f64 { self.x.get() } /// fn Y(&self) -> f64 { self.y.get() } /// fn Z(&self) -> f64 { self.z.get() } /// fn W(&self) -> f64 { self.w.get() } /// /// fn MatrixTransform( &self, matrix: &DOMMatrixInit, can_gc: CanGc, ) -> Fallible> { // Let matrixObject be the result of invoking create a DOMMatrix from the dictionary matrix. let matrix_object = match dommatrixinit_to_matrix(matrix) { Ok(converted) => converted, Err(exception) => { return Err(exception); }, }; // Steps 1-4 of transforming a point with a matrix let x = self.X(); let y = self.Y(); let z = self.Z(); let w = self.W(); // Steps 5-12 of transforming a point with a matrix let m = &matrix_object.1; let transformed_point = DOMPointInit { x: m.m11 * x + m.m21 * y + m.m31 * z + m.m41 * w, y: m.m12 * x + m.m22 * y + m.m32 * z + m.m42 * w, z: m.m13 * x + m.m23 * y + m.m33 * z + m.m43 * w, w: m.m14 * x + m.m24 * y + m.m34 * z + m.m44 * w, }; // Return the result of invoking transform a point with a matrix, given the current point // and matrixObject. The current point does not get modified. Ok(DOMPoint::new_from_init( &self.global(), &transformed_point, can_gc, )) } } #[allow(non_snake_case)] pub(crate) trait DOMPointWriteMethods { fn SetX(&self, value: f64); fn SetY(&self, value: f64); fn SetZ(&self, value: f64); fn SetW(&self, value: f64); } impl DOMPointWriteMethods for DOMPointReadOnly { fn SetX(&self, value: f64) { self.x.set(value); } fn SetY(&self, value: f64) { self.y.set(value); } fn SetZ(&self, value: f64) { self.z.set(value); } fn SetW(&self, value: f64) { self.w.set(value); } } impl Serializable for DOMPointReadOnly { type Index = DomPointIndex; type Data = DomPoint; fn serialize(&self) -> Result<(DomPointId, Self::Data), ()> { let serialized = DomPoint { x: self.x.get(), y: self.y.get(), z: self.z.get(), w: self.w.get(), }; Ok((DomPointId::new(), serialized)) } fn deserialize( owner: &GlobalScope, serialized: Self::Data, can_gc: CanGc, ) -> Result, ()> where Self: Sized, { Ok(Self::new( owner, serialized.x, serialized.y, serialized.z, serialized.w, can_gc, )) } fn serialized_storage<'a>( data: StructuredData<'a, '_>, ) -> &'a mut Option> { match data { StructuredData::Reader(r) => &mut r.points, StructuredData::Writer(w) => &mut w.points, } } }