1
0
Fork 0
nih-plug/nih_plug_vizia/src/lib.rs

189 lines
7.6 KiB
Rust
Raw Normal View History

2022-03-18 09:14:45 +11:00
//! [VIZIA](https://github.com/vizia/vizia) editor support for NIH plug.
// See the comment in the main `nih_plug` crate
#![allow(clippy::type_complexity)]
2022-03-18 09:14:45 +11:00
use crossbeam::atomic::AtomicCell;
use nih_plug::params::persist::PersistentField;
use nih_plug::prelude::{Editor, GuiContext};
use serde::{Deserialize, Serialize};
2023-03-08 07:22:15 +11:00
use std::fmt::Debug;
2022-03-18 09:14:45 +11:00
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Arc;
use vizia::prelude::*;
2022-03-18 09:14:45 +11:00
// Re-export for convenience
pub use vizia;
2022-03-18 10:22:58 +11:00
pub mod assets;
mod editor;
pub mod vizia_assets;
2022-03-19 01:26:19 +11:00
pub mod widgets;
2022-03-18 09:14:45 +11:00
/// Create an [`Editor`] instance using a [`vizia`][::vizia] GUI. The [`ViziaState`] passed to this
/// function contains the GUI's intitial size, and this is kept in sync whenever the GUI gets
/// resized. You can also use this to know if the GUI is open, so you can avoid performing
/// potentially expensive calculations while the GUI is not open. If you want this size to be
/// persisted when restoring a plugin instance, then you can store it in a `#[persist = "key"]`
/// field on your parameters struct.
///
/// The [`GuiContext`] is also passed to the app function. This is only meant for saving and
/// restoring state as part of your plugin's preset handling. You should not interact with this
/// directly to set parameters. Use the [`ParamEvent`][widgets::ParamEvent]s instead.
///
/// The `theming` argument controls what level of theming to apply. If you use
/// [`ViziaTheming::Custom`], then you **need** to call
/// [`nih_plug_vizia::assets::register_noto_sans_light()`][assets::register_noto_sans_light()] at
/// the start of your app function. Vizia's included fonts are also not registered by default. If
/// you use the Roboto font that normally comes with Vizia or any of its emoji or icon fonts, you
/// also need to register those using the functions in
/// [`nih_plug_vizia::vizia_assets`][crate::vizia_assets].
///
2022-03-18 09:14:45 +11:00
/// See [VIZIA](https://github.com/vizia/vizia)'s repository for examples on how to use this.
pub fn create_vizia_editor<F>(
vizia_state: Arc<ViziaState>,
theming: ViziaTheming,
app: F,
) -> Option<Box<dyn Editor>>
2022-03-18 09:14:45 +11:00
where
F: Fn(&mut Context, Arc<dyn GuiContext>) + 'static + Send + Sync,
2022-03-18 09:14:45 +11:00
{
Some(Box::new(editor::ViziaEditor {
2022-03-18 09:14:45 +11:00
vizia_state,
app: Arc::new(app),
theming,
// TODO: We can't get the size of the window when baseview does its own scaling, so if the
// host does not set a scale factor on Windows or Linux we should just use a factor of
// 1. That may make the GUI tiny but it also prevents it from getting cut off.
#[cfg(target_os = "macos")]
scaling_factor: AtomicCell::new(None),
#[cfg(not(target_os = "macos"))]
scaling_factor: AtomicCell::new(Some(1.0)),
emit_parameters_changed_event: Arc::new(AtomicBool::new(false)),
}))
}
/// Controls what level of theming to apply to the editor.
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Default)]
pub enum ViziaTheming {
/// Disable both `nih_plug_vizia`'s and vizia's built-in theming.
None,
/// Disable `nih_plug_vizia`'s custom theming. Vizia's included fonts are also not registered by
/// default. If you use the Roboto font that normally comes with Vizia or any of its emoji or
/// icon fonts, you need to register those using the functions in
/// [`nih_plug_vizia::vizia_assets`][crate::vizia_assets].
Builtin,
/// Apply `nih_plug_vizia`'s custom theming. This is the default. You **need** to call
/// [`nih_plug_vizia::assets::register_noto_sans_light()`][assets::register_noto_sans_light()]
/// at the start of your app function for the font to work correctly.
#[default]
Custom,
2022-03-18 09:14:45 +11:00
}
2023-03-08 07:22:15 +11:00
/// State for an `nih_plug_vizia` editor. The scale factor can be manipulated at runtime using
/// `cx.set_user_scale_factor()`.
#[derive(Serialize, Deserialize)]
2022-03-18 09:14:45 +11:00
pub struct ViziaState {
2023-03-08 07:22:15 +11:00
/// A function that returns the window's current size in logical pixels, before any sort of
/// scaling is applied. This size can be computed based on the plugin's current state.
#[serde(skip, default = "empty_size_fn")]
size_fn: Box<dyn Fn() -> (u32, u32) + Send + Sync>,
/// A scale factor that should be applied to `size` separate from from any system HiDPI scaling.
/// This can be used to allow GUIs to be scaled uniformly.
#[serde(with = "nih_plug::params::persist::serialize_atomic_cell")]
scale_factor: AtomicCell<f64>,
/// Whether the editor's window is currently open.
#[serde(skip)]
2022-03-18 09:14:45 +11:00
open: AtomicBool,
2023-03-08 07:22:15 +11:00
}
/// A default implementation for `size_fn` needed to be able to derive the `Deserialize` trait.
fn empty_size_fn() -> Box<dyn Fn() -> (u32, u32) + Send + Sync> {
Box::new(|| (0, 0))
}
impl Debug for ViziaState {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let (width, height) = (self.size_fn)();
2023-03-08 07:22:15 +11:00
f.debug_struct("ViziaState")
.field("size_fn", &format!("<fn> ({}, {})", width, height))
.field("scale_factor", &self.scale_factor)
.field("open", &self.open)
.finish()
}
2022-03-18 09:14:45 +11:00
}
impl<'a> PersistentField<'a, ViziaState> for Arc<ViziaState> {
fn set(&self, new_value: ViziaState) {
self.scale_factor.store(new_value.scale_factor.load());
}
fn map<F, R>(&self, f: F) -> R
where
F: Fn(&ViziaState) -> R,
{
f(self)
}
}
2022-03-18 09:14:45 +11:00
impl ViziaState {
/// Initialize the GUI's state. This value can be passed to [`create_vizia_editor()`]. The
2023-03-08 07:22:15 +11:00
/// callback always returns the window's current size is in logical pixels, so before it is
/// multiplied by the DPI scaling factor. This size can be computed based on the plugin's
/// current state.
pub fn new(size_fn: impl Fn() -> (u32, u32) + Send + Sync + 'static) -> Arc<ViziaState> {
2022-03-18 09:14:45 +11:00
Arc::new(ViziaState {
2023-03-08 07:22:15 +11:00
size_fn: Box::new(size_fn),
scale_factor: AtomicCell::new(1.0),
2022-03-18 09:14:45 +11:00
open: AtomicBool::new(false),
})
}
2023-03-08 07:22:15 +11:00
/// The same as [`new()`][Self::new()], but with a separate initial scale factor. This scale
/// factor gets applied on top of any HiDPI scaling, and it can be modified at runtime by
/// changing `cx.set_user_scale_factor()`.
pub fn new_with_default_scale_factor(
size_fn: impl Fn() -> (u32, u32) + Send + Sync + 'static,
default_scale_factor: f64,
) -> Arc<ViziaState> {
Arc::new(ViziaState {
2023-03-08 07:22:15 +11:00
size_fn: Box::new(size_fn),
scale_factor: AtomicCell::new(default_scale_factor),
open: AtomicBool::new(false),
})
}
/// Returns a `(width, height)` pair for the current size of the GUI in logical pixels, after
/// applying the user scale factor.
pub fn scaled_logical_size(&self) -> (u32, u32) {
2023-03-08 07:22:15 +11:00
let (logical_width, logical_height) = self.inner_logical_size();
let scale_factor = self.scale_factor.load();
(
(logical_width as f64 * scale_factor).round() as u32,
(logical_height as f64 * scale_factor).round() as u32,
)
}
/// Returns a `(width, height)` pair for the current size of the GUI in logical pixels before
/// applying the user scale factor.
pub fn inner_logical_size(&self) -> (u32, u32) {
2023-03-08 07:22:15 +11:00
(self.size_fn)()
2022-03-18 09:14:45 +11:00
}
/// Get the non-DPI related uniform scaling factor the GUI's size will be multiplied with. This
2022-04-12 00:23:34 +10:00
/// can be changed by changing `cx.user_scale_factor`.
pub fn user_scale_factor(&self) -> f64 {
self.scale_factor.load()
}
2022-03-18 09:14:45 +11:00
/// Whether the GUI is currently visible.
// Called `is_open()` instead of `open()` to avoid the ambiguity.
pub fn is_open(&self) -> bool {
self.open.load(Ordering::Acquire)
}
}