From 3283a795bb77c4d7017dc9a371fad645a256576b Mon Sep 17 00:00:00 2001 From: Paul Walker Date: Thu, 19 Sep 2024 17:00:07 -0400 Subject: [PATCH] Starting on depth slider throw scaling This commit injects a rescaling data adapter between the depth slider and the underlying depth which basically uses a cube / cuberoot on the depth slider. This is part of the plan for #767 but not all of it With this a 1 semitone pitch adjustment is about 25% of the total throw so it is already a vast improvement, so merge it for now to make the synth generaly way less annoying to use. --- src-ui/app/edit-screen/components/ModPane.cpp | 49 ++++++++++++++++++- 1 file changed, 47 insertions(+), 2 deletions(-) diff --git a/src-ui/app/edit-screen/components/ModPane.cpp b/src-ui/app/edit-screen/components/ModPane.cpp index 316d87db..a2e65911 100644 --- a/src-ui/app/edit-screen/components/ModPane.cpp +++ b/src-ui/app/edit-screen/components/ModPane.cpp @@ -49,6 +49,50 @@ template struct ModRow : juce::Component, HasEditor powerAttachment; using attachment_t = connectors::PayloadDataAttachment; std::unique_ptr depthAttachment; + + struct ReScaler : sst::jucegui::data::Continuous + { + sst::jucegui::data::Continuous *under{nullptr}; + ReScaler(sst::jucegui::data::Continuous *u) : under(u) {} + + std::string getLabel() const override { return under->getLabel(); } + float scaleFromUnder(float f) const + { + auto v01 = (f - under->getMin()) / (under->getMax() - under->getMin()); + auto res = v01 * 2 - 1; + res = std::cbrt(res); + return res; + } + float scaleToUnder(float f) const + { + // F is bipolar -1..1 so + auto uni = f * f * f * 0.5 + 0.5; + auto resc = uni * (under->getMax() - under->getMin()) + under->getMin(); + return resc; + } + + float getValue() const override { return scaleFromUnder(under->getValue()); } + float getValue01() override { return getValue() * 0.5 + 0.5; } + void setValueFromGUI(const float &f) override { under->setValueFromGUI(scaleToUnder(f)); } + void setValueFromGUIQuantized(const float &f) override + { + under->setValueFromGUIQuantized(scaleToUnder(f)); + } + void setValueFromModel(const float &f) override + { + under->setValueFromModel(scaleToUnder(f)); + } + float getDefaultValue() const override { return 0.f; } + std::string getValueAsStringFor(float f) const override + { + return under->getValueAsStringFor(scaleToUnder(f)); + } + float getMin() const override { return -1.f; } + float getMax() const override { return 1.f; } + bool isBipolar() const override { return true; } + }; + + std::unique_ptr depthRescaler; std::unique_ptr power; std::unique_ptr source, sourceVia, curve, target; std::unique_ptr x1, x2, a1, a2; @@ -113,10 +157,11 @@ template struct ModRow : juce::Component, HasEditor } }, row.depth); + depthRescaler = std::make_unique(depthAttachment.get()); depth = std::make_unique(); depth->verticalReduction = 3; - depth->setSource(depthAttachment.get()); + depth->setSource(depthRescaler.get()); depth->onBeginEdit = [w = juce::Component::SafePointer(this)]() { if (!w) return; @@ -369,7 +414,7 @@ template struct ModRow : juce::Component, HasEditor { rDelta.rowLeadingGlyph = jcmp::GlyphPainter::GlyphType::LEFT_RIGHT; rDelta.leftAlignText = v->baseValue; - rDelta.centerAlignText = v->valUp; + rDelta.rightAlignText = v->valUp; } rDepth.rowLeadingGlyph = jcmp::GlyphPainter::GlyphType::VOLUME; rDepth.leftAlignText = fmt::format("{:.2f} %", at.value * 100);