* bugfix from http://tracker.ardour.org/view.php?id=2155 thanks to kristian: fix...
[ardour.git] / gtk2_ardour / audio_streamview.cc
index 09510e8d428c8a52bd6a65423f20288081a5c26a..08ac532319ffa6f65d9e160f9da35548b56fa9d7 100644 (file)
@@ -18,6 +18,7 @@
 
 #include <cmath>
 #include <cassert>
+#include <utility>
 
 #include <gtkmm.h>
 
 #include "rgb_macros.h"
 #include "gui_thread.h"
 #include "utils.h"
-#include "color.h"
 
 #include "i18n.h"
 
+using namespace std;
 using namespace ARDOUR;
 using namespace PBD;
 using namespace Editing;
@@ -62,17 +63,16 @@ AudioStreamView::AudioStreamView (AudioTimeAxisView& tv)
        _waveform_shape = Traditional;
        
        if (tv.is_track())
-               stream_base_color = color_map[cAudioTrackBase];
+               stream_base_color = ARDOUR_UI::config()->canvasvar_AudioTrackBase.get();
        else
-               stream_base_color = color_map[cAudioBusBase];
+               stream_base_color = ARDOUR_UI::config()->canvasvar_AudioBusBase.get();
        
        canvas_rect->property_fill_color_rgba() = stream_base_color;
-       canvas_rect->property_outline_color_rgba() = color_map[cAudioTrackOutline];
+       canvas_rect->property_outline_color_rgba() = RGBA_BLACK;
 
        _amplitude_above_axis = 1.0;
 
        use_rec_regions = tv.editor.show_waveforms_recording ();
-
 }
 
 AudioStreamView::~AudioStreamView ()
@@ -111,17 +111,15 @@ AudioStreamView::set_amplitude_above_axis (gdouble app)
        return 0;
 }
 
-void
+RegionView*
 AudioStreamView::add_region_view_internal (boost::shared_ptr<Region> r, bool wait_for_waves)
 {
        AudioRegionView *region_view = 0;
 
-       ENSURE_GUI_THREAD (bind (mem_fun (*this, &AudioStreamView::add_region_view), r));
-
        boost::shared_ptr<AudioRegion> region = boost::dynamic_pointer_cast<AudioRegion> (r);
 
        if (region == 0) {
-               return;
+               return NULL;
        }
 
        for (list<RegionView *>::iterator i = region_views.begin(); i != region_views.end(); ++i) {
@@ -138,7 +136,7 @@ AudioStreamView::add_region_view_internal (boost::shared_ptr<Region> r, bool wai
                                arv->set_waveform_shape (_waveform_shape);
                        }
                                
-                       return;
+                       return NULL;
                }
        }
 
@@ -199,6 +197,8 @@ AudioStreamView::add_region_view_internal (boost::shared_ptr<Region> r, bool wai
        region->GoingAway.connect (bind (mem_fun (*this, &AudioStreamView::remove_region_view), boost::weak_ptr<Region> (r)));
 
        RegionViewAdded (region_view);
+
+       return region_view;
 }
 
 void
@@ -309,6 +309,7 @@ AudioStreamView::add_crossfade (boost::shared_ptr<Crossfade> crossfade)
 
        for (list<CrossfadeView *>::iterator i = crossfade_views.begin(); i != crossfade_views.end(); ++i) {
                if ((*i)->crossfade == crossfade) {
+
                        if (!crossfades_visible || layer_display == Stacked) {
                                (*i)->hide();
                        } else {
@@ -338,10 +339,9 @@ AudioStreamView::add_crossfade (boost::shared_ptr<Crossfade> crossfade)
                                               _samples_per_unit,
                                               region_color,
                                               *lview, *rview);
-
+       cv->set_valid (true);
        crossfade->Invalidated.connect (mem_fun (*this, &AudioStreamView::remove_crossfade));
        crossfade_views.push_back (cv);
-
        if (!Config->get_xfades_visible() || !crossfades_visible || layer_display == Stacked) {
                cv->hide ();
        }
@@ -388,6 +388,8 @@ AudioStreamView::redisplay_diskstream ()
                        apl->foreach_crossfade (this, &AudioStreamView::add_crossfade);
        }
 
+       RegionViewList  copy;
+
        for (i = region_views.begin(); i != region_views.end(); ) {
                tmp = i;
                tmp++;
@@ -395,9 +397,47 @@ AudioStreamView::redisplay_diskstream ()
                if (!(*i)->is_valid()) {
                        delete *i;
                        region_views.erase (i);
-               } 
+                       i = tmp;
+                       continue;
+               } else {
+                       (*i)->enable_display(true);
+               }
+
+               /* 
+                  sort regionviews by layer so that when we call region_layered ()
+                  the canvas layering works out (in non-stacked mode).
+               */
+
+               if (copy.size() == 0) {
+                       copy.push_front((*i));
+                       i = tmp;
+                       continue;
+               }
+
+               RegionViewList::iterator k = copy.begin();
+               RegionViewList::iterator l = copy.end();
+               l--;
+
+               if ((*i)->region()->layer() <= (*k)->region()->layer()) {
+                       copy.push_front((*i));
+                       i = tmp;
+                       continue;
+               } else if ((*i)->region()->layer() >= (*l)->region()->layer()) {
+                       copy.push_back((*i));
+                       i = tmp;
+                       continue;
+               }
+
+               for (RegionViewList::iterator j = copy.begin(); j != copy.end(); ++j) {
+                 
+                       if ((*j)->region()->layer() >= (*i)->region()->layer()) {
+                               copy.insert(j, (*i));
+                               break;
+                       }
+               }
 
                i = tmp;
+
        }
 
        for (xi = crossfade_views.begin(); xi != crossfade_views.end();) {
@@ -412,10 +452,12 @@ AudioStreamView::redisplay_diskstream ()
                xi = tmpx;
        }
 
-       /* now fix layering */
-
-       for (RegionViewList::iterator i = region_views.begin(); i != region_views.end(); ++i) {
-               region_layered (*i);
+       /* now fix canvas layering */
+       
+       for (RegionViewList::iterator j = copy.begin(); j != copy.end(); ++j) {
+                       (*j)->enable_display(true);
+                       (*j)->set_y_position_and_height(0, height);
+                       region_layered (*j);
        }
 }
 
@@ -424,8 +466,9 @@ AudioStreamView::set_show_waveforms (bool yn)
 {
        for (list<RegionView *>::iterator i = region_views.begin(); i != region_views.end(); ++i) {
                AudioRegionView* const arv = dynamic_cast<AudioRegionView*>(*i);
-               if (arv)
+               if (arv) {
                        arv->set_waveform_visible (yn);
+               }
        }
 }
 
@@ -493,7 +536,7 @@ AudioStreamView::setup_rec_box ()
                                
                                nframes_t start = 0;
                                if (rec_regions.size() > 0) {
-                                       start = rec_regions.back()->start() + _trackview.get_diskstream()->get_captured_frames(rec_regions.size()-1);
+                                       start = rec_regions.back().first->start() + _trackview.get_diskstream()->get_captured_frames(rec_regions.size()-1);
                                }
                                
                                boost::shared_ptr<AudioRegion> region (boost::dynamic_pointer_cast<AudioRegion>
@@ -501,12 +544,12 @@ AudioStreamView::setup_rec_box ()
                                assert(region);
                                region->set_position (_trackview.session().transport_frame(), this);
 
-                               rec_regions.push_back (region);
+                               rec_regions.push_back (make_pair(region, (RegionView*)0));
                        }
                        
                        /* start a new rec box */
 
-                       AudioTrack* at;
+                       boost::shared_ptr<AudioTrack> at;
 
                        at = _trackview.audio_track(); /* we know what it is already */
                        boost::shared_ptr<AudioDiskstream> ds = at->audio_diskstream();
@@ -518,12 +561,12 @@ AudioStreamView::setup_rec_box ()
                        switch (_trackview.audio_track()->mode()) {
                        case Normal:
                                xend = xstart;
-                               fill_color = color_map[cRecordingRectFill];
+                               fill_color = ARDOUR_UI::config()->canvasvar_RecordingRect.get();
                                break;
 
                        case Destructive:
                                xend = xstart + 2;
-                               fill_color = color_map[cRecordingRectFill];
+                               fill_color = ARDOUR_UI::config()->canvasvar_RecordingRect.get();
                                /* make the recording rect translucent to allow
                                   the user to see the peak data coming in, etc.
                                */
@@ -536,8 +579,9 @@ AudioStreamView::setup_rec_box ()
                        rec_rect->property_y1() = 1.0;
                        rec_rect->property_x2() = xend;
                        rec_rect->property_y2() = (double) _trackview.height - 1;
-                       rec_rect->property_outline_color_rgba() = color_map[cRecordingRectOutline];
+                       rec_rect->property_outline_color_rgba() = ARDOUR_UI::config()->canvasvar_RecordingRect.get();
                        rec_rect->property_fill_color_rgba() = fill_color;
+                       rec_rect->lower_to_bottom();
                        
                        RecBoxInfo recbox;
                        recbox.rectangle = rec_rect;
@@ -577,17 +621,16 @@ AudioStreamView::setup_rec_box ()
 
                        rec_updating = false;
                        rec_active = false;
-                       last_rec_data_frame = 0;
                        
                        /* remove temp regions */
 
-                       for (list<boost::shared_ptr<Region> >::iterator iter = rec_regions.begin(); iter != rec_regions.end(); ) {
-                               list<boost::shared_ptr<Region> >::iterator tmp;
+                       for (list<pair<boost::shared_ptr<Region>,RegionView*> >::iterator iter = rec_regions.begin(); iter != rec_regions.end(); ) {
+                               list<pair<boost::shared_ptr<Region>,RegionView*> >::iterator tmp;
 
                                tmp = iter;
                                ++tmp;
 
-                               (*iter)->drop_references ();
+                               (*iter).first->drop_references ();
 
                                iter = tmp;
                        }
@@ -648,9 +691,9 @@ AudioStreamView::update_rec_regions ()
 
                uint32_t n = 0;
 
-               for (list<boost::shared_ptr<Region> >::iterator iter = rec_regions.begin(); iter != rec_regions.end(); n++) {
+               for (list<pair<boost::shared_ptr<Region>,RegionView*> >::iterator iter = rec_regions.begin(); iter != rec_regions.end(); n++) {
 
-                       list<boost::shared_ptr<Region> >::iterator tmp;
+                       list<pair<boost::shared_ptr<Region>,RegionView*> >::iterator tmp;
 
                        tmp = iter;
                        ++tmp;
@@ -661,14 +704,14 @@ AudioStreamView::update_rec_regions ()
                                continue;
                        }
                        
-                       boost::shared_ptr<AudioRegion> region = boost::dynamic_pointer_cast<AudioRegion>(*iter);
+                       boost::shared_ptr<AudioRegion> region = boost::dynamic_pointer_cast<AudioRegion>(iter->first);
                        if (!region) {
                                continue;
                        }
 
                        nframes_t origlen = region->length();
 
-                       if (region == rec_regions.back() && rec_active) {
+                       if (region == rec_regions.back().first && rec_active) {
 
                                if (last_rec_data_frame > region->start()) {
 
@@ -759,26 +802,18 @@ AudioStreamView::reveal_xfades_involving (AudioRegionView& rv)
 }
 
 void
-AudioStreamView::color_handler (ColorID id, uint32_t val)
+AudioStreamView::color_handler ()
 {
-       switch (id) {
-       case cAudioTrackBase:
-               if (_trackview.is_track()) {
-                       canvas_rect->property_fill_color_rgba() = val;
-               } 
-               break;
-       case cAudioBusBase:
-               if (!_trackview.is_track()) {
-                       canvas_rect->property_fill_color_rgba() = val;
-               }
-               break;
-       case cAudioTrackOutline:
-               canvas_rect->property_outline_color_rgba() = val;
-               break;
+       //case cAudioTrackBase:
+       if (_trackview.is_track()) {
+               canvas_rect->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_AudioTrackBase.get();
+       } 
 
-       default:
-               break;
+       //case cAudioBusBase:
+       if (!_trackview.is_track()) {
+               canvas_rect->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_AudioBusBase.get();
        }
+
 }
 
 void