1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use crate::{
    primitives::{Arc, Line, Point},
    Vector,
};
use kurbo::Affine;

/// Something which can be scaled in *Drawing Space*
pub trait Scale {
    fn scale(&mut self, factor: f64, base: Vector);

    fn scaled(&self, factor: f64, base: Vector) -> Self
    where 
        Self: Sized + Clone,
        {
            let mut clone = self.clone();
            clone.scale(factor, base);

            clone
        }

}

impl<'t, T: Scale + ?Sized> Scale for &'t mut T {
    fn scale(&mut self, factor: f64, base: Vector) {
        (*self).scale(factor, base);
    }
}

impl Scale for Vector {
    fn scale(&mut self, factor: f64, base: Vector) {
        let translate_to_base = Affine::translate(base * -1.0);
        let scale = Affine::scale(factor);
        let translate_back = Affine::translate(base);
        let combined_transform = translate_back * scale * translate_to_base;
        let new_pos = combined_transform * *self;
        self.x = new_pos.x;
        self.y = new_pos.y;
    }
}

impl Scale for Point {
    fn scale(&mut self, factor: f64, base: Vector) {
        self.location.scale(factor, base);
    }
}

impl Scale for Line {
    fn scale(&mut self, factor: f64, base: Vector) {
        self.start.scale(factor, base);
        self.end.scale(factor, base);
    }
}

impl Scale for Arc {
    fn scale(&mut self, factor: f64, base: Vector) {
        let mut centre = self.centre();
        centre.scale(factor, base);
        *self = Arc::from_centre_radius(
            centre, 
            self.radius() * factor, 
            self.start_angle(), 
            self.sweep_angle(),
        );   
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn vector() {
        let original = Vector::new(1.0, 1.0);
        let factor = 2.0;

        let actual = original.scaled(factor, Vector::zero());
        let expected = Vector::new(2.0, 2.0);

        assert_eq!(actual, expected);
    }

    #[test]
    fn line() {
        let start = Vector::new(2.0, 4.0);
        let end = Vector::new(3.0, -5.0);
        let original = Line::new(start, end);
        let factor = 1.5;

        let actual = original.scaled(factor, Vector::zero());
        let expected = Line::new(Vector::new(3.0, 6.0), Vector::new(4.5, -7.5));

        assert_eq!(actual, expected);

        // scale by line mid-point as reference
        let actual = original.scaled(factor, start + original.displacement() * 0.5);
        let expected = Line::new(Vector::new(1.75, 6.25), Vector::new(3.25, -7.25));

        assert_eq!(actual, expected);
    }

    #[test]
    fn arc() {
        let centre = Vector::new(-1.4, 2.0);
        let radius = 5.0;
        let start_angle = 0.5;
        let sweep_angle = 1.0;
        let original = Arc::from_centre_radius(centre, radius, start_angle, sweep_angle);
        let factor = 2.0;

        let actual = original.scaled(factor, Vector::zero());
        let expected = Arc::from_centre_radius(Vector::new(-2.8, 4.0), radius * factor, start_angle, sweep_angle);

        assert_eq!(actual, expected);

        let actual = original.scaled(factor, centre);
        let expected = Arc::from_centre_radius(centre, radius * factor, start_angle, sweep_angle);

        assert_eq!(actual, expected);
    }
}