Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Firefox/third_party/rust/euclid/src/   (Browser von der Mozilla Stiftung Version 136.0.1©)  Datei vom 10.2.2025 mit Größe 30 kB image not shown  

Quelle  translation.rs   Sprache: unbekannt

 
// Copyright 2018 The Servo Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

use crate::num::*;
use crate::UnknownUnit;
use crate::{point2, point3, vec2, vec3, Box2D, Box3D, Rect, Size2D};
use crate::{Point2D, Point3D, Transform2D, Transform3D, Vector2D, Vector3D};

use core::cmp::{Eq, PartialEq};
use core::fmt;
use core::hash::Hash;
use core::marker::PhantomData;
use core::ops::{Add, AddAssign, Neg, Sub, SubAssign};

#[cfg(feature = "bytemuck")]
use bytemuck::{Pod, Zeroable};
use num_traits::NumCast;
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};

/// A 2d transformation from a space to another that can only express translations.
///
/// The main benefit of this type over a Vector2D is the ability to cast
/// between a source and a destination spaces.
///
/// Example:
///
/// ```
/// use euclid::{Translation2D, Point2D, point2};
/// struct ParentSpace;
/// struct ChildSpace;
/// type ScrollOffset = Translation2D<i32, ParentSpace, ChildSpace>;
/// type ParentPoint = Point2D<i32, ParentSpace>;
/// type ChildPoint = Point2D<i32, ChildSpace>;
///
/// let scrolling = ScrollOffset::new(0, 100);
/// let p1: ParentPoint = point2(0, 0);
/// let p2: ChildPoint = scrolling.transform_point(p1);
/// ```
///
#[repr(C)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[cfg_attr(
    feature = "serde",
    serde(bound(
        serialize = "T: serde::Serialize",
        deserialize = "T: serde::Deserialize<'de>"
    ))
)]
pub struct Translation2D<T, Src, Dst> {
    pub x: T,
    pub y: T,
    #[doc(hidden)]
    pub _unit: PhantomData<(Src, Dst)>,
}

#[cfg(feature = "arbitrary")]
impl<'a, T, Src, Dst> arbitrary::Arbitrary<'a> for Translation2D<T, Src, Dst>
where
    T: arbitrary::Arbitrary<'a>,
{
    fn arbitrary(u: &mut arbitrary::Unstructured<'a>) -> arbitrary::Result<Self> {
        let (x, y) = arbitrary::Arbitrary::arbitrary(u)?;
        Ok(Translation2D {
            x,
            y,
            _unit: PhantomData,
        })
    }
}

impl<T: Copy, Src, Dst> Copy for Translation2D<T, Src, Dst> {}

impl<T: Clone, Src, Dst> Clone for Translation2D<T, Src, Dst> {
    fn clone(&self) -> Self {
        Translation2D {
            x: self.x.clone(),
            y: self.y.clone(),
            _unit: PhantomData,
        }
    }
}

impl<T, Src, Dst> Eq for Translation2D<T, Src, Dst> where T: Eq {}

impl<T, Src, Dst> PartialEq for Translation2D<T, Src, Dst>
where
    T: PartialEq,
{
    fn eq(&self, other: &Self) -> bool {
        self.x == other.x && self.y == other.y
    }
}

impl<T, Src, Dst> Hash for Translation2D<T, Src, Dst>
where
    T: Hash,
{
    fn hash<H: core::hash::Hasher>(&self, h: &mut H) {
        self.x.hash(h);
        self.y.hash(h);
    }
}

impl<T, Src, Dst> Translation2D<T, Src, Dst> {
    #[inline]
    pub const fn new(x: T, y: T) -> Self {
        Translation2D {
            x,
            y,
            _unit: PhantomData,
        }
    }

    #[inline]
    pub fn splat(v: T) -> Self
    where
        T: Clone,
    {
        Translation2D {
            x: v.clone(),
            y: v,
            _unit: PhantomData,
        }
    }

    /// Creates no-op translation (both `x` and `y` is `zero()`).
    #[inline]
    pub fn identity() -> Self
    where
        T: Zero,
    {
        Self::new(T::zero(), T::zero())
    }

    /// Check if translation does nothing (both x and y is `zero()`).
    ///
    /// ```rust
    /// use euclid::default::Translation2D;
    ///
    /// assert_eq!(Translation2D::<f32>::identity().is_identity(), true);
    /// assert_eq!(Translation2D::new(0, 0).is_identity(), true);
    /// assert_eq!(Translation2D::new(1, 0).is_identity(), false);
    /// assert_eq!(Translation2D::new(0, 1).is_identity(), false);
    /// ```
    #[inline]
    pub fn is_identity(&self) -> bool
    where
        T: Zero + PartialEq,
    {
        let _0 = T::zero();
        self.x == _0 && self.y == _0
    }

    /// No-op, just cast the unit.
    #[inline]
    pub fn transform_size(&self, s: Size2D<T, Src>) -> Size2D<T, Dst> {
        Size2D::new(s.width, s.height)
    }
}

impl<T: Copy, Src, Dst> Translation2D<T, Src, Dst> {
    /// Cast into a 2D vector.
    #[inline]
    pub fn to_vector(&self) -> Vector2D<T, Src> {
        vec2(self.x, self.y)
    }

    /// Cast into an array with x and y.
    #[inline]
    pub fn to_array(&self) -> [T; 2] {
        [self.x, self.y]
    }

    /// Cast into a tuple with x and y.
    #[inline]
    pub fn to_tuple(&self) -> (T, T) {
        (self.x, self.y)
    }

    /// Drop the units, preserving only the numeric value.
    #[inline]
    pub fn to_untyped(&self) -> Translation2D<T, UnknownUnit, UnknownUnit> {
        Translation2D {
            x: self.x,
            y: self.y,
            _unit: PhantomData,
        }
    }

    /// Tag a unitless value with units.
    #[inline]
    pub fn from_untyped(t: &Translation2D<T, UnknownUnit, UnknownUnit>) -> Self {
        Translation2D {
            x: t.x,
            y: t.y,
            _unit: PhantomData,
        }
    }

    /// Returns the matrix representation of this translation.
    #[inline]
    pub fn to_transform(&self) -> Transform2D<T, Src, Dst>
    where
        T: Zero + One,
    {
        (*self).into()
    }

    /// Translate a point and cast its unit.
    #[inline]
    pub fn transform_point(&self, p: Point2D<T, Src>) -> Point2D<T::Output, Dst>
    where
        T: Add,
    {
        point2(p.x + self.x, p.y + self.y)
    }

    /// Translate a rectangle and cast its unit.
    #[inline]
    pub fn transform_rect(&self, r: &Rect<T, Src>) -> Rect<T::Output, Dst>
    where
        T: Add<Output = T>,
    {
        Rect {
            origin: self.transform_point(r.origin),
            size: self.transform_size(r.size),
        }
    }

    /// Translate a 2D box and cast its unit.
    #[inline]
    pub fn transform_box(&self, r: &Box2D<T, Src>) -> Box2D<T::Output, Dst>
    where
        T: Add,
    {
        Box2D {
            min: self.transform_point(r.min),
            max: self.transform_point(r.max),
        }
    }

    /// Return the inverse transformation.
    #[inline]
    pub fn inverse(&self) -> Translation2D<T::Output, Dst, Src>
    where
        T: Neg,
    {
        Translation2D::new(-self.x, -self.y)
    }
}

impl<T: NumCast + Copy, Src, Dst> Translation2D<T, Src, Dst> {
    /// Cast from one numeric representation to another, preserving the units.
    ///
    /// When casting from floating vector to integer coordinates, the decimals are truncated
    /// as one would expect from a simple cast, but this behavior does not always make sense
    /// geometrically. Consider using `round()`, `ceil()` or `floor()` before casting.
    #[inline]
    pub fn cast<NewT: NumCast>(self) -> Translation2D<NewT, Src, Dst> {
        self.try_cast().unwrap()
    }

    /// Fallible cast from one numeric representation to another, preserving the units.
    ///
    /// When casting from floating vector to integer coordinates, the decimals are truncated
    /// as one would expect from a simple cast, but this behavior does not always make sense
    /// geometrically. Consider using `round()`, `ceil()` or `floor()` before casting.
    pub fn try_cast<NewT: NumCast>(self) -> Option<Translation2D<NewT, Src, Dst>> {
        match (NumCast::from(self.x), NumCast::from(self.y)) {
            (Some(x), Some(y)) => Some(Translation2D::new(x, y)),
            _ => None,
        }
    }

    // Convenience functions for common casts.

    /// Cast into an `f32` vector.
    #[inline]
    pub fn to_f32(self) -> Translation2D<f32, Src, Dst> {
        self.cast()
    }

    /// Cast into an `f64` vector.
    #[inline]
    pub fn to_f64(self) -> Translation2D<f64, Src, Dst> {
        self.cast()
    }

    /// Cast into an `usize` vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_usize(self) -> Translation2D<usize, Src, Dst> {
        self.cast()
    }

    /// Cast into an `u32` vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_u32(self) -> Translation2D<u32, Src, Dst> {
        self.cast()
    }

    /// Cast into an i32 vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_i32(self) -> Translation2D<i32, Src, Dst> {
        self.cast()
    }

    /// Cast into an i64 vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_i64(self) -> Translation2D<i64, Src, Dst> {
        self.cast()
    }
}

#[cfg(feature = "bytemuck")]
unsafe impl<T: Zeroable, Src, Dst> Zeroable for Translation2D<T, Src, Dst> {}

#[cfg(feature = "bytemuck")]
unsafe impl<T: Pod, Src: 'static, Dst: 'static> Pod for Translation2D<T, Src, Dst> {}

impl<T: Add, Src, Dst1, Dst2> Add<Translation2D<T, Dst1, Dst2>> for Translation2D<T, Src, Dst1> {
    type Output = Translation2D<T::Output, Src, Dst2>;

    fn add(self, other: Translation2D<T, Dst1, Dst2>) -> Self::Output {
        Translation2D::new(self.x + other.x, self.y + other.y)
    }
}

impl<T: AddAssign, Src, Dst> AddAssign<Translation2D<T, Dst, Dst>> for Translation2D<T, Src, Dst> {
    fn add_assign(&mut self, other: Translation2D<T, Dst, Dst>) {
        self.x += other.x;
        self.y += other.y;
    }
}

impl<T: Sub, Src, Dst1, Dst2> Sub<Translation2D<T, Dst1, Dst2>> for Translation2D<T, Src, Dst2> {
    type Output = Translation2D<T::Output, Src, Dst1>;

    fn sub(self, other: Translation2D<T, Dst1, Dst2>) -> Self::Output {
        Translation2D::new(self.x - other.x, self.y - other.y)
    }
}

impl<T: SubAssign, Src, Dst> SubAssign<Translation2D<T, Dst, Dst>> for Translation2D<T, Src, Dst> {
    fn sub_assign(&mut self, other: Translation2D<T, Dst, Dst>) {
        self.x -= other.x;
        self.y -= other.y;
    }
}

impl<T, Src, Dst> From<Vector2D<T, Src>> for Translation2D<T, Src, Dst> {
    fn from(v: Vector2D<T, Src>) -> Self {
        Translation2D::new(v.x, v.y)
    }
}

impl<T, Src, Dst> From<Translation2D<T, Src, Dst>> for Vector2D<T, Src> {
    fn from(t: Translation2D<T, Src, Dst>) -> Self {
        vec2(t.x, t.y)
    }
}

impl<T, Src, Dst> From<Translation2D<T, Src, Dst>> for Transform2D<T, Src, Dst>
where
    T: Zero + One,
{
    fn from(t: Translation2D<T, Src, Dst>) -> Self {
        Transform2D::translation(t.x, t.y)
    }
}

impl<T, Src, Dst> Default for Translation2D<T, Src, Dst>
where
    T: Zero,
{
    fn default() -> Self {
        Self::identity()
    }
}

impl<T: fmt::Debug, Src, Dst> fmt::Debug for Translation2D<T, Src, Dst> {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "Translation({:?},{:?})", self.x, self.y)
    }
}

/// A 3d transformation from a space to another that can only express translations.
///
/// The main benefit of this type over a Vector3D is the ability to cast
/// between a source and a destination spaces.
#[repr(C)]
pub struct Translation3D<T, Src, Dst> {
    pub x: T,
    pub y: T,
    pub z: T,
    #[doc(hidden)]
    pub _unit: PhantomData<(Src, Dst)>,
}

impl<T: Copy, Src, Dst> Copy for Translation3D<T, Src, Dst> {}

impl<T: Clone, Src, Dst> Clone for Translation3D<T, Src, Dst> {
    fn clone(&self) -> Self {
        Translation3D {
            x: self.x.clone(),
            y: self.y.clone(),
            z: self.z.clone(),
            _unit: PhantomData,
        }
    }
}

#[cfg(feature = "serde")]
impl<'de, T, Src, Dst> serde::Deserialize<'de> for Translation3D<T, Src, Dst>
where
    T: serde::Deserialize<'de>,
{
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: serde::Deserializer<'de>,
    {
        let (x, y, z) = serde::Deserialize::deserialize(deserializer)?;
        Ok(Translation3D {
            x,
            y,
            z,
            _unit: PhantomData,
        })
    }
}

#[cfg(feature = "serde")]
impl<T, Src, Dst> serde::Serialize for Translation3D<T, Src, Dst>
where
    T: serde::Serialize,
{
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: serde::Serializer,
    {
        (&self.x, &self.y, &self.z).serialize(serializer)
    }
}

impl<T, Src, Dst> Eq for Translation3D<T, Src, Dst> where T: Eq {}

impl<T, Src, Dst> PartialEq for Translation3D<T, Src, Dst>
where
    T: PartialEq,
{
    fn eq(&self, other: &Self) -> bool {
        self.x == other.x && self.y == other.y && self.z == other.z
    }
}

impl<T, Src, Dst> Hash for Translation3D<T, Src, Dst>
where
    T: Hash,
{
    fn hash<H: core::hash::Hasher>(&self, h: &mut H) {
        self.x.hash(h);
        self.y.hash(h);
        self.z.hash(h);
    }
}

impl<T, Src, Dst> Translation3D<T, Src, Dst> {
    #[inline]
    pub const fn new(x: T, y: T, z: T) -> Self {
        Translation3D {
            x,
            y,
            z,
            _unit: PhantomData,
        }
    }

    #[inline]
    pub fn splat(v: T) -> Self
    where
        T: Clone,
    {
        Translation3D {
            x: v.clone(),
            y: v.clone(),
            z: v,
            _unit: PhantomData,
        }
    }

    /// Creates no-op translation (`x`, `y` and `z` is `zero()`).
    #[inline]
    pub fn identity() -> Self
    where
        T: Zero,
    {
        Translation3D::new(T::zero(), T::zero(), T::zero())
    }

    /// Check if translation does nothing (`x`, `y` and `z` is `zero()`).
    ///
    /// ```rust
    /// use euclid::default::Translation3D;
    ///
    /// assert_eq!(Translation3D::<f32>::identity().is_identity(), true);
    /// assert_eq!(Translation3D::new(0, 0, 0).is_identity(), true);
    /// assert_eq!(Translation3D::new(1, 0, 0).is_identity(), false);
    /// assert_eq!(Translation3D::new(0, 1, 0).is_identity(), false);
    /// assert_eq!(Translation3D::new(0, 0, 1).is_identity(), false);
    /// ```
    #[inline]
    pub fn is_identity(&self) -> bool
    where
        T: Zero + PartialEq,
    {
        let _0 = T::zero();
        self.x == _0 && self.y == _0 && self.z == _0
    }

    /// No-op, just cast the unit.
    #[inline]
    pub fn transform_size(self, s: Size2D<T, Src>) -> Size2D<T, Dst> {
        Size2D::new(s.width, s.height)
    }
}

impl<T: Copy, Src, Dst> Translation3D<T, Src, Dst> {
    /// Cast into a 3D vector.
    #[inline]
    pub fn to_vector(&self) -> Vector3D<T, Src> {
        vec3(self.x, self.y, self.z)
    }

    /// Cast into an array with x, y and z.
    #[inline]
    pub fn to_array(&self) -> [T; 3] {
        [self.x, self.y, self.z]
    }

    /// Cast into a tuple with x, y and z.
    #[inline]
    pub fn to_tuple(&self) -> (T, T, T) {
        (self.x, self.y, self.z)
    }

    /// Drop the units, preserving only the numeric value.
    #[inline]
    pub fn to_untyped(&self) -> Translation3D<T, UnknownUnit, UnknownUnit> {
        Translation3D {
            x: self.x,
            y: self.y,
            z: self.z,
            _unit: PhantomData,
        }
    }

    /// Tag a unitless value with units.
    #[inline]
    pub fn from_untyped(t: &Translation3D<T, UnknownUnit, UnknownUnit>) -> Self {
        Translation3D {
            x: t.x,
            y: t.y,
            z: t.z,
            _unit: PhantomData,
        }
    }

    /// Returns the matrix representation of this translation.
    #[inline]
    pub fn to_transform(&self) -> Transform3D<T, Src, Dst>
    where
        T: Zero + One,
    {
        (*self).into()
    }

    /// Translate a point and cast its unit.
    #[inline]
    pub fn transform_point3d(&self, p: &Point3D<T, Src>) -> Point3D<T::Output, Dst>
    where
        T: Add,
    {
        point3(p.x + self.x, p.y + self.y, p.z + self.z)
    }

    /// Translate a point and cast its unit.
    #[inline]
    pub fn transform_point2d(&self, p: &Point2D<T, Src>) -> Point2D<T::Output, Dst>
    where
        T: Add,
    {
        point2(p.x + self.x, p.y + self.y)
    }

    /// Translate a 2D box and cast its unit.
    #[inline]
    pub fn transform_box2d(&self, b: &Box2D<T, Src>) -> Box2D<T::Output, Dst>
    where
        T: Add,
    {
        Box2D {
            min: self.transform_point2d(&b.min),
            max: self.transform_point2d(&b.max),
        }
    }

    /// Translate a 3D box and cast its unit.
    #[inline]
    pub fn transform_box3d(&self, b: &Box3D<T, Src>) -> Box3D<T::Output, Dst>
    where
        T: Add,
    {
        Box3D {
            min: self.transform_point3d(&b.min),
            max: self.transform_point3d(&b.max),
        }
    }

    /// Translate a rectangle and cast its unit.
    #[inline]
    pub fn transform_rect(&self, r: &Rect<T, Src>) -> Rect<T, Dst>
    where
        T: Add<Output = T>,
    {
        Rect {
            origin: self.transform_point2d(&r.origin),
            size: self.transform_size(r.size),
        }
    }

    /// Return the inverse transformation.
    #[inline]
    pub fn inverse(&self) -> Translation3D<T::Output, Dst, Src>
    where
        T: Neg,
    {
        Translation3D::new(-self.x, -self.y, -self.z)
    }
}

impl<T: NumCast + Copy, Src, Dst> Translation3D<T, Src, Dst> {
    /// Cast from one numeric representation to another, preserving the units.
    ///
    /// When casting from floating vector to integer coordinates, the decimals are truncated
    /// as one would expect from a simple cast, but this behavior does not always make sense
    /// geometrically. Consider using `round()`, `ceil()` or `floor()` before casting.
    #[inline]
    pub fn cast<NewT: NumCast>(self) -> Translation3D<NewT, Src, Dst> {
        self.try_cast().unwrap()
    }

    /// Fallible cast from one numeric representation to another, preserving the units.
    ///
    /// When casting from floating vector to integer coordinates, the decimals are truncated
    /// as one would expect from a simple cast, but this behavior does not always make sense
    /// geometrically. Consider using `round()`, `ceil()` or `floor()` before casting.
    pub fn try_cast<NewT: NumCast>(self) -> Option<Translation3D<NewT, Src, Dst>> {
        match (
            NumCast::from(self.x),
            NumCast::from(self.y),
            NumCast::from(self.z),
        ) {
            (Some(x), Some(y), Some(z)) => Some(Translation3D::new(x, y, z)),
            _ => None,
        }
    }

    // Convenience functions for common casts.

    /// Cast into an `f32` vector.
    #[inline]
    pub fn to_f32(self) -> Translation3D<f32, Src, Dst> {
        self.cast()
    }

    /// Cast into an `f64` vector.
    #[inline]
    pub fn to_f64(self) -> Translation3D<f64, Src, Dst> {
        self.cast()
    }

    /// Cast into an `usize` vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_usize(self) -> Translation3D<usize, Src, Dst> {
        self.cast()
    }

    /// Cast into an `u32` vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_u32(self) -> Translation3D<u32, Src, Dst> {
        self.cast()
    }

    /// Cast into an i32 vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_i32(self) -> Translation3D<i32, Src, Dst> {
        self.cast()
    }

    /// Cast into an i64 vector, truncating decimals if any.
    ///
    /// When casting from floating vector vectors, it is worth considering whether
    /// to `round()`, `ceil()` or `floor()` before the cast in order to obtain
    /// the desired conversion behavior.
    #[inline]
    pub fn to_i64(self) -> Translation3D<i64, Src, Dst> {
        self.cast()
    }
}

#[cfg(feature = "bytemuck")]
unsafe impl<T: Zeroable, Src, Dst> Zeroable for Translation3D<T, Src, Dst> {}

#[cfg(feature = "bytemuck")]
unsafe impl<T: Pod, Src: 'static, Dst: 'static> Pod for Translation3D<T, Src, Dst> {}

impl<T: Add, Src, Dst1, Dst2> Add<Translation3D<T, Dst1, Dst2>> for Translation3D<T, Src, Dst1> {
    type Output = Translation3D<T::Output, Src, Dst2>;

    fn add(self, other: Translation3D<T, Dst1, Dst2>) -> Self::Output {
        Translation3D::new(self.x + other.x, self.y + other.y, self.z + other.z)
    }
}

impl<T: AddAssign, Src, Dst> AddAssign<Translation3D<T, Dst, Dst>> for Translation3D<T, Src, Dst> {
    fn add_assign(&mut self, other: Translation3D<T, Dst, Dst>) {
        self.x += other.x;
        self.y += other.y;
        self.z += other.z;
    }
}

impl<T: Sub, Src, Dst1, Dst2> Sub<Translation3D<T, Dst1, Dst2>> for Translation3D<T, Src, Dst2> {
    type Output = Translation3D<T::Output, Src, Dst1>;

    fn sub(self, other: Translation3D<T, Dst1, Dst2>) -> Self::Output {
        Translation3D::new(self.x - other.x, self.y - other.y, self.z - other.z)
    }
}

impl<T: SubAssign, Src, Dst> SubAssign<Translation3D<T, Dst, Dst>> for Translation3D<T, Src, Dst> {
    fn sub_assign(&mut self, other: Translation3D<T, Dst, Dst>) {
        self.x -= other.x;
        self.y -= other.y;
        self.z -= other.z;
    }
}

impl<T, Src, Dst> From<Vector3D<T, Src>> for Translation3D<T, Src, Dst> {
    fn from(v: Vector3D<T, Src>) -> Self {
        Translation3D::new(v.x, v.y, v.z)
    }
}

impl<T, Src, Dst> From<Translation3D<T, Src, Dst>> for Vector3D<T, Src> {
    fn from(t: Translation3D<T, Src, Dst>) -> Self {
        vec3(t.x, t.y, t.z)
    }
}

impl<T, Src, Dst> From<Translation3D<T, Src, Dst>> for Transform3D<T, Src, Dst>
where
    T: Zero + One,
{
    fn from(t: Translation3D<T, Src, Dst>) -> Self {
        Transform3D::translation(t.x, t.y, t.z)
    }
}

impl<T, Src, Dst> Default for Translation3D<T, Src, Dst>
where
    T: Zero,
{
    fn default() -> Self {
        Self::identity()
    }
}

impl<T: fmt::Debug, Src, Dst> fmt::Debug for Translation3D<T, Src, Dst> {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "Translation({:?},{:?},{:?})", self.x, self.y, self.z)
    }
}

#[cfg(test)]
mod _2d {
    #[test]
    fn simple() {
        use crate::{rect, Rect, Translation2D};

        struct A;
        struct B;

        type Translation = Translation2D<i32, A, B>;
        type SrcRect = Rect<i32, A>;
        type DstRect = Rect<i32, B>;

        let tx = Translation::new(10, -10);
        let r1: SrcRect = rect(10, 20, 30, 40);
        let r2: DstRect = tx.transform_rect(&r1);
        assert_eq!(r2, rect(20, 10, 30, 40));

        let inv_tx = tx.inverse();
        assert_eq!(inv_tx.transform_rect(&r2), r1);

        assert!((tx + inv_tx).is_identity());
    }

    /// Operation tests
    mod ops {
        use crate::default::Translation2D;

        #[test]
        fn test_add() {
            let t1 = Translation2D::new(1.0, 2.0);
            let t2 = Translation2D::new(3.0, 4.0);
            assert_eq!(t1 + t2, Translation2D::new(4.0, 6.0));

            let t1 = Translation2D::new(1.0, 2.0);
            let t2 = Translation2D::new(0.0, 0.0);
            assert_eq!(t1 + t2, Translation2D::new(1.0, 2.0));

            let t1 = Translation2D::new(1.0, 2.0);
            let t2 = Translation2D::new(-3.0, -4.0);
            assert_eq!(t1 + t2, Translation2D::new(-2.0, -2.0));

            let t1 = Translation2D::new(0.0, 0.0);
            let t2 = Translation2D::new(0.0, 0.0);
            assert_eq!(t1 + t2, Translation2D::new(0.0, 0.0));
        }

        #[test]
        pub fn test_add_assign() {
            let mut t = Translation2D::new(1.0, 2.0);
            t += Translation2D::new(3.0, 4.0);
            assert_eq!(t, Translation2D::new(4.0, 6.0));

            let mut t = Translation2D::new(1.0, 2.0);
            t += Translation2D::new(0.0, 0.0);
            assert_eq!(t, Translation2D::new(1.0, 2.0));

            let mut t = Translation2D::new(1.0, 2.0);
            t += Translation2D::new(-3.0, -4.0);
            assert_eq!(t, Translation2D::new(-2.0, -2.0));

            let mut t = Translation2D::new(0.0, 0.0);
            t += Translation2D::new(0.0, 0.0);
            assert_eq!(t, Translation2D::new(0.0, 0.0));
        }

        #[test]
        pub fn test_sub() {
            let t1 = Translation2D::new(1.0, 2.0);
            let t2 = Translation2D::new(3.0, 4.0);
            assert_eq!(t1 - t2, Translation2D::new(-2.0, -2.0));

            let t1 = Translation2D::new(1.0, 2.0);
            let t2 = Translation2D::new(0.0, 0.0);
            assert_eq!(t1 - t2, Translation2D::new(1.0, 2.0));

            let t1 = Translation2D::new(1.0, 2.0);
            let t2 = Translation2D::new(-3.0, -4.0);
            assert_eq!(t1 - t2, Translation2D::new(4.0, 6.0));

            let t1 = Translation2D::new(0.0, 0.0);
            let t2 = Translation2D::new(0.0, 0.0);
            assert_eq!(t1 - t2, Translation2D::new(0.0, 0.0));
        }

        #[test]
        pub fn test_sub_assign() {
            let mut t = Translation2D::new(1.0, 2.0);
            t -= Translation2D::new(3.0, 4.0);
            assert_eq!(t, Translation2D::new(-2.0, -2.0));

            let mut t = Translation2D::new(1.0, 2.0);
            t -= Translation2D::new(0.0, 0.0);
            assert_eq!(t, Translation2D::new(1.0, 2.0));

            let mut t = Translation2D::new(1.0, 2.0);
            t -= Translation2D::new(-3.0, -4.0);
            assert_eq!(t, Translation2D::new(4.0, 6.0));

            let mut t = Translation2D::new(0.0, 0.0);
            t -= Translation2D::new(0.0, 0.0);
            assert_eq!(t, Translation2D::new(0.0, 0.0));
        }
    }
}

#[cfg(test)]
mod _3d {
    #[test]
    fn simple() {
        use crate::{point3, Point3D, Translation3D};

        struct A;
        struct B;

        type Translation = Translation3D<i32, A, B>;
        type SrcPoint = Point3D<i32, A>;
        type DstPoint = Point3D<i32, B>;

        let tx = Translation::new(10, -10, 100);
        let p1: SrcPoint = point3(10, 20, 30);
        let p2: DstPoint = tx.transform_point3d(&p1);
        assert_eq!(p2, point3(20, 10, 130));

        let inv_tx = tx.inverse();
        assert_eq!(inv_tx.transform_point3d(&p2), p1);

        assert!((tx + inv_tx).is_identity());
    }

    /// Operation tests
    mod ops {
        use crate::default::Translation3D;

        #[test]
        pub fn test_add() {
            let t1 = Translation3D::new(1.0, 2.0, 3.0);
            let t2 = Translation3D::new(4.0, 5.0, 6.0);
            assert_eq!(t1 + t2, Translation3D::new(5.0, 7.0, 9.0));

            let t1 = Translation3D::new(1.0, 2.0, 3.0);
            let t2 = Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t1 + t2, Translation3D::new(1.0, 2.0, 3.0));

            let t1 = Translation3D::new(1.0, 2.0, 3.0);
            let t2 = Translation3D::new(-4.0, -5.0, -6.0);
            assert_eq!(t1 + t2, Translation3D::new(-3.0, -3.0, -3.0));

            let t1 = Translation3D::new(0.0, 0.0, 0.0);
            let t2 = Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t1 + t2, Translation3D::new(0.0, 0.0, 0.0));
        }

        #[test]
        pub fn test_add_assign() {
            let mut t = Translation3D::new(1.0, 2.0, 3.0);
            t += Translation3D::new(4.0, 5.0, 6.0);
            assert_eq!(t, Translation3D::new(5.0, 7.0, 9.0));

            let mut t = Translation3D::new(1.0, 2.0, 3.0);
            t += Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t, Translation3D::new(1.0, 2.0, 3.0));

            let mut t = Translation3D::new(1.0, 2.0, 3.0);
            t += Translation3D::new(-4.0, -5.0, -6.0);
            assert_eq!(t, Translation3D::new(-3.0, -3.0, -3.0));

            let mut t = Translation3D::new(0.0, 0.0, 0.0);
            t += Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t, Translation3D::new(0.0, 0.0, 0.0));
        }

        #[test]
        pub fn test_sub() {
            let t1 = Translation3D::new(1.0, 2.0, 3.0);
            let t2 = Translation3D::new(4.0, 5.0, 6.0);
            assert_eq!(t1 - t2, Translation3D::new(-3.0, -3.0, -3.0));

            let t1 = Translation3D::new(1.0, 2.0, 3.0);
            let t2 = Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t1 - t2, Translation3D::new(1.0, 2.0, 3.0));

            let t1 = Translation3D::new(1.0, 2.0, 3.0);
            let t2 = Translation3D::new(-4.0, -5.0, -6.0);
            assert_eq!(t1 - t2, Translation3D::new(5.0, 7.0, 9.0));

            let t1 = Translation3D::new(0.0, 0.0, 0.0);
            let t2 = Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t1 - t2, Translation3D::new(0.0, 0.0, 0.0));
        }

        #[test]
        pub fn test_sub_assign() {
            let mut t = Translation3D::new(1.0, 2.0, 3.0);
            t -= Translation3D::new(4.0, 5.0, 6.0);
            assert_eq!(t, Translation3D::new(-3.0, -3.0, -3.0));

            let mut t = Translation3D::new(1.0, 2.0, 3.0);
            t -= Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t, Translation3D::new(1.0, 2.0, 3.0));

            let mut t = Translation3D::new(1.0, 2.0, 3.0);
            t -= Translation3D::new(-4.0, -5.0, -6.0);
            assert_eq!(t, Translation3D::new(5.0, 7.0, 9.0));

            let mut t = Translation3D::new(0.0, 0.0, 0.0);
            t -= Translation3D::new(0.0, 0.0, 0.0);
            assert_eq!(t, Translation3D::new(0.0, 0.0, 0.0));
        }
    }
}

[ Dauer der Verarbeitung: 0.35 Sekunden  (vorverarbeitet)  ]