paulxstretch/Source/envelope_component.cpp
essej c89c1558e3 Various items before 1.5.1
- fixed passthrough when capturing
 - added declicking fades when bypassing input passthrough, playback, and recording
 - layout tweaks for extra small windows
 - waveform scrollbar improvements, visible edge drag handles, and allows dragging from anywhere
 - minor visual tweaks
 - license header mods, and LICENSE file consolidation
2022-04-21 23:04:30 -04:00

512 lines
14 KiB
C++

// SPDX-License-Identifier: GPLv3-or-later WITH Appstore-exception
// Copyright (C) 2017 Xenakios
// Copyright (C) 2020 Jesse Chappell
#include "envelope_component.h"
EnvelopeComponent::EnvelopeComponent(CriticalSection* cs) : m_cs(cs)
{
OnEnvelopeEdited = [](breakpoint_envelope*) {};
setWantsKeyboardFocus(true);
YFromNormalized = [](double x) { return x; };
XFromNormalized = [](double x) { return x; };
addChildComponent(&m_bubble);
setOpaque(true);
#if JUCE_IOS
m_menubutton.setButtonText("...");
m_menubutton.onClick = [this]() {
showPopupMenu();
};
addAndMakeVisible(&m_menubutton);
#endif
}
EnvelopeComponent::~EnvelopeComponent()
{
}
void EnvelopeComponent::show_bubble(int x, int y, const envelope_point& node)
{
double scaledtime = XFromNormalized(node.pt_x);
double scaledvalue = YFromNormalized(node.pt_y);
x -= 50;
if (x < 0)
x = 0;
if (x + 100 > getWidth())
x = getWidth() - 100;
if (y < 0)
y = 0;
if (y + 20 > getHeight())
y = getHeight() - 20;
AttributedString temp(String::formatted("%.2f %.2f", scaledtime, scaledvalue));
temp.setColour(Colours::white);
m_bubble.showAt({ x,y,100,20 }, temp , 5000);
}
void EnvelopeComponent::resized()
{
#if JUCE_IOS
int butw = 38;
int buth = 34;
m_menubutton.setBounds((getWidth() - butw)/2, 1, butw, buth);
#endif
}
void EnvelopeComponent::paint(Graphics& g)
{
float targsize = 10.0;
#if JUCE_IOS
targsize = 16.0;
#endif
if (!EnvelopeUnderlayDraw)
{
g.fillAll(Colours::black);
g.setColour(Colours::white.darker());
juce::Rectangle<int> rect(0, 0, getWidth(), getHeight());
g.setFont(15.0);
}
else
{
g.saveState();
EnvelopeUnderlayDraw(this, g);
g.restoreState();
}
if (m_envelope == nullptr)
{
g.drawText("No envelope set", 10, 10, getWidth(), getHeight(), Justification::centred);
return;
}
if (m_envelope.use_count() == 1)
{
g.drawText("Envelope is orphaned (may be a bug)", 10, 10, getWidth(), getHeight(), Justification::centred);
return;
}
int freqdivs = jmax(2, jmin(10, (int) (getWidth() / 60.0f)));
int divwidth = getWidth() / freqdivs;
for (int i = 0; i < freqdivs; ++i)
{
double norm = 1.0 / freqdivs * i;
double hz = XFromNormalized(norm);
int xcor = divwidth * i;
g.drawFittedText(String(hz, 1), xcor, getHeight() - 20, divwidth, 20, Justification::topLeft, 1);
}
String name = m_envelope->GetName();
if (name.isEmpty() == true)
name = "Untitled envelope";
g.drawText(name, 10, 10, getWidth(), getHeight(), Justification::topLeft);
auto draw_env = [this, &g](Colour envcolor, bool drawTransformed, float linethickness)
{
g.setColour(envcolor);
double y0 = 0.0;
if (drawTransformed==false)
y0 = m_envelope->GetInterpolatedEnvelopeValue(0.0);
else y0 = m_envelope->getTransformedValue(0.0);
const int drawstep = 1;
for (int i = 1; i < getWidth(); ++i)
{
double env_x = 1.0 / getWidth()*i;
double y1 = 0.0;
if (drawTransformed==false)
y1 = m_envelope->GetInterpolatedEnvelopeValue(env_x);
else y1 = m_envelope->getTransformedValue(env_x);
double foo_y0 = (double)getHeight() - jmap<double>(y0, m_view_start_value, m_view_end_value, 0.0, getHeight());
double foo_y1 = (double)getHeight() - jmap<double>(y1, m_view_start_value, m_view_end_value, 0.0, getHeight());
g.drawLine((float)i, foo_y0, (float)i + 1, foo_y1, linethickness);
y0 = y1;
}
};
draw_env(m_env_color, false, 1.0f);
draw_env(Colours::aquamarine.darker(), true, 1.0f);
for (int i = 0; i < m_envelope->GetNumPoints(); ++i)
{
const envelope_point& pt = m_envelope->GetNodeAtIndex(i);
double xcor = jmap(pt.pt_x, m_view_start_time, m_view_end_time, 0.0, (double)getWidth());
double ycor = (double)getHeight() - jmap(pt.pt_y, m_view_start_value, m_view_end_value, 0.0, (double)getHeight());
g.setColour(Colours::white);
if (pt.Status == 0)
g.drawRect((float)xcor - targsize*0.5f, (float)ycor - targsize*0.5f, targsize, targsize, 1.0f);
else g.fillRect((float)xcor - targsize*0.5f, (float)ycor - targsize*0.5f, targsize, targsize);
}
}
void EnvelopeComponent::changeListenerCallback(ChangeBroadcaster*)
{
repaint();
}
void EnvelopeComponent::timerCallback(int)
{
}
void EnvelopeComponent::set_envelope(std::shared_ptr<breakpoint_envelope> env, String name)
{
m_envelope = env;
m_name = name;
repaint();
}
void EnvelopeComponent::mouseDrag(const MouseEvent& ev)
{
if (m_envelope == nullptr)
return;
if (m_segment_drag_info.first >= 0 && ev.mods.isAltDown())
{
double dist = jmap<double>(ev.getDistanceFromDragStartX(), -300.0, 300.0, -1.0, 1.0);
m_envelope->performRelativeTransformation([dist, this](int index, envelope_point& point)
{
if (index == m_segment_drag_info.first)
{
point.ShapeParam1 += dist;
m_segment_drag_info.second = true;
}
});
repaint();
return;
}
if (m_segment_drag_info.first >= 0)
{
double dist = jmap<double>(ev.getDistanceFromDragStartY(), -getHeight(), getHeight(), -1.0, 1.0);
m_envelope->adjustEnvelopeSegmentValues(m_segment_drag_info.first, -dist);
m_envelope->updateMinMaxValues();
repaint();
return;
}
if (m_node_to_drag >= 0)
{
//Logger::writeToLog("trying to move pt " + String(m_node_to_drag));
envelope_point& pt = m_envelope->GetNodeAtIndex(m_node_to_drag);
double left_bound = m_view_start_time;
double right_bound = m_view_end_time;
if (m_node_to_drag > 0 )
{
left_bound = m_envelope->GetNodeAtIndex(m_node_to_drag - 1).pt_x;
}
if (m_node_to_drag < m_envelope->GetNumPoints() - 1)
{
right_bound = m_envelope->GetNodeAtIndex(m_node_to_drag + 1).pt_x;
}
double normx = jmap((double)ev.x - m_touch_offset.x, 0.0, (double)getWidth(), m_view_start_time, m_view_end_time);
double normy = jmap((double)getHeight() - (ev.y - m_touch_offset.y ), 0.0, (double)getHeight(), m_view_start_value, m_view_end_value);
pt.pt_x=jlimit(left_bound+0.001, right_bound - 0.001, normx);
pt.pt_y=jlimit(0.0,1.0,normy);
m_envelope->updateMinMaxValues();
m_last_tip = String(pt.pt_x, 2) + " " + String(pt.pt_y, 2);
show_bubble(ev.x, ev.y, pt);
m_node_that_was_dragged = m_node_to_drag;
repaint();
return;
}
}
void EnvelopeComponent::mouseMove(const MouseEvent & ev)
{
if (m_envelope == nullptr)
return;
m_node_to_drag = find_hot_envelope_point(ev.x, ev.y);
if (m_node_to_drag >= 0)
{
if (m_mouse_down == false)
{
#if !JUCE_IOS
show_bubble(ev.x, ev.y, m_envelope->GetNodeAtIndex(m_node_to_drag));
setMouseCursor(MouseCursor::PointingHandCursor);
#endif
}
}
else
{
int temp = findHotEnvelopeSegment(ev.x, ev.y, true);
if (temp>=0)
setMouseCursor(MouseCursor::UpDownResizeCursor);
else
setMouseCursor(MouseCursor::NormalCursor);
m_bubble.setVisible(false);
}
}
void EnvelopeComponent::showPopupMenu()
{
PopupMenu menu;
PopupMenu::Options opts;
menu.addItem(1, "Reset");
menu.addItem(5, "Delete selected");
menu.addItem(2, "Invert");
menu.addItem(3, "Wrap envelope X transform", true, m_envelope->m_transform_wrap_x);
menu.addItem(4, "Envelope Y random linear interpolation", true, m_envelope->m_transform_y_random_linear_interpolation);
#if JUCE_IOS
opts = opts.withStandardItemHeight(34);
#endif
auto callback = [this] (int r) {
if (r == 1)
{
ScopedLock locker(*m_cs);
m_envelope->ResetEnvelope();
}
else if (r == 2)
{
for (int i = 0; i < m_envelope->GetNumPoints(); ++i)
{
double val = 1.0 - m_envelope->GetNodeAtIndex(i).pt_y;
m_envelope->GetNodeAtIndex(i).pt_y = val;
}
}
else if (r == 3)
{
toggleBool(m_envelope->m_transform_wrap_x);
}
else if (r == 4)
{
toggleBool(m_envelope->m_transform_y_random_linear_interpolation);
}
else if (r == 5)
{
deleteSelectedNodes();
}
repaint();
};
if (!JUCEApplicationBase::isStandaloneApp()) {
opts = opts.withParentComponent(this);
}
menu.showMenuAsync(opts, callback);
}
void EnvelopeComponent::mouseDown(const MouseEvent & ev)
{
if (m_envelope == nullptr)
return;
if (ev.mods.isRightButtonDown() == true)
{
showPopupMenu();
return;
}
m_node_to_drag = find_hot_envelope_point(ev.x, ev.y);
m_touch_offset = {0, 0};
m_mouse_down = true;
m_segment_drag_info = { findHotEnvelopeSegment(ev.x, ev.y, true),false };
if (m_segment_drag_info.first >= 0)
{
m_envelope->beginRelativeTransformation();
return;
}
if (m_node_to_drag >= 0 && ev.mods.isAltDown() == true)
{
if (m_envelope->GetNumPoints() < 2)
{
m_bubble.showAt({ ev.x,ev.y, 0,0 }, AttributedString("Can't remove last node"), 3000, false, false);
return;
}
m_cs->enter();
m_envelope->DeleteNode(m_node_to_drag);
m_cs->exit();
m_envelope->updateMinMaxValues();
m_node_to_drag = -1;
OnEnvelopeEdited(m_envelope.get());
repaint();
return;
}
if (m_node_to_drag >= 0)
{
const envelope_point& pt = m_envelope->GetNodeAtIndex(m_node_to_drag);
double xcor = jmap(pt.pt_x, m_view_start_time, m_view_end_time, 0.0, (double)getWidth());
double ycor = (double)getHeight() - jmap(pt.pt_y, m_view_start_value, m_view_end_value, 0.0, (double)getHeight());
m_touch_offset = { ev.x - (int)xcor , ev.y - (int)ycor };
if (!ev.mods.isShiftDown()) {
m_envelope->SetNodeStatusForAll(0);
}
int oldstatus = m_envelope->GetNodeAtIndex(m_node_to_drag).Status;
if (oldstatus==0)
m_envelope->GetNodeAtIndex(m_node_to_drag).Status=1;
else m_envelope->GetNodeAtIndex(m_node_to_drag).Status=0;
repaint();
return;
}
if (m_node_to_drag == -1)
{
double normx = jmap((double)ev.x, 0.0, (double)getWidth(), m_view_start_time, m_view_end_time);
double normy = jmap((double)getHeight() - ev.y, 0.0, (double)getHeight(), m_view_start_value, m_view_end_value);
m_cs->enter();
m_envelope->AddNode ({ normx,normy, 0.5});
m_envelope->SortNodes();
m_cs->exit();
m_envelope->updateMinMaxValues();
m_node_to_drag = find_hot_envelope_point(ev.x, ev.y);
// deselect all
m_envelope->SetNodeStatusForAll(0);
m_envelope->GetNodeAtIndex(m_node_to_drag).Status=1; // select it
//m_mouse_down = false;
OnEnvelopeEdited(m_envelope.get());
repaint();
}
}
void EnvelopeComponent::mouseUp(const MouseEvent &ev)
{
#if JUCE_IOS
m_bubble.setVisible(false);
#endif
if (ev.mods == ModifierKeys::noModifiers) {
m_bubble.setVisible(false);
}
if (m_node_that_was_dragged >= 0 || m_segment_drag_info.second==true)
{
OnEnvelopeEdited(m_envelope.get());
}
m_mouse_down = false;
m_node_that_was_dragged = -1;
m_node_to_drag = -1;
if (m_segment_drag_info.second == true)
{
m_segment_drag_info = { -1,false };
m_envelope->endRelativeTransformation();
}
}
void EnvelopeComponent::deleteSelectedNodes()
{
m_node_to_drag = -1;
{
ScopedLock locker(*m_cs);
m_envelope->removePointsConditionally([](const envelope_point& pt) { return pt.Status == 1; });
if (m_envelope->GetNumPoints() == 0)
m_envelope->AddNode({ 0.0,0.5 });
}
OnEnvelopeEdited(m_envelope.get());
}
bool EnvelopeComponent::keyPressed(const KeyPress & ev)
{
if (m_envelope == nullptr)
return false;
auto f = [this](auto& env_var, double amt)
{
env_var+=amt;
return true;
};
bool r = false;
if (ev == 'Q')
r = f(m_envelope->m_transform_x_shift,-0.01);
if (ev == 'W')
r = f(m_envelope->m_transform_x_shift,0.01);
if (ev == 'E')
r = f(m_envelope->m_transform_y_shift,0.01);
if (ev == 'D')
r = f(m_envelope->m_transform_y_shift,-0.01);
if (ev == 'R')
r = f(m_envelope->m_transform_y_scale,0.05);
if (ev == 'F')
r = f(m_envelope->m_transform_y_scale,-0.05);
if (ev == 'T')
r = f(m_envelope->m_transform_y_sinus,0.01);
if (ev == 'G')
r = f(m_envelope->m_transform_y_sinus,-0.01);
if (ev == 'Y')
r = f(m_envelope->m_transform_y_tilt,0.02);
if (ev == 'H')
r = f(m_envelope->m_transform_y_tilt,-0.02);
if (ev == 'V')
r = f(m_envelope->m_transform_y_sinus_freq,1.0);
if (ev == 'B')
r = f(m_envelope->m_transform_y_sinus_freq,-1.0);
m_envelope->m_transform_y_sinus_freq = jlimit(1.0,64.0, m_envelope->m_transform_y_sinus_freq);
if (r==true)
{
repaint();
return true;
}
if (ev == 'A')
{
m_envelope->m_transform_x_shift = 0.0;
m_envelope->m_transform_y_scale = 1.0;
m_envelope->m_transform_y_shift = 0.0;
m_envelope->m_transform_y_sinus = 0.0;
repaint();
return true;
}
if (ev == KeyPress::deleteKey)
{
deleteSelectedNodes();
repaint();
return true;
}
return false;
}
int EnvelopeComponent::find_hot_envelope_point(double xcor, double ycor)
{
if (m_envelope == nullptr)
return -1;
for (int i = 0; i < m_envelope->GetNumPoints(); ++i)
{
const envelope_point& pt = m_envelope->GetNodeAtIndex(i);
double ptxcor = jmap(pt.pt_x, m_view_start_time, m_view_end_time, 0.0, (double)getWidth());
double ptycor = (double)getHeight() - jmap(pt.pt_y, m_view_start_value, m_view_end_value, 0.0, (double)getHeight());
float targsize = 10.0;
#if JUCE_IOS
targsize = 20;
#endif
juce::Rectangle<double> target(ptxcor - targsize*0.5f, ptycor - targsize*0.5, targsize, targsize);
if (target.contains(xcor, ycor) == true)
{
return i;
}
}
return -1;
}
int EnvelopeComponent::findHotEnvelopeSegment(double xcor, double ycor, bool detectsegment)
{
float targsize = 10.0;
#if JUCE_IOS
targsize = 20.0;
#endif
if (m_envelope == nullptr)
return -1;
for (int i = 0; i < m_envelope->GetNumPoints()-1; ++i)
{
const envelope_point& pt0 = m_envelope->GetNodeAtIndex(i);
const envelope_point& pt1 = m_envelope->GetNodeAtIndex(i+1);
float xcor0 = (float)jmap<double>(pt0.pt_x, m_view_start_time, m_view_end_time, 0.0, getWidth());
float xcor1 = (float)jmap<double>(pt1.pt_x, m_view_start_time, m_view_end_time, 0.0, getWidth());
float segwidth = xcor1 - xcor0;
juce::Rectangle<float> segrect(xcor0+targsize, 0.0f, segwidth-2*targsize, (float)getHeight());
if (segrect.contains((float)xcor, (float)ycor))
{
if (detectsegment == false)
return i;
else
{
double normx = jmap<double>(xcor, 0.0, getWidth(), m_view_start_time, m_view_end_time);
double yval = m_envelope->GetInterpolatedEnvelopeValue(normx);
float ycor0 = (float)(getHeight()-jmap<double>(yval, 0.0, 1.0, 0.0, getHeight()));
juce::Rectangle<float> segrect2((float)(xcor - 20), (float)(ycor - 10), 40, 20);
if (segrect2.contains((float)xcor, ycor0))
return i;
}
}
}
return -1;
}