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
use crate::image::Image;
use crate::scaling::Scaling::*;
use std::num::NonZeroUsize;

#[derive(Debug, Clone, Copy)]
pub enum Scaling {
    /// Increase size of image by x_scale and y_scale
    /// Where {2,2} doubles the size
    NearestNeighbour {
        x_scale: NonZeroUsize,
        y_scale: NonZeroUsize,
    },
    Epx2x,
    Epx4x,
}

impl Scaling {
    pub fn nearest_neighbour(x: usize, y: usize) -> Scaling {
        assert!(x > 0, "x must be > 0");
        assert!(y > 0, "y must be > 0");
        unsafe {
            NearestNeighbour {
                x_scale: NonZeroUsize::new_unchecked(x),
                y_scale: NonZeroUsize::new_unchecked(y),
            }
        }
    }

    /// Double image size using nearest neighbour
    pub fn nn_double() -> Scaling {
        unsafe {
            NearestNeighbour {
                x_scale: NonZeroUsize::new_unchecked(2),
                y_scale: NonZeroUsize::new_unchecked(2),
            }
        }
    }
}

pub(crate) fn scale_nearest_neighbor(image: &Image, x_scale: usize, y_scale: usize) -> Image {
    let new_width = image.width() * x_scale;
    let new_height = image.height() * y_scale;
    let mut new_image = Image::new_blank(new_width, new_height);
    let x_scale = 1.0 / x_scale as f32;
    let y_scale = 1.0 / y_scale as f32;
    for y in 0..new_height {
        for x in 0..new_width {
            let px = (x as f32 * x_scale).floor() as usize;
            let py = (y as f32 * y_scale).floor() as usize;
            new_image.set_pixel(x, y, image.get_pixel(px, py));
        }
    }
    new_image
}

pub(crate) fn scale_epx(image: &Image) -> Image {
    let new_width = image.width() * 2;
    let new_height = image.height() * 2;
    let mut new_image = Image::new_blank(new_width, new_height);
    for x in 0..image.width() {
        for y in 0..image.height() {
            let mut p1 = image.get_pixel(x, y);
            let mut p2 = p1;
            let mut p3 = p1;
            let mut p4 = p1;
            let a = image.get_pixel(x, if y > 0 { y - 1 } else { y });
            let c = image.get_pixel(if x > 0 { x - 1 } else { x }, y);
            let b = image.get_pixel(if x < image.width() - 2 { x + 1 } else { x }, y);
            let d = image.get_pixel(x, if y < image.height() - 2 { y + 1 } else { y });

            if c == a && c != d && a != b {
                p1 = a
            }
            if a == b && a != c && b != d {
                p2 = b
            }
            if d == c && d != b && c != a {
                p3 = c
            }
            if b == d && b != a && d != c {
                p4 = d
            }

            let nx = x * 2;
            let ny = y * 2;
            new_image.set_pixel(nx, ny, p1);
            new_image.set_pixel(nx + 1, ny, p2);
            new_image.set_pixel(nx, ny + 1, p3);
            new_image.set_pixel(nx + 1, ny + 1, p4);
        }
    }
    new_image
}