wfc_image 0.8.0

Create patterns for wfc using image files
Documentation
extern crate coord_2d;
extern crate image;
extern crate pixel_grid;
extern crate rand;
extern crate rand_xorshift;
extern crate simon;
extern crate wfc;
extern crate wfc_image;

use coord_2d::{Coord, Size};
use pixel_grid::{Window, WindowSpec};
use rand::{Rng, SeedableRng};
use rand_xorshift::XorShiftRng;
use std::num::NonZeroU32;
use wfc::wrap::*;
use wfc::*;
use wfc_image::ImagePatterns;

fn main() {
    use simon::*;
    let (seed, output_path, animate): (u64, Option<String>, bool) = args_all! {
        opt("s", "seed", "rng seed", "INT")
            .map(|seed| seed.unwrap_or_else(|| rand::thread_rng().gen())),
        opt("o", "output", "output path", "PATH"),
        flag("a", "animate", "animate"),
    }
    .with_help_default()
    .parse_env_default_or_exit();
    println!("seed: {}", seed);
    let mut rng = XorShiftRng::seed_from_u64(seed);
    let image = image::load_from_memory(include_bytes!("flowers.png")).unwrap();
    let pattern_size = NonZeroU32::new(3).unwrap();
    let mut image_patterns =
        ImagePatterns::new(&image, pattern_size, &[Orientation::Original]);
    let output_size = Size::new(48, 48);
    let window_spec = WindowSpec {
        title: "flowers".to_string(),
        grid_size: output_size,
        cell_size: Size::new(8, 8),
    };
    let mut window = if animate {
        Some(Window::new(window_spec))
    } else {
        None
    };
    let start_time = ::std::time::Instant::now();
    let id_grid = image_patterns.id_grid_original_orientation();
    let bottom_left_corner_coord =
        Coord::new(0, image_patterns.grid().size().y() as i32 - 1);
    let bottom_left_corner_id = *id_grid.get_checked(bottom_left_corner_coord);
    let sprout_id = *id_grid.get_checked(Coord::new(7, 21));
    let flower_id = *id_grid.get_checked(Coord::new(4, 1));

    image_patterns
        .pattern_mut(bottom_left_corner_id)
        .clear_count();

    let wave = {
        let global_stats = image_patterns.global_stats();
        let mut wave = Wave::new(output_size);
        'generate: loop {
            let mut context = Context::new();
            let mut run =
                RunBorrow::new(&mut context, &mut wave, &global_stats, WrapXY, &mut rng);
            let sprout_coord = Coord::new(
                (rng.gen::<u32>() % output_size.width()) as i32,
                output_size.height() as i32 - 2,
            );
            run.forbid_all_patterns_except(sprout_coord, sprout_id)
                .unwrap();
            for i in 0..(output_size.width() as i32) {
                let coord = Coord::new(i, output_size.height() as i32 - 1);
                run.forbid_all_patterns_except(coord, bottom_left_corner_id)
                    .unwrap();
            }
            for i in 0..8 {
                for j in 0..(output_size.width() as i32) {
                    let coord = Coord::new(j, output_size.height() as i32 - 1 - i);
                    run.forbid_pattern(coord, flower_id).unwrap();
                }
            }
            'inner: loop {
                match run.step(&mut rng) {
                    Ok(observe) => {
                        if let Some(window) = window.as_mut() {
                            window.with_pixel_grid(|mut pixel_grid| {
                                run.wave_cell_ref_iter()
                                    .zip(pixel_grid.iter_mut())
                                    .for_each(|(cell, mut pixel)| {
                                        let colour =
                                            image_patterns.weighted_average_colour(&cell);
                                        pixel.set_colour_array_rgba_u8(colour.data);
                                    });
                            });
                            window.draw();
                            if window.is_closed() {
                                return;
                            }
                        }
                        match observe {
                            Observe::Complete => break 'generate,
                            Observe::Incomplete => (),
                        }
                    }
                    Err(PropagateError::Contradiction) => break 'inner,
                }
            }
        }
        let end_time = ::std::time::Instant::now();
        println!("{:?}", end_time - start_time);
        wave
    };
    if let Some(output_path) = output_path.as_ref() {
        image_patterns
            .image_from_wave(&wave)
            .save(output_path)
            .unwrap();
    }
}