Add a simple generic UI to Diopser
This commit is contained in:
parent
9b9799eb35
commit
b70af50fcd
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -288,6 +288,7 @@ name = "diopser"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"nih_plug",
|
"nih_plug",
|
||||||
|
"nih_plug_egui",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|
|
@ -16,3 +16,4 @@ simd = ["nih_plug/simd"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
nih_plug = { path = "../../", features = ["assert_process_allocs"] }
|
nih_plug = { path = "../../", features = ["assert_process_allocs"] }
|
||||||
|
nih_plug_egui = { path = "../../nih_plug_egui" }
|
||||||
|
|
39
plugins/diopser/src/editor.rs
Normal file
39
plugins/diopser/src/editor.rs
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
// Diopser: a phase rotation plugin
|
||||||
|
// Copyright (C) 2021-2022 Robbert van der Helm
|
||||||
|
//
|
||||||
|
// This program is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
use nih_plug::prelude::Editor;
|
||||||
|
use nih_plug_egui::widgets::generic_ui;
|
||||||
|
use nih_plug_egui::{create_egui_editor, egui, EguiState};
|
||||||
|
use std::pin::Pin;
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
use crate::DiopserParams;
|
||||||
|
|
||||||
|
// Makes sense to also definei this here, makes it a bit easier to keep track of
|
||||||
|
pub fn default_state() -> Arc<EguiState> {
|
||||||
|
EguiState::from_size(220, 330)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create(
|
||||||
|
params: Pin<Arc<DiopserParams>>,
|
||||||
|
editor_state: Arc<EguiState>,
|
||||||
|
) -> Option<Box<dyn Editor>> {
|
||||||
|
create_egui_editor(editor_state, (), move |egui_ctx, setter, _state| {
|
||||||
|
egui::CentralPanel::default().show(egui_ctx, |ui| {
|
||||||
|
generic_ui::create(ui, params.as_ref(), setter, generic_ui::GenericSlider);
|
||||||
|
});
|
||||||
|
})
|
||||||
|
}
|
|
@ -23,11 +23,13 @@ compile_error!("Compiling without SIMD support is currently not supported");
|
||||||
extern crate nih_plug;
|
extern crate nih_plug;
|
||||||
|
|
||||||
use nih_plug::prelude::*;
|
use nih_plug::prelude::*;
|
||||||
|
use nih_plug_egui::EguiState;
|
||||||
use std::pin::Pin;
|
use std::pin::Pin;
|
||||||
use std::simd::f32x2;
|
use std::simd::f32x2;
|
||||||
use std::sync::atomic::{AtomicBool, Ordering};
|
use std::sync::atomic::{AtomicBool, Ordering};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
mod editor;
|
||||||
mod filter;
|
mod filter;
|
||||||
|
|
||||||
/// How many all-pass filters we can have in series at most. The filter stages parameter determines
|
/// How many all-pass filters we can have in series at most. The filter stages parameter determines
|
||||||
|
@ -42,12 +44,10 @@ const MAX_AUTOMATION_STEP_SIZE: u32 = 512;
|
||||||
// All features from the original Diopser have been implemented (and the spread control has been
|
// All features from the original Diopser have been implemented (and the spread control has been
|
||||||
// improved). Other features I want to implement are:
|
// improved). Other features I want to implement are:
|
||||||
// - Briefly muting the output when changing the number of filters to get rid of the clicks
|
// - Briefly muting the output when changing the number of filters to get rid of the clicks
|
||||||
// - A GUI
|
// - A proper GUI
|
||||||
//
|
|
||||||
// TODO: Decide on whether to keep the scalar version or to just only support SIMD. Issue is that
|
|
||||||
// packed_simd requires a nightly compiler.
|
|
||||||
struct Diopser {
|
struct Diopser {
|
||||||
params: Pin<Box<DiopserParams>>,
|
params: Pin<Arc<DiopserParams>>,
|
||||||
|
editor_state: Arc<EguiState>,
|
||||||
|
|
||||||
/// Needed for computing the filter coefficients.
|
/// Needed for computing the filter coefficients.
|
||||||
sample_rate: f32,
|
sample_rate: f32,
|
||||||
|
@ -72,7 +72,7 @@ struct Diopser {
|
||||||
// TODO: Some combinations of parameters can cause really loud resonance. We should limit the
|
// TODO: Some combinations of parameters can cause really loud resonance. We should limit the
|
||||||
// resonance and filter stages parameter ranges in the GUI until the user unlocks.
|
// resonance and filter stages parameter ranges in the GUI until the user unlocks.
|
||||||
#[derive(Params)]
|
#[derive(Params)]
|
||||||
struct DiopserParams {
|
pub struct DiopserParams {
|
||||||
/// The number of all-pass filters applied in series.
|
/// The number of all-pass filters applied in series.
|
||||||
#[id = "stages"]
|
#[id = "stages"]
|
||||||
filter_stages: IntParam,
|
filter_stages: IntParam,
|
||||||
|
@ -112,7 +112,8 @@ impl Default for Diopser {
|
||||||
let should_update_filters = Arc::new(AtomicBool::new(false));
|
let should_update_filters = Arc::new(AtomicBool::new(false));
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
params: Box::pin(DiopserParams::new(should_update_filters.clone())),
|
params: Arc::pin(DiopserParams::new(should_update_filters.clone())),
|
||||||
|
editor_state: editor::default_state(),
|
||||||
|
|
||||||
sample_rate: 1.0,
|
sample_rate: 1.0,
|
||||||
|
|
||||||
|
@ -125,7 +126,7 @@ impl Default for Diopser {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DiopserParams {
|
impl DiopserParams {
|
||||||
pub fn new(should_update_filters: Arc<AtomicBool>) -> Self {
|
fn new(should_update_filters: Arc<AtomicBool>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
filter_stages: IntParam::new(
|
filter_stages: IntParam::new(
|
||||||
"Filter Stages",
|
"Filter Stages",
|
||||||
|
@ -221,6 +222,10 @@ impl Plugin for Diopser {
|
||||||
self.params.as_ref()
|
self.params.as_ref()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn editor(&self) -> Option<Box<dyn Editor>> {
|
||||||
|
editor::create(self.params.clone(), self.editor_state.clone())
|
||||||
|
}
|
||||||
|
|
||||||
fn accepts_bus_config(&self, config: &BusConfig) -> bool {
|
fn accepts_bus_config(&self, config: &BusConfig) -> bool {
|
||||||
// The SIMD version only supports stereo
|
// The SIMD version only supports stereo
|
||||||
config.num_input_channels == config.num_output_channels && config.num_input_channels == 2
|
config.num_input_channels == config.num_output_channels && config.num_input_channels == 2
|
||||||
|
|
|
@ -85,6 +85,8 @@ impl Plugin for Gain {
|
||||||
(),
|
(),
|
||||||
move |egui_ctx, setter, _state| {
|
move |egui_ctx, setter, _state| {
|
||||||
egui::CentralPanel::default().show(egui_ctx, |ui| {
|
egui::CentralPanel::default().show(egui_ctx, |ui| {
|
||||||
|
// NOTE: See `plugins/diopser/src/editor.rs` for an example using the generic UI widget
|
||||||
|
|
||||||
// This is a fancy widget that can get all the information it needs to properly
|
// This is a fancy widget that can get all the information it needs to properly
|
||||||
// display and modify the parameter from the parametr itself
|
// display and modify the parameter from the parametr itself
|
||||||
// It's not yet fully implemented, as the text is missing.
|
// It's not yet fully implemented, as the text is missing.
|
||||||
|
|
Loading…
Reference in a new issue