cuneus 0.5.0

A WGPU-based shader development tool
Documentation
use cuneus::compute::{ComputeShader, COMPUTE_TEXTURE_FORMAT_RGBA16};
use cuneus::{Core, RenderKit, ShaderApp, ShaderControls, ShaderManager};
use cuneus::{ExportManager};
use cuneus::WindowEvent;

cuneus::uniform_params! {
    struct ShaderParams {
    background_color: f32,
    _pad0: f32,
    _pad00: f32,
    _pad000: f32,
    hue_color: [f32; 3],
    _pad1: f32,

    light_intensity: f32,
    rim_power: f32,
    ao_strength: f32,
    env_light_strength: f32,

    iridescence_power: f32,
    falloff_distance: f32,
    global_light: f32,
    alpha_threshold: f32,

    mix_factor_scale: f32,
    _pad2: f32,
    _pad3: f32,
    _pad4: f32,

    _pad5: f32,
    _pad6: f32,
    _pad7: f32,
    _pad8: f32,
    _pad9: f32,
    _pad10: f32,
    _pad_align1: f32,
    _pad_align2: f32,
    }
}

struct Shader {
    base: RenderKit,
    compute_shader: ComputeShader,
    current_params: ShaderParams}

fn main() -> Result<(), Box<dyn std::error::Error>> {
    env_logger::init();
    let (app, event_loop) = ShaderApp::new("cuneus", 800, 600);
    app.run(event_loop, Shader::init)
}

impl Shader {
    fn clear_buffers(&mut self, core: &Core) {
        self.compute_shader.clear_all_buffers(core);
    }
}

impl ShaderManager for Shader {
    fn init(core: &Core) -> Self {
        let base = RenderKit::new(core);

        let initial_params = ShaderParams {
            background_color: 0.4,
            _pad0: 0.0,
            _pad00: 0.0,
            _pad000: 0.0,
            hue_color: [1.0, 2.0, 3.0],
            _pad1: 0.0,

            light_intensity: 1.8,
            rim_power: 3.0,
            ao_strength: 0.1,
            env_light_strength: 0.5,

            iridescence_power: 0.2,
            falloff_distance: 1.0,
            global_light: 1.0,
            alpha_threshold: 1.0,

            mix_factor_scale: 0.3,
            _pad2: 0.0,
            _pad3: 0.0,
            _pad4: 0.0,

            _pad5: 0.0,
            _pad6: 0.0,
            _pad7: 0.0,
            _pad8: 0.0,
            _pad9: 0.0,
            _pad10: 0.0,
            _pad_align1: 0.0,
            _pad_align2: 0.0,
        };

        // Entry point configuration
        let config = ComputeShader::builder()
            .with_entry_point("main")
            .with_custom_uniforms::<ShaderParams>()
            .with_audio(1024) // Automatically goes to @group(2)
            .with_workgroup_size([16, 16, 1])
            .with_texture_format(COMPUTE_TEXTURE_FORMAT_RGBA16)
            .with_label("Cuneus Compute")
            .build();

        let compute_shader = cuneus::compute_shader!(core, "shaders/cuneus.wgsl", config);

        compute_shader.set_custom_params(initial_params, &core.queue);

        Self {
            base,
            compute_shader,
            current_params: initial_params}
    }

    fn update(&mut self, core: &Core) {
        // Handle export
        self.compute_shader.handle_export(core, &mut self.base);
    }

    fn resize(&mut self, core: &Core) {
        self.base.default_resize(core, &mut self.compute_shader);
    }

    fn render(&mut self, core: &Core) -> Result<(), cuneus::SurfaceError> {
        let mut frame = self.base.begin_frame(core)?;

        let mut params = self.current_params;
        let mut changed = false;
        let mut should_start_export = false;
        let mut export_request = self.base.export_manager.get_ui_request();
        let mut controls_request = self
            .base
            .controls
            .get_ui_request(&self.base.start_time, &core.size, self.base.fps_tracker.fps());

        let full_output = if self.base.key_handler.show_ui {
            self.base.render_ui(core, |ctx| {
                RenderKit::apply_default_style(ctx);

                egui::Window::new("Cuneus")
                    .collapsible(true)
                    .resizable(true)
                    .default_width(250.0)
                    .show(ctx, |ui| {
                        egui::CollapsingHeader::new("Colors")
                            .default_open(false)
                            .show(ui, |ui| {
                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.background_color, 0.0..=1.0)
                                            .text("Background"),
                                    )
                                    .changed();

                                changed |=
                                    ui.color_edit_button_rgb(&mut params.hue_color).changed();
                                ui.label("Base Color");
                            });

                        egui::CollapsingHeader::new("Lighting")
                            .default_open(false)
                            .show(ui, |ui| {
                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.light_intensity, 0.0..=3.2)
                                            .text("Light Intensity"),
                                    )
                                    .changed();

                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.ao_strength, 0.0..=10.0)
                                            .text("AO Strength"),
                                    )
                                    .changed();

                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.global_light, 0.1..=2.0)
                                            .text("Global Light"),
                                    )
                                    .changed();

                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.rim_power, 0.1..=10.0)
                                            .text("Rim Power"),
                                    )
                                    .changed();

                                changed |= ui
                                    .add(
                                        egui::Slider::new(
                                            &mut params.env_light_strength,
                                            0.0..=1.0,
                                        )
                                        .text("Environment Light"),
                                    )
                                    .changed();
                            });

                        egui::CollapsingHeader::new("Effects")
                            .default_open(false)
                            .show(ui, |ui| {
                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.alpha_threshold, 0.0..=3.0)
                                            .text("Alpha Threshold"),
                                    )
                                    .changed();

                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.mix_factor_scale, 0.0..=1.5)
                                            .text("Mix Factor Scale"),
                                    )
                                    .changed();

                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.iridescence_power, 0.0..=1.0)
                                            .text("Iridescence"),
                                    )
                                    .changed();

                                changed |= ui
                                    .add(
                                        egui::Slider::new(&mut params.falloff_distance, 0.5..=5.0)
                                            .text("Light Falloff"),
                                    )
                                    .changed();
                            });

                        ui.separator();
                        ShaderControls::render_controls_widget(ui, &mut controls_request);
                        ui.separator();
                        should_start_export =
                            ExportManager::render_export_ui_widget(ui, &mut export_request);
                    });
            })
        } else {
            self.base.render_ui(core, |_ctx| {})
        };

        self.base.export_manager.apply_ui_request(export_request);
        if controls_request.should_clear_buffers {
            self.clear_buffers(core);
        }
        self.base.apply_control_request(controls_request);

        let current_time = self.base.controls.get_time(&self.base.start_time);

        let delta = 1.0 / 60.0;
        self.compute_shader
            .set_time(current_time, delta, &core.queue);

        if changed {
            self.current_params = params;
            self.compute_shader.set_custom_params(params, &core.queue);
        }

        if should_start_export {
            self.base.export_manager.start_export();
        }

        self.compute_shader.dispatch(&mut frame.encoder, core);

        self.base.renderer.render_to_view(&mut frame.encoder, &frame.view, &self.compute_shader.get_output_texture().bind_group);

        self.base.end_frame(core, frame, full_output);

        Ok(())
    }

    fn handle_input(&mut self, core: &Core, event: &WindowEvent) -> bool {
        self.base.default_handle_input(core, event)
    }
}