/*
- Copyright (C) 2003 Paul Davis
+ Copyright (C) 2003 Paul Davis
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
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- $Id$
*/
#include <algorithm>
-#include <ardour/region.h>
-#include <gtkmmext/doi.h>
+#include "ardour/region.h"
+#include <gtkmm2ext/doi.h>
#include "canvas-simplerect.h"
#include "canvas-curve.h"
#include "crossfade_view.h"
+#include "gui_thread.h"
#include "rgb_macros.h"
#include "audio_time_axis.h"
#include "public_editor.h"
-#include "regionview.h"
+#include "audio_region_view.h"
#include "utils.h"
+#include "canvas_impl.h"
+#include "ardour_ui.h"
-using namespace SigC;
using namespace ARDOUR;
+using namespace PBD;
using namespace Editing;
+using namespace Gnome;
+using namespace Canvas;
-SigC::Signal1<void,CrossfadeView*> CrossfadeView::GoingAway;
+PBD::Signal1<void,CrossfadeView*> CrossfadeView::CatchDeletion;
-CrossfadeView::CrossfadeView (GtkCanvasGroup *parent,
- AudioTimeAxisView &tv,
- Crossfade& xf,
+CrossfadeView::CrossfadeView (ArdourCanvas::Group *parent,
+ RouteTimeAxisView &tv,
+ boost::shared_ptr<Crossfade> xf,
double spu,
- GdkColor& basic_color,
+ Gdk::Color& basic_color,
AudioRegionView& lview,
AudioRegionView& rview)
-
- : TimeAxisViewItem ("xf.name()", parent, tv, spu, basic_color, xf.position(),
- xf.overlap_length(), TimeAxisViewItem::Visibility (TimeAxisViewItem::ShowFrame)),
+
+ : TimeAxisViewItem ("xfade" /*xf.name()*/, *parent, tv, spu, basic_color, xf->position(),
+ xf->length(), false, TimeAxisViewItem::Visibility (TimeAxisViewItem::ShowFrame)),
crossfade (xf),
left_view (lview),
- right_view (rview)
-
+ right_view (rview)
{
_valid = true;
_visible = true;
- fade_in = gtk_canvas_item_new (GTK_CANVAS_GROUP(group),
- gtk_canvas_line_get_type(),
- "fill_color_rgba", color_map[cCrossfadeLine],
- "width_pixels", (guint) 1,
- NULL);
-
- fade_out = gtk_canvas_item_new (GTK_CANVAS_GROUP(group),
- gtk_canvas_line_get_type(),
- "fill_color_rgba", color_map[cCrossfadeLine],
- "width_pixels", (guint) 1,
- NULL);
-
- set_height (get_time_axis_view().height);
+ fade_in = new Line (*group);
+ fade_in->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_CrossfadeLine.get();
+ fade_in->property_width_pixels() = 1;
+
+ fade_out = new Line (*group);
+ fade_out->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_CrossfadeLine.get();
+ fade_out->property_width_pixels() = 1;
/* no frame around the xfade or overlap rects */
- gtk_canvas_item_set (frame, "outline_what", 0, NULL);
+ frame->property_outline_what() = 0;
/* never show the vestigial frame */
-
- gtk_canvas_item_hide (vestigial_frame);
+ vestigial_frame->hide();
show_vestigial = false;
- gtk_object_set_data (GTK_OBJECT(group), "crossfadeview", this);
- gtk_signal_connect (GTK_OBJECT(group), "event",
- (GtkSignalFunc) PublicEditor::canvas_crossfade_view_event,
- this);
+ group->signal_event().connect (sigc::bind (sigc::mem_fun (tv.editor(), &PublicEditor::canvas_crossfade_view_event), group, this));
- crossfade_changed (Change (~0));
+ PropertyChange all_crossfade_properties;
+ all_crossfade_properties.add (ARDOUR::Properties::active);
+ all_crossfade_properties.add (ARDOUR::Properties::follow_overlap);
+ crossfade_changed (all_crossfade_properties);
- crossfade.StateChanged.connect (slot (*this, &CrossfadeView::crossfade_changed));
+ crossfade->PropertyChanged.connect (*this, invalidator (*this), ui_bind (&CrossfadeView::crossfade_changed, this, _1), gui_context());
+ crossfade->FadesChanged.connect (*this, invalidator (*this), ui_bind (&CrossfadeView::crossfade_fades_changed, this), gui_context());
+ ColorsChanged.connect (sigc::mem_fun (*this, &CrossfadeView::color_handler));
}
CrossfadeView::~CrossfadeView ()
{
- GoingAway (this) ; /* EMIT_SIGNAL */
-}
-
-std::string
-CrossfadeView::get_item_name ()
-{
- return "xfade";
-// return crossfade.name();
+ CatchDeletion (this) ; /* EMIT_SIGNAL */
}
void
active_changed ();
if (pixel_width < 5) {
- gtk_canvas_item_hide (fade_in);
- gtk_canvas_item_hide (fade_out);
+ fade_in->hide();
+ fade_out->hide();
}
}
void
-CrossfadeView::set_height (double height)
+CrossfadeView::set_height (double h)
{
- if (height == TimeAxisView::Smaller ||
- height == TimeAxisView::Small)
- TimeAxisViewItem::set_height (height - 3 );
- else
- TimeAxisViewItem::set_height (height - NAME_HIGHLIGHT_SIZE - 3 );
+ if (h > TimeAxisView::preset_height (HeightSmall)) {
+ h -= NAME_HIGHLIGHT_SIZE;
+ }
+
+ TimeAxisViewItem::set_height (h);
redraw_curves ();
}
void
-CrossfadeView::crossfade_changed (Change what_changed)
+CrossfadeView::crossfade_changed (const PropertyChange& what_changed)
{
bool need_redraw_curves = false;
- if (what_changed & BoundsChanged) {
- set_position (crossfade.position(), this);
- set_duration (crossfade.overlap_length(), this);
+ if (what_changed.contains (ARDOUR::bounds_change)) {
+ set_position (crossfade->position(), this);
+ set_duration (crossfade->length(), this);
+
+ /* set_duration will call reset_width_dependent_items which in turn will call redraw_curves via active_changed,
+ so no need for us to call it */
+ need_redraw_curves = false;
+ }
+
+ if (what_changed.contains (ARDOUR::Properties::follow_overlap)) {
need_redraw_curves = true;
}
-
- if (what_changed & Crossfade::ActiveChanged) {
+
+ if (what_changed.contains (ARDOUR::Properties::active)) {
/* calls redraw_curves */
active_changed ();
} else if (need_redraw_curves) {
void
CrossfadeView::redraw_curves ()
{
- GtkCanvasPoints* points;
+ Points* points;
int32_t npoints;
float* vec;
-
- double h;
-
- /*
- At "height - 3.0" the bottom of the crossfade touches the name highlight or the bottom of the track (if the
- track is either Small or Smaller.
- */
- switch(get_time_axis_view().height) {
- case TimeAxisView::Smaller:
- case TimeAxisView::Small:
- h = get_time_axis_view().height - 3.0;
- break;
-
- default:
- h = get_time_axis_view().height - NAME_HIGHLIGHT_SIZE - 3.0;
+
+ if (!crossfade->following_overlap()) {
+ /* curves should not be visible */
+ fade_in->hide ();
+ fade_out->hide ();
+ return;
}
- if (h < 0) {
+ if (_height < 0) {
/* no space allocated yet */
return;
}
- npoints = get_time_axis_view().editor.frame_to_pixel (crossfade.length());
- npoints = std::min (gdk_screen_width(), npoints);
+ npoints = get_time_axis_view().editor().frame_to_pixel (crossfade->length());
+ // npoints = std::min (gdk_screen_width(), npoints);
- if (!_visible || !crossfade.active() || npoints < 3) {
- gtk_canvas_item_hide (fade_in);
- gtk_canvas_item_hide (fade_out);
+ if (!_visible || !crossfade->active() || npoints < 3) {
+ fade_in->hide();
+ fade_out->hide();
return;
} else {
- gtk_canvas_item_show (fade_in);
- gtk_canvas_item_show (fade_out);
- }
+ fade_in->show();
+ fade_out->show();
+ }
points = get_canvas_points ("xfade edit redraw", npoints);
vec = new float[npoints];
- crossfade.fade_in().get_vector (0, crossfade.length(), vec, npoints);
+ crossfade->fade_in().curve().get_vector (0, crossfade->length(), vec, npoints);
+
for (int i = 0, pci = 0; i < npoints; ++i) {
- points->coords[pci++] = i;
- points->coords[pci++] = 2.0 + h - (h * vec[i]);
+ Art::Point &p = (*points)[pci++];
+ p.set_x (i + 1);
+ p.set_y (_height - ((_height - 2) * vec[i]));
}
- gtk_canvas_item_set (fade_in, "points", points, NULL);
+
+ fade_in->property_points() = *points;
+
+ crossfade->fade_out().curve().get_vector (0, crossfade->length(), vec, npoints);
- crossfade.fade_out().get_vector (0, crossfade.length(), vec, npoints);
for (int i = 0, pci = 0; i < npoints; ++i) {
- points->coords[pci++] = i;
- points->coords[pci++] = 2.0 + h - (h * vec[i]);
+ Art::Point &p = (*points)[pci++];
+ p.set_x (i + 1);
+ p.set_y (_height - ((_height - 2) * vec[i]));
}
- gtk_canvas_item_set (fade_out, "points", points, NULL);
+
+ fade_out->property_points() = *points;
delete [] vec;
- gtk_canvas_points_unref (points);
+ delete points;
/* XXX this is ugly, but it will have to wait till Crossfades are reimplented
as regions. This puts crossfade views on top of a track, above all regions.
*/
- gtk_canvas_item_raise_to_top (group);
+ group->raise_to_top();
}
void
CrossfadeView::active_changed ()
{
- if (crossfade.active()) {
- gtk_canvas_item_set (frame, "fill_color_rgba", color_map[cActiveCrossfade], NULL);
+ if (crossfade->active()) {
+ frame->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_ActiveCrossfade.get();
} else {
- gtk_canvas_item_set (frame, "fill_color_rgba", color_map[cInactiveCrossfade], NULL);
+ frame->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_InactiveCrossfade.get();
}
redraw_curves ();
}
+void
+CrossfadeView::color_handler ()
+{
+ active_changed ();
+}
+
void
CrossfadeView::set_valid (bool yn)
{
AudioRegionView&
CrossfadeView::upper_regionview () const
{
- if (left_view.region.layer() > right_view.region.layer()) {
+ if (left_view.region()->layer() > right_view.region()->layer()) {
return left_view;
} else {
return right_view;
void
CrossfadeView::show ()
{
- gtk_canvas_item_show (group);
+ group->show();
_visible = true;
}
void
CrossfadeView::hide ()
{
- gtk_canvas_item_hide (group);
+ group->hide();
_visible = false;
}
void
CrossfadeView::fake_hide ()
{
- gtk_canvas_item_hide (group);
+ group->hide();
+}
+
+void
+CrossfadeView::crossfade_fades_changed ()
+{
+ redraw_curves ();
}