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
121
122
123
124
125
126
//! A 2D canvas which represents an image of RGB colors. Pixels can be individually changed
//! individually.
//! Get the canvas in PPM format with [Canvas::to_ppm].
//!
//! # Examples
//!
//! A new (black) canvas can be created with new:
//! ```
//! # use truster::canvas::Canvas;
//! let canvas = Canvas::new(10, 20);
//! assert_eq!(canvas.width(), 10);
//! assert_eq!(canvas.height(), 20);
//! ```
//!
//! Pixels can be accessed and mutated with indexing:
//! ```
//! # use truster::canvas::Canvas;
//! use truster::color::Color;
//!
//! let mut canvas = Canvas::new(10, 20);
//! let red = Color::new(1.0, 0.0, 0.0);
//! canvas[[2, 3]] = red;
//! assert_eq!(canvas[[2, 3]], red);
//! ```
//!
//! Save a canvas in PPM format with [Canvas::to_ppm]:
//! ```
//! # use truster::canvas::Canvas;
//! use truster::color::Color;
//!
//! use std::fs::File;
//! use std::io::prelude::*;
//!
//! fn main() -> std::io::Result<()> {
//!     let mut canvas = Canvas::new(5, 3);
//!     canvas[[0, 0]] = Color::new(1.5, 0.0, 0.0);
//!     canvas[[2, 1]] = Color::new(0.0, 0.5, 0.0);
//!     canvas[[4, 2]] = Color::new(-0.5, 0.0, 1.0);
//!
//!     let mut file = File::create("foo.ppm")?;
//!     canvas.to_ppm(&mut file);
//!     let mut file = File::open("foo.ppm")?;
//!     let mut output = String::new();
//! 	file.read_to_string(&mut output);
//!
//!     assert_eq!(output, "P3
//! 5 3
//! 255
//! 255 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 128 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 0
//! 0 0 255
//! ");
//!
//! 	Ok(())
//! }
//! ```

use std::io::{Error, Write};
use std::ops::{Index, IndexMut};

use crate::color::Color;

/// A 2D image. See the module's documentation for more information.
pub struct Canvas {
    pixels: Vec<Vec<Color>>,
}

impl Canvas {
    /// Creates a new canvas with the given width and height. The new canvas is entirely black.
    pub fn new(width: usize, height: usize) -> Self {
        Self {
            pixels: vec![vec![Color::default(); width]; height],
        }
    }

    /// Returns `self`'s width, that is the number of columns in the image.
    pub fn width(&self) -> usize {
        if self.pixels.len() == 0 {
            0
        } else {
            self.pixels[0].len()
        }
    }

    /// Returns `self`'s height, that is the number of rows in the image.
    pub fn height(&self) -> usize {
        self.pixels.len()
    }

    /// Writes `self` to `file` in PPM format. See the module's documentation for an example.
    pub fn to_ppm(&self, file: &mut dyn Write) -> Result<(), Error> {
        write!(file, "P3\n{} {}\n255\n", self.width(), self.height())?;
        for row in self.pixels.iter() {
            for color in row {
                writeln!(file, "{}", color)?;
            }
        }
        Ok(())
    }
}

impl Index<[usize; 2]> for Canvas {
    type Output = Color;

    fn index(&self, index: [usize; 2]) -> &Self::Output {
        &self.pixels[index[1]][index[0]]
    }
}

impl IndexMut<[usize; 2]> for Canvas {
    fn index_mut(&mut self, index: [usize; 2]) -> &mut Self::Output {
        &mut self.pixels[index[1]][index[0]]
    }
}