region gain and MIDI CC freehand line drawing

This involved a significant change in event handling for automation region views,
but it brings it into line with how it works for other things. On button press
we initiate a drag, then if no motion occurs, the Drag returns false during
finalization, and only then do we continue through Editor::button_release_handler()
to eventually end up in ::add_automation_event().

Although it is a substantial change, the fact that it now works the same
way for audio regions, automation regions and automation tracks seems
like a definite plus.
This commit is contained in:
Paul Davis 2023-09-30 11:09:01 -06:00
parent 6b64ebdb27
commit 18819a48a9
11 changed files with 270 additions and 46 deletions

View File

@ -63,6 +63,7 @@
#include "public_editor.h"
#include "audio_region_editor.h"
#include "audio_streamview.h"
#include "mergeable_line.h"
#include "region_gain_line.h"
#include "control_point.h"
#include "ghostregion.h"
@ -276,6 +277,8 @@ AudioRegionView::init (bool wfd)
setup_waveform_visibility ();
get_canvas_frame()->set_data ("linemerger", (LineMerger*) this);
/* XXX sync mark drag? */
}
@ -1850,3 +1853,9 @@ AudioRegionView::parameter_changed (string const & p)
setup_waveform_visibility ();
}
}
MergeableLine*
AudioRegionView::make_merger ()
{
return new MergeableLine (gain_line, std::shared_ptr<AutomationControl>(), boost::bind (&Region::absolute_time_to_region_time, _region, _1), nullptr, nullptr);
}

View File

@ -38,6 +38,7 @@
#include "waveview/wave_view.h"
#include "line_merger.h"
#include "region_view.h"
#include "time_axis_view_item.h"
#include "automation_line.h"
@ -54,7 +55,7 @@ class GhostRegion;
class AutomationTimeAxisView;
class RouteTimeAxisView;
class AudioRegionView : public RegionView
class AudioRegionView : public RegionView, public LineMerger
{
public:
AudioRegionView (ArdourCanvas::Container *,
@ -147,6 +148,8 @@ public:
return _end_xfade_visible;
}
MergeableLine* make_merger();
protected:
/* this constructor allows derived types

View File

@ -38,6 +38,7 @@
#include "editor.h"
#include "editor_drag.h"
#include "gui_thread.h"
#include "mergeable_line.h"
#include "midi_automation_line.h"
#include "public_editor.h"
#include "ui_config.h"
@ -88,6 +89,8 @@ AutomationRegionView::init (bool /*wfd*/)
set_height (trackview.current_height());
set_colors ();
get_canvas_frame()->set_data ("linemerger", (LineMerger*) this);
}
void
@ -132,43 +135,37 @@ AutomationRegionView::canvas_group_event (GdkEvent* ev)
return false;
}
return RegionView::canvas_group_event (ev);
}
void
AutomationRegionView::add_automation_event (GdkEvent* ev)
{
double x = ev->button.x;
double y = ev->button.y;
/* convert to item coordinates in the time axis view */
automation_view()->canvas_display()->canvas_to_item (x, y);
/* clamp y */
y = std::max (y, 0.0);
y = std::min (y, _height - NAME_HIGHLIGHT_SIZE);
/* the time domain doesn't matter here, because the automation
* list will force the position to its own time domain when
* adding the point.
*/
PublicEditor& e = trackview.editor ();
if (ev->type == GDK_BUTTON_RELEASE &&
ev->button.button == 1 &&
(e.current_mouse_mode() == Editing::MouseDraw || e.current_mouse_mode() == Editing::MouseObject) &&
!e.drags()->active()) {
double x = ev->button.x;
double y = ev->button.y;
/* convert to item coordinates in the time axis view */
automation_view()->canvas_display()->canvas_to_item (x, y);
/* clamp y */
y = std::max (y, 0.0);
y = std::min (y, _height - NAME_HIGHLIGHT_SIZE);
/* guard points only if primary modifier is used */
bool with_guard_points = Gtkmm2ext::Keyboard::modifier_state_equals (ev->button.state, Gtkmm2ext::Keyboard::PrimaryModifier);
/* the time domain doesn't matter here, because the automation
* list will force the position to its own time domain when
* adding the point.
*/
add_automation_event (ev, timepos_t (e.pixel_to_sample (x)), y, with_guard_points);
return true;
}
return RegionView::canvas_group_event (ev);
add_automation_event (timepos_t (e.pixel_to_sample (x)), y, false);
}
/** @param when Position is global time position
* @param y y position, relative to our TimeAxisView.
*/
void
AutomationRegionView::add_automation_event (GdkEvent *, timepos_t const & w, double y, bool with_guard_points)
AutomationRegionView::add_automation_event (timepos_t const & w, double y, bool with_guard_points)
{
std::shared_ptr<Evoral::Control> c = _region->control(_parameter, true);
std::shared_ptr<ARDOUR::AutomationControl> ac = std::dynamic_pointer_cast<ARDOUR::AutomationControl>(c);
@ -352,3 +349,17 @@ AutomationRegionView::set_selected (bool yn)
group->raise_to_top ();
}
}
timepos_t
AutomationRegionView::drawn_time_filter (timepos_t const & t)
{
return timepos_t (_region->absolute_time_to_source_beats (t));
}
MergeableLine*
AutomationRegionView::make_merger()
{
std::shared_ptr<Evoral::Control> c = _region->control(_parameter, true);
std::shared_ptr<ARDOUR::AutomationControl> ac = std::dynamic_pointer_cast<ARDOUR::AutomationControl>(c);
return new MergeableLine (_line, ac, boost::bind (&AutomationRegionView::drawn_time_filter, this, _1), nullptr, nullptr);
}

View File

@ -27,6 +27,7 @@
#include "automation_time_axis.h"
#include "automation_line.h"
#include "enums.h"
#include "line_merger.h"
namespace ARDOUR {
class AutomationList;
@ -35,7 +36,7 @@ namespace ARDOUR {
class TimeAxisView;
class AutomationRegionView : public RegionView
class AutomationRegionView : public RegionView, public LineMerger
{
public:
AutomationRegionView(ArdourCanvas::Container*,
@ -72,12 +73,17 @@ public:
void tempo_map_changed ();
MergeableLine* make_merger ();
void add_automation_event (GdkEvent* event);
Temporal::timepos_t drawn_time_filter (Temporal::timepos_t const &);
protected:
void create_line(std::shared_ptr<ARDOUR::AutomationList> list);
bool set_position(Temporal::timepos_t const & pos, void* src, double* ignored);
void region_resized (const PBD::PropertyChange&);
bool canvas_group_event(GdkEvent* ev);
void add_automation_event (GdkEvent* event, Temporal::timepos_t const & when, double y, bool with_guard_points);
void add_automation_event (Temporal::timepos_t const & when, double y, bool with_guard_points);
void mouse_mode_changed ();
void entered();
void exited();

View File

@ -79,11 +79,12 @@ AutomationStreamView::add_region_view_internal (std::shared_ptr<Region> region,
return 0;
}
std::shared_ptr<AutomationList> list;
const std::shared_ptr<AutomationControl> control = std::dynamic_pointer_cast<AutomationControl> (
region->control (_automation_view.parameter(), true)
);
std::shared_ptr<AutomationList> list;
if (control) {
list = std::dynamic_pointer_cast<AutomationList>(control->list());
if (control->list() && !list) {

View File

@ -167,6 +167,7 @@ AutomationTimeAxisView::AutomationTimeAxisView (
_base_rect->set_outline (false);
_base_rect->set_fill_color (UIConfiguration::instance().color_mod (fill_color_name, "automation track fill"));
_base_rect->set_data ("trackview", this);
_base_rect->set_data ("linemerger", (LineMerger*) this);
_base_rect->Event.connect (sigc::bind (sigc::mem_fun (_editor, &PublicEditor::canvas_automation_track_event), _base_rect, this));
if (!a) {
_base_rect->lower_to_bottom();
@ -1285,6 +1286,6 @@ AutomationTimeAxisView::set_selected_regionviews (RegionSelection& rs)
MergeableLine*
AutomationTimeAxisView::make_merger ()
{
return new MergeableLine (_line, _control, boost::bind (&AutomationTimeAxisView::set_automation_state, this, _1), boost::bind (RouteTimeAxisView::signal_ctrl_touched, false));
return new MergeableLine (_line, _control, nullptr, boost::bind (&AutomationTimeAxisView::set_automation_state, this, _1), boost::bind (RouteTimeAxisView::signal_ctrl_touched, false));
}

View File

@ -7329,10 +7329,20 @@ FreehandLineDrag<OrderedPointList,OrderedPoint>::maybe_add_point (GdkEvent* ev,
ArdourCanvas::Rect r = base_rect.item_to_canvas (base_rect.get());
/* Adjust event coordinates to be relative to the base rectangle */
double x = pointer_x - r.x0;
double y = ev->motion.y - r.y0;
x = std::max (0., x);
if (x < 0) {
dragging_line->clear ();
drawn_points.clear ();
edge_x = 0;
return;
}
/* Clamp y coordinate to the area of the base rect */
y = std::max (0., std::min (r.height(), y));
bool add_point = false;
@ -7341,7 +7351,7 @@ FreehandLineDrag<OrderedPointList,OrderedPoint>::maybe_add_point (GdkEvent* ev,
const bool line = Keyboard::modifier_state_equals (ev->motion.state, Keyboard::PrimaryModifier);
if (direction > 0) {
if (line || (pointer_x > edge_x) || (pointer_x == edge_x && ev->motion.y != last_pointer_y())) {
if (x < r.width() && (line || (pointer_x > edge_x) || (pointer_x == edge_x && ev->motion.y != last_pointer_y()))) {
if (line && dragging_line->get().size() > 1) {
pop_point = true;
@ -7352,7 +7362,7 @@ FreehandLineDrag<OrderedPointList,OrderedPoint>::maybe_add_point (GdkEvent* ev,
} else if (direction < 0) {
if (line || (pointer_x < edge_x) || (pointer_x == edge_x && ev->motion.y != last_pointer_y())) {
if (x >= 0. && (line || (pointer_x < edge_x) || (pointer_x == edge_x && ev->motion.y != last_pointer_y()))) {
if (line && dragging_line->get().size() > 1) {
pop_point = true;
@ -7461,15 +7471,15 @@ AutomationDrawDrag::finished (GdkEvent* event, bool motion_occured)
return;
}
AutomationTimeAxisView* atv = static_cast<AutomationTimeAxisView*>(base_rect.get_data ("trackview"));
LineMerger* lm = static_cast<LineMerger*>(base_rect.get_data ("linemerger"));
if (!atv) {
if (!lm) {
return;
}
FreehandLineDrag<Evoral::ControlList::OrderedPoints,Evoral::ControlList::OrderedPoint>::finished (event, motion_occured);
MergeableLine* ml = atv->make_merger ();
MergeableLine* ml = lm->make_merger();
ml->merge_drawn_line (*_editor, *_editor->session(), drawn_points, !did_snap);
delete ml;
}

View File

@ -1388,7 +1388,7 @@ Editor::button_press_handler_1 (ArdourCanvas::Item* item, GdkEvent* event, ItemT
case RegionItem: {
RegionView* rv;
if ((rv = dynamic_cast<RegionView*> (clicked_regionview))) {
ArdourCanvas::Rectangle* r = static_cast <ArdourCanvas::Rectangle*> (rv->get_canvas_frame());
ArdourCanvas::Rectangle* r = dynamic_cast<ArdourCanvas::Rectangle*> (rv->get_canvas_frame());
_drags->set (new AutomationDrawDrag (this, *r, Temporal::AudioTime), event);
}
}
@ -1925,10 +1925,20 @@ Editor::button_release_handler (ArdourCanvas::Item* item, GdkEvent* event, ItemT
points when doing this.
*/
AudioRegionView* arv = dynamic_cast<AudioRegionView*> (clicked_regionview);
if (!were_dragging && arv) {
bool with_guard_points = Keyboard::modifier_state_equals (event->button.state, Keyboard::PrimaryModifier);
arv->add_gain_point_event (item, event, with_guard_points);
if (!were_dragging) {
if (arv) {
bool with_guard_points = Keyboard::modifier_state_equals (event->button.state, Keyboard::PrimaryModifier);
arv->add_gain_point_event (item, event, with_guard_points);
}
} else {
AutomationRegionView* atv = dynamic_cast<AutomationRegionView*> (clicked_regionview);
if (atv) {
atv->add_automation_event (event);
}
}
return true;
break;
}
@ -3116,7 +3126,7 @@ Editor::choose_mapping_drag (ArdourCanvas::Item* item, GdkEvent* event)
if (after_after) {
after = after_after;
} else {
at_end = true;
at_end = true;
}
} else if (ramped) {

View File

@ -0,0 +1,111 @@
/*
* Copyright (C) 2023 Paul Davis <paul@linuxaudiosystems.com>
*
* 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 2 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, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include <algorithm>
#include "ardour/session.h"
#include "automation_line.h"
#include "editor.h"
#include "mergeable_line.h"
#include "route_time_axis.h"
#include "selectable.h"
#include "ui_config.h"
#include "pbd/i18n.h"
using namespace ARDOUR;
void
MergeableLine::merge_drawn_line (Editor& e, Session& s, Evoral::ControlList::OrderedPoints& points, bool thin)
{
if (points.empty()) {
return;
}
if (!_line) {
return;
}
std::shared_ptr<AutomationList> list = _line->the_list ();
if (list->in_write_pass()) {
/* do not allow the GUI to add automation events during an
automation write pass.
*/
return;
}
XMLNode& before = list->get_state();
std::list<Selectable*> results;
/* If necessary convert all point times. This is necessary
for region-based automation data, because the time values for the
points drawn are in absolute time, but the ControlList expects data
in source-reference time.
*/
if (time_filter) {
for (auto & p : points) {
p.when = time_filter (p.when);
}
}
Temporal::timepos_t earliest = points.front().when;
Temporal::timepos_t latest = points.back().when;
if (earliest > latest) {
std::swap (earliest, latest);
}
/* Convert each point's "value" from geometric coordinate space to
* value space for the control
*/
for (auto & dp : points) {
/* compute vertical fractional position */
dp.value = 1.0 - (dp.value / _line->height());
/* map using line */
_line->view_to_model_coord_y (dp.value);
}
list->freeze ();
list->editor_add_ordered (points, false);
if (thin) {
list->thin (Config->get_automation_thinning_factor());
}
list->thaw ();
if (_control && _control->automation_state () == ARDOUR::Off) {
automation_state_callback (ARDOUR::Play);
}
if (UIConfiguration::instance().get_automation_edit_cancels_auto_hide () && _control == s.recently_touched_controllable ()) {
control_touched_callback ();
}
XMLNode& after = list->get_state();
e.begin_reversible_command (_("draw automation"));
s.add_command (new MementoCommand<ARDOUR::AutomationList> (*list.get (), &before, &after));
_line->get_selectables (earliest, latest, 0.0, 1.0, results);
e.get_selection ().set (results);
e.commit_reversible_command ();
s.set_dirty ();
}

View File

@ -0,0 +1,63 @@
/*
* Copyright (C) 2023 Paul Davis <paul@linuxaudiosystems.com>
*
* 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 2 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, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifndef __gtk2_ardour_mergeable_line__
#define __gtk2_ardour_mergeable_line__
#include <functional>
#include <memory>
#include "evoral/ControlList.h"
#include "ardour/types.h"
class AutomationLine;
class RouteTimeAxisView;
class Editor;
namespace ARDOUR {
class Session;
class AutomationControl;
}
class MergeableLine
{
public:
MergeableLine (std::shared_ptr<AutomationLine> l, std::shared_ptr<ARDOUR::AutomationControl> c,
std::function<Temporal::timepos_t(Temporal::timepos_t const &)> tf,
std::function<void(ARDOUR::AutoState)> asc,
std::function<void()> ctc)
: _line (l)
, _control (c)
, time_filter (tf)
, automation_state_callback (asc)
, control_touched_callback (ctc) {}
virtual ~MergeableLine() {}
void merge_drawn_line (Editor& e, ARDOUR::Session& s, Evoral::ControlList::OrderedPoints& points, bool thin);
private:
std::shared_ptr<AutomationLine> _line;
std::shared_ptr<ARDOUR::AutomationControl> _control;
std::function<Temporal::timepos_t(Temporal::timepos_t const &)> time_filter;
std::function<void(ARDOUR::AutoState)> automation_state_callback;
std::function<void()> control_touched_callback;
};
#endif /* __gtk2_ardour_mergeable_line__ */

View File

@ -1492,7 +1492,6 @@ MidiTimeAxisView::create_automation_child (const Evoral::Parameter& param, bool
_route->describe_parameter(param)));
if (_view) {
std::cerr << "Adding ghosts of each MIDI region\n";
_view->foreach_regionview (sigc::mem_fun (*track.get(), &TimeAxisView::add_ghost));
}