// 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 . use nih_plug::prelude::Editor; use nih_plug_vizia::vizia::prelude::*; use nih_plug_vizia::widgets::*; use nih_plug_vizia::{assets, create_vizia_editor, ViziaState, ViziaTheming}; use std::sync::atomic::AtomicBool; use std::sync::Arc; use self::button::SafeModeButton; use crate::DiopserParams; mod button; /// VIZIA uses points instead of pixels for text const POINT_SCALE: f32 = 0.75; const EDITOR_WIDTH: u32 = 600; const EDITOR_HEIGHT: u32 = 490; #[derive(Lens)] struct Data { params: Arc, /// Whether the safe mode button is enabled. The number of filter stages is capped at 40 while /// this is active. safe_mode: Arc, } impl Model for Data {} // Makes sense to also define this here, makes it a bit easier to keep track of pub(crate) fn default_state() -> Arc { ViziaState::from_size(EDITOR_WIDTH, EDITOR_HEIGHT) } pub(crate) fn create( params: Arc, editor_state: Arc, ) -> Option> { create_vizia_editor(editor_state, ViziaTheming::Custom, move |cx, _| { assets::register_noto_sans_light(cx); assets::register_noto_sans_thin(cx); cx.add_theme(include_str!("theme.css")); Data { params: params.clone(), safe_mode: params.safe_mode.clone(), } .build(cx); ResizeHandle::new(cx); VStack::new(cx, |cx| { top_bar(cx); spectrum_analyzer(cx); other_params(cx); }); }) } /// This contain's the plugin's name, a bypass button, and some other controls. fn top_bar(cx: &mut Context) { HStack::new(cx, |cx| { Label::new(cx, "Diopser") .font(assets::NOTO_SANS_THIN) .font_size(50.0 * POINT_SCALE) .top(Pixels(-2.0)) .left(Pixels(7.0)); HStack::new(cx, |cx| { ParamSlider::new(cx, Data::params, |params| ¶ms.automation_precision) .with_label("Automation Precision") .id("automation-precision"); SafeModeButton::new(cx, Data::safe_mode, "Safe mode").left(Pixels(10.0)); ParamButton::new(cx, Data::params, |params| ¶ms.bypass) .for_bypass() .left(Pixels(10.0)); }) .child_space(Pixels(10.0)) .left(Stretch(1.0)); }) .id("top-bar"); } /// This shows a spectrum analyzer for the plugin's output, and also acts as an X-Y pad for the /// frequency and resonance parameters. fn spectrum_analyzer(cx: &mut Context) { Label::new(cx, "When I grow up, I want to be a spectrum analyzer!"); ParamLabel::new(cx, Data::params, |params| ¶ms.filter_frequency); // This is automatically rotated vertically ParamLabel::new(cx, Data::params, |params| ¶ms.filter_resonance) .width(Pixels(20.0)) .height(Pixels(180.0)); } /// The area below the spectrum analyzer that contains all of the other parameters. fn other_params(cx: &mut Context) { // Just to center the old generic UI for now ZStack::new(cx, |cx| { GenericUi::new(cx, Data::params).width(Auto).height(Auto); }) .width(Percentage(100.0)) .height(Stretch(1.0)) .child_space(Stretch(1.0)); }