ardour/gtk2_ardour/audio_clip_editor.cc

424 lines
9.7 KiB
C++
Raw Normal View History

/*
* Copyright (C) 2011-2017 Paul Davis <paul@linuxaudiosystems.com>
* Copyright (C) 2021 Ben Loftis <ben@harrisonconsoles.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 "pbd/compose.h"
#include "gtkmm2ext/gui_thread.h"
#include "gtkmm2ext/utils.h"
#include "gtkmm2ext/actions.h"
#include "canvas/canvas.h"
#include "canvas/debug.h"
#include "canvas/utils.h"
#include "waveview/wave_view.h"
#include "ardour/audioregion.h"
#include "ardour/location.h"
#include "ardour/profile.h"
#include "ardour/region_factory.h"
#include "ardour/session.h"
#include "ardour/source.h"
#include "widgets/ardour_button.h"
#include "widgets/ardour_icon.h"
#include "audio_clip_editor.h"
#include "audio_clock.h"
#include "automation_line.h"
#include "control_point.h"
#include "editor.h"
#include "region_view.h"
#include "ui_config.h"
#include "pbd/i18n.h"
using namespace Gtk;
using namespace ARDOUR;
using namespace ArdourCanvas;
using namespace ArdourWaveView;
using namespace ArdourWidgets;
using std::min;
using std::max;
/* ------------ */
AudioClipEditor::AudioClipEditor ()
: _spp (0)
, _current_drag (0)
{
const double scale = UIConfiguration::instance().get_ui_scale();
frame = new Rectangle (root());
frame->name = "audio clip editor frame";
frame->set_fill (false);
frame->Event.connect (sigc::mem_fun (*this, &AudioClipEditor::event_handler));
2021-12-10 11:47:56 -05:00
waves_container = new ArdourCanvas::ScrollGroup (frame, ScrollGroup::ScrollsHorizontally);
2021-12-10 11:47:56 -05:00
line_container = new ArdourCanvas::Container (frame);
const double line_width = 3.;
2021-12-10 11:47:56 -05:00
start_line = new Line (line_container);
start_line->set_outline_width (line_width * scale);
2021-12-10 11:47:56 -05:00
end_line = new Line (line_container);
end_line->set_outline_width (line_width * scale);
2021-12-10 11:47:56 -05:00
loop_line = new Line (line_container);
loop_line->set_outline_width (line_width * scale);
start_line->Event.connect (sigc::bind (sigc::mem_fun (*this, &AudioClipEditor::line_event_handler), start_line));
end_line->Event.connect (sigc::bind (sigc::mem_fun (*this, &AudioClipEditor::line_event_handler), end_line));
loop_line->Event.connect (sigc::bind (sigc::mem_fun (*this, &AudioClipEditor::line_event_handler), loop_line));
/* hide lines until there is a region */
2021-12-10 11:47:56 -05:00
line_container->hide ();
set_colors ();
}
AudioClipEditor::~AudioClipEditor ()
{
drop_waves ();
}
bool
AudioClipEditor::line_event_handler (GdkEvent* ev, ArdourCanvas::Line* l)
{
std::cerr << "event type " << Gtkmm2ext::event_type_string (ev->type) << " on line " << std::endl;
switch (ev->type) {
case GDK_BUTTON_PRESS:
_current_drag = new LineDrag (*this, *l);
return true;
case GDK_BUTTON_RELEASE:
if (_current_drag) {
_current_drag->end (&ev->button);
delete _current_drag;
_current_drag = 0;
return true;
}
break;
case GDK_MOTION_NOTIFY:
if (_current_drag) {
_current_drag->motion (&ev->motion);
return true;
}
break;
default:
break;
}
return false;
}
void
AudioClipEditor::position_lines ()
{
if (!audio_region) {
return;
}
start_line->set_x0 (sample_to_pixel (audio_region->start().samples()));
start_line->set_x1 (sample_to_pixel (audio_region->start().samples()));
end_line->set_x0 (sample_to_pixel (audio_region->end().samples()));
end_line->set_x1 (sample_to_pixel (audio_region->end().samples()));
}
double
AudioClipEditor::sample_to_pixel (samplepos_t s)
{
return s / _spp;
}
samplepos_t
AudioClipEditor::pixel_to_sample (double p)
{
return p * _spp;
}
AudioClipEditor::LineDrag::LineDrag (AudioClipEditor& ed, ArdourCanvas::Line& l)
: editor (ed)
, line (l)
{
line.grab();
}
void
AudioClipEditor::LineDrag::begin (GdkEventButton* ev)
{
}
void
AudioClipEditor::LineDrag::end (GdkEventButton* ev)
{
line.ungrab();
std::cerr << "grab end\n";
}
void
AudioClipEditor::LineDrag::motion (GdkEventMotion* ev)
{
line.set_x0 (ev->x);
line.set_x1 (ev->x);
std::cerr << "move to " << ev->x << ", " << ev->y << std::endl;
}
2021-12-10 11:47:56 -05:00
void
AudioClipEditor::set_colors ()
{
set_background_color (UIConfiguration::instance().color (X_("theme:bg")));
frame->set_outline_color (UIConfiguration::instance().color (X_("neutral:midground")));
2021-12-10 11:47:56 -05:00
start_line->set_outline_color (UIConfiguration::instance().color (X_("theme:contrasting clock")));
end_line->set_outline_color (UIConfiguration::instance().color (X_("theme:contrasting alt")));
loop_line->set_outline_color (UIConfiguration::instance().color (X_("theme:contrasting selection")));
set_waveform_colors ();
}
void
AudioClipEditor::drop_waves ()
{
for (auto & wave : waves) {
delete wave;
}
waves.clear ();
}
void
AudioClipEditor::set_region (boost::shared_ptr<AudioRegion> r)
{
drop_waves ();
audio_region = r;
uint32_t n_chans = r->n_channels ();
samplecnt_t len;
len = r->source (0)->length().samples ();
for (uint32_t n = 0; n < n_chans; ++n) {
boost::shared_ptr<Region> wr = RegionFactory::get_whole_region_for_source (r->source (n));
if (!wr) {
continue;
}
boost::shared_ptr<AudioRegion> war = boost::dynamic_pointer_cast<AudioRegion> (wr);
if (!war) {
continue;
}
2021-12-10 11:47:56 -05:00
WaveView* wv = new WaveView (waves_container, war);
wv->set_channel (n);
wv->set_show_zero_line (false);
wv->set_clip_level (1.0);
waves.push_back (wv);
}
set_spp_from_length (len);
set_wave_heights (frame->get().height() - 2.0);
set_waveform_colors ();
2021-12-10 11:47:56 -05:00
line_container->show ();
}
void
AudioClipEditor::on_size_allocate (Gtk::Allocation& alloc)
{
GtkCanvas::on_size_allocate (alloc);
ArdourCanvas::Rect r (1, 1, alloc.get_width() - 2, alloc.get_height() - 2);
frame->set (r);
position_lines ();
2021-12-10 11:47:56 -05:00
start_line->set_y1 (frame->get().height() - 2.);
end_line->set_y1 (frame->get().height() - 2.);
loop_line->set_y1 (frame->get().height() - 2.);
set_wave_heights (r.height() - 2.0);
}
void
AudioClipEditor::set_spp (double samples_per_pixel)
{
_spp = samples_per_pixel;
position_lines ();
for (auto & wave : waves) {
wave->set_samples_per_pixel (_spp);
}
}
void
AudioClipEditor::set_spp_from_length (samplecnt_t len)
{
double available_width = frame->get().width();
double s = floor (len / available_width);
set_spp (s);
}
void
AudioClipEditor::set_wave_heights (int h)
{
if (waves.empty()) {
return;
}
uint32_t n = 0;
Distance ht = h / waves.size();
std::cerr << "wave heights: " << ht << std::endl;
for (auto & wave : waves) {
wave->set_height (ht);
wave->set_y_position (n * ht);
++n;
}
}
void
AudioClipEditor::set_waveform_colors ()
{
Gtkmm2ext::Color clip = UIConfiguration::instance().color ("clipped waveform");
Gtkmm2ext::Color zero = UIConfiguration::instance().color ("zero line");
Gtkmm2ext::Color fill = UIConfiguration::instance().color ("waveform fill");
Gtkmm2ext::Color outline = UIConfiguration::instance().color ("waveform outline");
for (auto & wave : waves) {
wave->set_fill_color (fill);
wave->set_outline_color (outline);
wave->set_clip_color (clip);
wave->set_zero_color (zero);
}
}
bool
AudioClipEditor::event_handler (GdkEvent* ev)
{
switch (ev->type) {
case GDK_BUTTON_PRESS:
// PublicEditor::instance().get_selection().set (this);
break;
case GDK_ENTER_NOTIFY:
// redraw ();
break;
case GDK_LEAVE_NOTIFY:
// redraw ();
break;
default:
break;
}
return false;
}
/* ====================================================== */
AudioClipEditorBox::AudioClipEditorBox ()
{
_header_label.set_text(_("AUDIO Region Trimmer:"));
_header_label.set_alignment(0.0, 0.5);
zoom_in_button.set_icon (ArdourIcon::ZoomIn);
zoom_out_button.set_icon (ArdourIcon::ZoomOut);
zoom_in_button.signal_clicked.connect (sigc::mem_fun (*this, &AudioClipEditorBox::zoom_in_click));
zoom_out_button.signal_clicked.connect (sigc::mem_fun (*this, &AudioClipEditorBox::zoom_out_click));
header_box.pack_start (_header_label, false, false);
header_box.pack_start (zoom_in_button, false, false);
header_box.pack_start (zoom_out_button, false, false);
pack_start(header_box, false, false, 6);
editor = manage (new AudioClipEditor);
editor->set_size_request(600,120);
pack_start(*editor, true, true);
editor->show();
}
AudioClipEditorBox::~AudioClipEditorBox ()
{
delete editor;
}
void
AudioClipEditorBox::zoom_in_click ()
{
editor->set_spp (editor->spp() / 2.);
}
void
AudioClipEditorBox::zoom_out_click ()
{
editor->set_spp (editor->spp() * 2.);
}
void
AudioClipEditorBox::set_region (boost::shared_ptr<Region> r)
{
boost::shared_ptr<AudioRegion> ar = boost::dynamic_pointer_cast<AudioRegion> (r);
if (!ar) {
return;
}
set_session(&r->session());
state_connection.disconnect();
_region = r;
editor->set_region (ar);
PBD::PropertyChange interesting_stuff;
region_changed(interesting_stuff);
_region->PropertyChanged.connect (state_connection, invalidator (*this), boost::bind (&AudioClipEditorBox::region_changed, this, _1), gui_context());
}
void
AudioClipEditorBox::region_changed (const PBD::PropertyChange& what_changed)
{
//ToDo: refactor the region_editor.cc to cover this basic stuff
// if (what_changed.contains (ARDOUR::Properties::name)) {
// name_changed ();
// }
// PBD::PropertyChange interesting_stuff;
// interesting_stuff.add (ARDOUR::Properties::length);
// interesting_stuff.add (ARDOUR::Properties::start);
// if (what_changed.contains (interesting_stuff))
{
}
}