From 847887db9db33380a856c2f49c6c77ef77f9bd84 Mon Sep 17 00:00:00 2001 From: Robbert van der Helm Date: Mon, 14 Feb 2022 02:26:11 +0100 Subject: [PATCH] Allow custom names for enum variants --- src/lib.rs | 2 +- src/param.rs | 43 ++++++++++++++++++++++++++---------------- src/param/internals.rs | 4 ++-- 3 files changed, 30 insertions(+), 19 deletions(-) diff --git a/src/lib.rs b/src/lib.rs index bb4fedba..1c19a455 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -18,7 +18,7 @@ pub use param::range::Range; pub use param::smoothing::{Smoother, SmoothingStyle}; pub use param::{BoolParam, FloatParam, IntParam, Param}; // TODO: Consider re-exporting these from another module so you can import them all at once -pub use param::{Display, EnumIter, EnumParam}; +pub use param::{Display, EnumIter, EnumMessage, EnumParam}; pub use plugin::{ BufferConfig, BusConfig, Editor, NoteEvent, ParentWindowHandle, Plugin, ProcessStatus, Vst3Plugin, diff --git a/src/param.rs b/src/param.rs index 22b0ad63..8cffd0f6 100644 --- a/src/param.rs +++ b/src/param.rs @@ -7,7 +7,7 @@ use std::sync::Arc; // Re-export for the [EnumParam] // TODO: Consider re-exporting this from a non-root module to make it a bit less spammy:w -pub use strum::{Display, EnumIter, IntoEnumIterator as EnumIter}; +pub use strum::{Display, EnumIter, EnumMessage, IntoEnumIterator as EnumIter}; pub mod internals; pub mod range; @@ -149,11 +149,11 @@ pub struct BoolParam { } /// An [IntParam]-backed categorical parameter that allows convenient conversion to and from a -/// simple enum. This enum must derive the re-exported [EnumIter], [EnumString] and [Display] -/// traits. +/// simple enum. This enum must derive the re-exported [EnumIter] and [EnumMessage] and [Display] +/// traits. You can use the `#[strum(message = "Foo Bar")]` to override the name of the variant. // // TODO: Figure out a more sound way to get the same interface -pub struct EnumParam { +pub struct EnumParam { /// The integer parameter backing this enum parameter. pub inner: IntParam, /// An associative list of the variants converted to an i32 and their names. We need this @@ -196,15 +196,16 @@ impl Default for BoolParam { } } -impl Default for EnumParam { +impl Default for EnumParam { fn default() -> Self { - let variants: Vec<_> = T::iter().map(|v| (v, v.to_string())).collect(); + let variants: Vec<_> = Self::build_variants(); let default = T::default(); Self { inner: IntParam { - value: T::iter() - .position(|v| v == default) + value: variants + .iter() + .position(|(v, _)| v == &default) .expect("Invalid variant in init") as i32, range: Range::Linear { min: 0, @@ -393,7 +394,7 @@ impl Param for BoolParam { } } -impl Param for EnumParam { +impl Param for EnumParam { type Plain = T; fn update_smoother(&mut self, sample_rate: f32, reset: bool) { @@ -480,7 +481,7 @@ impl Display for BoolParam { } } -impl Display for EnumParam { +impl Display for EnumParam { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!(f, "{}", self.variants[self.inner.plain_value() as usize].1) } @@ -602,11 +603,11 @@ impl BoolParam { } } -impl EnumParam { +impl EnumParam { /// Build a new [Self]. Use the other associated functions to modify the behavior of the /// parameter. pub fn new(name: &'static str, default: T) -> Self { - let variants: Vec<_> = T::iter().map(|v| (v, v.to_string())).collect(); + let variants: Vec<_> = Self::build_variants(); Self { inner: IntParam { @@ -630,7 +631,7 @@ impl EnumParam { // TODO: Do exactly that } -impl EnumParam { +impl EnumParam { // TODO: There doesn't seem to be a single enum crate that gives you a dense [0, n_variatns) // mapping between integers and enum variants. So far linear search over this variants has // been the best approach. We should probably replace this with our own macro at some @@ -645,13 +646,11 @@ impl EnumParam { // // TODO: Come up with a sounder way to do this. #[allow(clippy::len_without_is_empty)] - #[inline(never)] pub fn len(&self) -> usize { self.variants.len() } /// Get the index associated to an enum variant. - #[inline(never)] fn to_index(&self, variant: T) -> i32 { self.variants .iter() @@ -667,10 +666,22 @@ impl EnumParam { /// /// indices `>= Self::len()` will trigger a panic. #[allow(clippy::wrong_self_convention)] - #[inline(never)] fn from_index(&self, index: i32) -> T { self.variants[index as usize].0 } + + fn build_variants() -> Vec<(T, String)> { + T::iter() + .map(|v| { + ( + v, + v.get_message() + .map(|custom_name| custom_name.to_string()) + .unwrap_or_else(|| v.to_string()), + ) + }) + .collect() + } } /// Caldculate how many decimals to round to when displaying a floating point value with a specific diff --git a/src/param/internals.rs b/src/param/internals.rs index 7179276a..915761cf 100644 --- a/src/param/internals.rs +++ b/src/param/internals.rs @@ -3,7 +3,7 @@ use std::collections::HashMap; use std::pin::Pin; -use super::{Display, EnumIter, Param}; +use super::{Display, EnumIter, EnumMessage, Param}; /// Re-export for use in the [Params] proc-macro. pub use serde_json::from_str as deserialize_field; @@ -50,7 +50,7 @@ pub trait Params { /// Dummy enum for in [ParamPtr]. This type needs an explicit representation size so we can compare /// the discriminants. -#[derive(Display, Clone, Copy, PartialEq, Eq, EnumIter)] +#[derive(Display, Clone, Copy, PartialEq, Eq, EnumIter, EnumMessage)] #[repr(i32)] pub enum AnyEnum { Foo,