use crate::definitions::Image;
use image::Pixel;
pub fn flood_fill<P>(image: &Image<P>, x: u32, y: u32, fill_with: P) -> Image<P>
where
P: Pixel + PartialEq,
{
let mut filled_image = image.clone();
flood_fill_mut(&mut filled_image, x, y, fill_with);
filled_image
}
#[doc=generate_mut_doc_comment!("flood_fill")]
pub fn flood_fill_mut<P>(image: &mut Image<P>, x: u32, y: u32, fill_with: P)
where
P: Pixel + PartialEq,
{
let target = *image.get_pixel(x, y);
let mut stack = Vec::new();
stack.push((x as i32, x as i32, y as i32, 1_i32));
stack.push((x as i32, x as i32, y as i32 - 1, -1_i32));
while let Some((x1, x2, y, dy)) = stack.pop() {
let mut x1 = x1;
let mut x = x1;
if inside(image, x, y, target) {
while inside(image, x - 1, y, target) {
image.put_pixel(x as u32 - 1, y as u32, fill_with);
x -= 1;
}
if x < x1 {
stack.push((x, x1 - 1, y - dy, -dy))
}
}
while x1 <= x2 {
while inside(image, x1, y, target) {
image.put_pixel(x1 as u32, y as u32, fill_with);
x1 += 1;
}
if x1 > x {
stack.push((x, x1 - 1, y + dy, dy))
}
if x1 - 1 > x2 {
stack.push((x2 + 1, x1 - 1, y - dy, -dy))
}
x1 += 1;
while x1 < x2 && !inside(image, x1, y, target) {
x1 += 1
}
x = x1
}
}
}
fn inside<P>(image: &Image<P>, x: i32, y: i32, target_pixel: P) -> bool
where
P: Pixel + PartialEq,
{
if x < 0 || y < 0 {
return false;
}
let x = x as u32;
let y = y as u32;
let (width, height) = image.dimensions();
x < width && y < height && *image.get_pixel(x, y) == target_pixel
}
#[cfg(test)]
mod tests {
use super::flood_fill;
use image::Luma;
#[test]
fn test_flood_fill() {
let border = 0u8;
let region = 1u8;
let filled = 2u8;
let image = gray_image!(
border, border, border, border, border;
border, region, region, region, border;
border, region, border, region, border;
border, region, region, region, border;
border, border, border, border, border);
let expected = gray_image!(
border, border, border, border, border;
border, filled, filled, filled, border;
border, filled, border, filled, border;
border, filled, filled, filled, border;
border, border, border, border, border);
let actual = flood_fill(&image, 1, 1, Luma([filled]));
assert_pixels_eq!(actual, expected);
}
}