Make waveform show / scale / shape a global option in the prefs dialog to clean thing...
[ardour.git] / gtk2_ardour / audio_streamview.cc
index 398b9c404ba522e2e223b4b3c3203914295fdca3..7dbae624e1f76eccf10352fec66e83d16bd5f23d 100644 (file)
 
 #include <cmath>
 #include <cassert>
+#include <utility>
 
 #include <gtkmm.h>
 
 #include <gtkmm2ext/gtk_ui.h>
 
-#include <ardour/audioplaylist.h>
-#include <ardour/audioregion.h>
-#include <ardour/audiofilesource.h>
-#include <ardour/audio_diskstream.h>
-#include <ardour/audio_track.h>
-#include <ardour/playlist_templates.h>
-#include <ardour/source.h>
-#include <ardour/region_factory.h>
+#include "ardour/audioplaylist.h"
+#include "ardour/audioregion.h"
+#include "ardour/audiofilesource.h"
+#include "ardour/audio_diskstream.h"
+#include "ardour/audio_track.h"
+#include "ardour/source.h"
+#include "ardour/region_factory.h"
+#include "ardour/profile.h"
 
 #include "audio_streamview.h"
 #include "audio_region_view.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;
@@ -58,44 +59,18 @@ AudioStreamView::AudioStreamView (AudioTimeAxisView& tv)
        : StreamView (tv)
 {
        crossfades_visible = true;
-       _waveform_scale = LinearWaveform;
-       _waveform_shape = Traditional;
-       
-       if (tv.is_track())
-               stream_base_color = color_map[cAudioTrackBase];
-       else
-               stream_base_color = color_map[cAudioBusBase];
-       
-       canvas_rect->property_fill_color_rgba() = stream_base_color;
-       canvas_rect->property_outline_color_rgba() = color_map[cAudioTrackOutline];
-
+       color_handler ();
        _amplitude_above_axis = 1.0;
 
-       use_rec_regions = tv.editor.show_waveforms_recording ();
+       use_rec_regions = tv.editor().show_waveforms_recording ();
 
+       Config->ParameterChanged.connect (sigc::mem_fun (*this, &AudioStreamView::parameter_changed));
 }
 
 AudioStreamView::~AudioStreamView ()
 {
 }
 
-int
-AudioStreamView::set_height (gdouble h)
-{
-       /* limit the values to something sane-ish */
-       if (h < 10.0 || h > 1000.0) {
-               return -1;
-       }
-
-       StreamView::set_height(h);
-
-       for (CrossfadeViewList::iterator i = crossfade_views.begin(); i != crossfade_views.end(); ++i) {
-               (*i)->set_height (h);
-       }
-
-       return 0;
-}
-
 int 
 AudioStreamView::set_samples_per_unit (gdouble spp)
 {
@@ -128,45 +103,32 @@ AudioStreamView::set_amplitude_above_axis (gdouble app)
        return 0;
 }
 
-void
-AudioStreamView::add_region_view_internal (boost::shared_ptr<Region> r, bool wait_for_waves)
+RegionView*
+AudioStreamView::create_region_view (boost::shared_ptr<Region> r, bool wait_for_waves, bool recording)
 {
        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;
-       }
-
-       for (list<RegionView *>::iterator i = region_views.begin(); i != region_views.end(); ++i) {
-               if ((*i)->region() == r) {
-                       
-                       /* great. we already have a AudioRegionView for this Region. use it again. */
-                       
-                       (*i)->set_valid (true);
-
-                       // this might not be necessary
-                       AudioRegionView* const arv = dynamic_cast<AudioRegionView*>(*i);
-                       if (arv) {
-                               arv->set_waveform_scale (_waveform_scale);
-                               arv->set_waveform_shape (_waveform_shape);
-                       }
-                               
-                       return;
-               }
+               return 0;
        }
 
        switch (_trackview.audio_track()->mode()) {
+       
+       case NonLayered:
        case Normal:
-               region_view = new AudioRegionView (canvas_group, _trackview, region, 
-                                                  _samples_per_unit, region_color);
+               if (recording) {
+                       region_view = new AudioRegionView (canvas_group, _trackview, region, 
+                                       _samples_per_unit, region_color, recording, TimeAxisViewItem::Visibility(
+                                                       TimeAxisViewItem::ShowFrame | TimeAxisViewItem::HideFrameRight));
+               } else {
+                       region_view = new AudioRegionView (canvas_group, _trackview, region, 
+                                       _samples_per_unit, region_color);
+               }
                break;
        case Destructive:
                region_view = new TapeAudioRegionView (canvas_group, _trackview, region, 
-                                                      _samples_per_unit, region_color);
+                               _samples_per_unit, region_color);
                break;
        default:
                fatal << string_compose (_("programming error: %1"), "illegal track mode in ::add_region_view_internal") << endmsg;
@@ -176,9 +138,8 @@ AudioStreamView::add_region_view_internal (boost::shared_ptr<Region> r, bool wai
 
        region_view->init (region_color, wait_for_waves);
        region_view->set_amplitude_above_axis(_amplitude_above_axis);
-       region_views.push_front (region_view);
+       region_view->set_height (child_height ());
 
-       
        /* if its the special single-sample length that we use for rec-regions, make it 
           insensitive to events 
        */
@@ -187,35 +148,50 @@ AudioStreamView::add_region_view_internal (boost::shared_ptr<Region> r, bool wai
                region_view->set_sensitive (false);
        }
 
-       /* if this was the first one, then lets query the waveform scale and shape.
-          otherwise, we set it to the current value */
-          
-       if (region_views.size() == 1) {
-               if (region_view->waveform_logscaled()) {
-                       _waveform_scale = LogWaveform;
-               } else {
-                       _waveform_scale = LinearWaveform;
-               }
+       region_view->set_waveform_scale (Config->get_waveform_scale ());
+       region_view->set_waveform_shape (Config->get_waveform_shape ());
+       region_view->set_waveform_visible (Config->get_show_waveforms ());
 
-               if (region_view->waveform_rectified()) {
-                       _waveform_shape = Rectified;
-               } else {
-                       _waveform_shape = Traditional;
-               }
-       }
-       else {
-               region_view->set_waveform_scale(_waveform_scale);
-               region_view->set_waveform_shape(_waveform_shape);
+       return region_view;     
+}
+
+RegionView*
+AudioStreamView::add_region_view_internal (boost::shared_ptr<Region> r, bool wait_for_waves, bool recording)
+{
+       RegionView *region_view = create_region_view (r, wait_for_waves, recording);
+       if (region_view == 0) {
+               return 0;
        }
-       
-       /* follow global waveform setting */
 
-       region_view->set_waveform_visible(_trackview.editor.show_waveforms());
+//     if(!recording){
+//             for (list<RegionView *>::iterator i = region_views.begin(); i != region_views.end(); ++i) {
+//                     if ((*i)->region() == r) {
+//                             cerr << "audio_streamview in add_region_view_internal region found" << endl;
+                               /* great. we already have a AudioRegionView for this Region. use it again. */
+                               
+//                             (*i)->set_valid (true);
+
+                               // this might not be necessary
+//                             AudioRegionView* const arv = dynamic_cast<AudioRegionView*>(*i);
+
+//                             if (arv) {
+//                                     arv->set_waveform_scale (_waveform_scale);
+//                                     arv->set_waveform_shape (_waveform_shape);
+//                             }
+                                       
+//                             return NULL;
+//                     }
+//             }
+//     }
+
+       region_views.push_front (region_view);
 
        /* catch regionview going away */
-       region->GoingAway.connect (bind (mem_fun (*this, &AudioStreamView::remove_region_view), boost::weak_ptr<Region> (r)));
+       r->GoingAway.connect (bind (mem_fun (*this, &AudioStreamView::remove_region_view), boost::weak_ptr<Region> (r)));
 
        RegionViewAdded (region_view);
+
+       return region_view;
 }
 
 void
@@ -263,11 +239,21 @@ AudioStreamView::undisplay_diskstream ()
 }
 
 void
-AudioStreamView::playlist_modified ()
+AudioStreamView::playlist_modified_weak (boost::weak_ptr<Diskstream> ds)
 {
-       ENSURE_GUI_THREAD (mem_fun (*this, &AudioStreamView::playlist_modified));
+       boost::shared_ptr<Diskstream> sp (ds.lock());
+       if (sp) {
+               playlist_modified (sp);
+       }
+}
 
-       StreamView::playlist_modified();
+void
+AudioStreamView::playlist_modified (boost::shared_ptr<Diskstream> ds)
+{
+       /* we do not allow shared_ptr<T> to be bound to slots */
+       ENSURE_GUI_THREAD (bind (mem_fun (*this, &AudioStreamView::playlist_modified_weak), ds));
+
+       StreamView::playlist_modified (ds);
        
        /* make sure xfades are on top and all the regionviews are stacked correctly. */
 
@@ -276,16 +262,29 @@ AudioStreamView::playlist_modified ()
        }
 }
 
+void
+AudioStreamView::playlist_changed_weak (boost::weak_ptr<Diskstream> ds)
+{
+       boost::shared_ptr<Diskstream> sp (ds.lock());
+       if (sp) {
+               playlist_changed (sp);
+       }
+}
+
 void
 AudioStreamView::playlist_changed (boost::shared_ptr<Diskstream> ds)
 {
-       ENSURE_GUI_THREAD (bind (mem_fun (*this, &AudioStreamView::playlist_changed), ds));
+       ENSURE_GUI_THREAD (bind (
+                       mem_fun (*this, &AudioStreamView::playlist_changed_weak),
+                       boost::weak_ptr<Diskstream> (ds)));
 
        StreamView::playlist_changed(ds);
 
        boost::shared_ptr<AudioPlaylist> apl = boost::dynamic_pointer_cast<AudioPlaylist>(ds->playlist());
-       if (apl)
-               playlist_connections.push_back (apl->NewCrossfade.connect (mem_fun (*this, &AudioStreamView::add_crossfade)));
+       if (apl) {
+               playlist_connections.push_back (apl->NewCrossfade.connect (
+                               mem_fun (*this, &AudioStreamView::add_crossfade)));
+       }
 }
 
 void
@@ -314,7 +313,8 @@ 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) {
+
+                       if (!crossfades_visible || _layer_display == Stacked) {
                                (*i)->hide();
                        } else {
                                (*i)->show ();
@@ -337,25 +337,26 @@ AudioStreamView::add_crossfade (boost::shared_ptr<Crossfade> crossfade)
                }
        }
 
-       CrossfadeView *cv = new CrossfadeView (_trackview.canvas_display,
+       CrossfadeView *cv = new CrossfadeView (_trackview.canvas_display (),
                                               _trackview,
                                                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) {
+       if (!_trackview.session().config.get_xfades_visible() || !crossfades_visible || _layer_display == Stacked) {
                cv->hide ();
        }
 }
 
 void
-AudioStreamView::remove_crossfade (boost::shared_ptr<Crossfade> xfade)
+AudioStreamView::remove_crossfade (boost::shared_ptr<Region> r)
 {
-       ENSURE_GUI_THREAD (bind (mem_fun (*this, &AudioStreamView::remove_crossfade), xfade));
+       ENSURE_GUI_THREAD (bind (mem_fun (*this, &AudioStreamView::remove_crossfade), r));
+
+       boost::shared_ptr<Crossfade> xfade = boost::dynamic_pointer_cast<Crossfade> (r);
 
        for (list<CrossfadeView*>::iterator i = crossfade_views.begin(); i != crossfade_views.end(); ++i) {
                if ((*i)->crossfade == xfade) {
@@ -369,40 +370,40 @@ AudioStreamView::remove_crossfade (boost::shared_ptr<Crossfade> xfade)
 void
 AudioStreamView::redisplay_diskstream ()
 {
-       list<RegionView *>::iterator i, tmp;
+       list<RegionView *>::iterator i;
        list<CrossfadeView*>::iterator xi, tmpx;
 
+       // Flag region views as invalid and disable drawing
        for (i = region_views.begin(); i != region_views.end(); ++i) {
                (*i)->set_valid (false);
+               (*i)->enable_display (false);
        }
 
+       // Flag crossfade views as invalid
        for (xi = crossfade_views.begin(); xi != crossfade_views.end(); ++xi) {
                (*xi)->set_valid (false);
-               if ((*xi)->visible()) {
+               if ((*xi)->visible() && _layer_display != Stacked) {
                        (*xi)->show ();
                }
        }
 
-       if (_trackview.is_audio_track()) {
-               _trackview.get_diskstream()->playlist()->foreach_region (static_cast<StreamView*>(this), &StreamView::add_region_view);
+       // Add and display region and crossfade views, and flag them as valid
 
-               boost::shared_ptr<AudioPlaylist> apl = boost::dynamic_pointer_cast<AudioPlaylist>(_trackview.get_diskstream()->playlist());
-               if (apl)
-                       apl->foreach_crossfade (this, &AudioStreamView::add_crossfade);
-       }
-
-       for (i = region_views.begin(); i != region_views.end(); ) {
-               tmp = i;
-               tmp++;
-
-               if (!(*i)->is_valid()) {
-                       delete *i;
-                       region_views.erase (i);
-               } 
+       if (_trackview.is_audio_track()) {
+               _trackview.get_diskstream()->playlist()->foreach_region(
+                       sigc::hide_return (sigc::mem_fun (*this, &StreamView::add_region_view))
+                       );
 
-               i = tmp;
+               boost::shared_ptr<AudioPlaylist> apl = boost::dynamic_pointer_cast<AudioPlaylist>(
+                               _trackview.get_diskstream()->playlist()
+                       );
+               
+               if (apl) {
+                       apl->foreach_crossfade (sigc::mem_fun (*this, &AudioStreamView::add_crossfade));
+               }
        }
-
+       
+       // Remove invalid crossfade views
        for (xi = crossfade_views.begin(); xi != crossfade_views.end();) {
                tmpx = xi;
                tmpx++;
@@ -415,11 +416,8 @@ AudioStreamView::redisplay_diskstream ()
                xi = tmpx;
        }
 
-       /* now fix layering */
-
-       for (RegionViewList::iterator i = region_views.begin(); i != region_views.end(); ++i) {
-               region_layered (*i);
-       }
+       // Stack regions by layer, and remove invalid regions
+       layer_regions();
 }
 
 void
@@ -427,8 +425,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);
+               }
        }
 }
 
@@ -440,7 +439,6 @@ AudioStreamView::set_waveform_shape (WaveformShape shape)
                if (arv)
                        arv->set_waveform_shape (shape);
        }
-       _waveform_shape = shape;
 }              
 
 void
@@ -448,16 +446,16 @@ AudioStreamView::set_waveform_scale (WaveformScale scale)
 {
        for (RegionViewList::iterator i = region_views.begin(); i != region_views.end(); ++i) {
                AudioRegionView* const arv = dynamic_cast<AudioRegionView*>(*i);
-               if (arv) 
+               if (arv) {
                        arv->set_waveform_scale (scale);
+               }
        }
-       _waveform_scale = scale;
 }              
 
 void
 AudioStreamView::setup_rec_box ()
 {
-       // cerr << _trackview.name() << " streamview SRB\n";
+       //cerr << _trackview.name() << " streamview SRB region_views.size() = " << region_views.size() << endl;
 
        if (_trackview.session().transport_rolling()) {
 
@@ -466,7 +464,6 @@ AudioStreamView::setup_rec_box ()
                if (!rec_active && 
                    _trackview.session().record_status() == Session::Recording && 
                    _trackview.get_diskstream()->record_enabled()) {
-
                        if (_trackview.audio_track()->mode() == Normal && use_rec_regions && rec_regions.size() == rec_rects.size()) {
 
                                /* add a new region, but don't bother if they set use_rec_regions mid-record */
@@ -482,7 +479,7 @@ AudioStreamView::setup_rec_box ()
                                boost::shared_ptr<AudioDiskstream> ads = boost::dynamic_pointer_cast<AudioDiskstream>(_trackview.get_diskstream());
                                assert(ads);
 
-                               for (uint32_t n=0; n < ads->n_channels().get(DataType::AUDIO); ++n) {
+                               for (uint32_t n=0; n < ads->n_channels().n_audio(); ++n) {
                                        boost::shared_ptr<AudioFileSource> src = boost::static_pointer_cast<AudioFileSource> (ads->write_source (n));
                                        if (src) {
                                                sources.push_back (src);
@@ -496,37 +493,37 @@ 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>
                                                                       (RegionFactory::create (sources, start, 1 , "", 0, (Region::Flag)(Region::DefaultFlags | Region::DoNotSaveState), false)));
                                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();
                        nframes_t frame_pos = ds->current_capture_start ();
-                       gdouble xstart = _trackview.editor.frame_to_pixel (frame_pos);
+                       gdouble xstart = _trackview.editor().frame_to_pixel (frame_pos);
                        gdouble xend;
                        uint32_t fill_color;
 
                        switch (_trackview.audio_track()->mode()) {
                        case Normal:
+                       case NonLayered:
                                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.
                                */
@@ -538,9 +535,11 @@ AudioStreamView::setup_rec_box ()
                        rec_rect->property_x1() = xstart;
                        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_y2() = child_height ();
+                       rec_rect->property_outline_color_rgba() = ARDOUR_UI::config()->canvasvar_TimeAxisFrame.get();
+                       rec_rect->property_outline_what() = 0x1 | 0x2 | 0x4 | 0x8;
                        rec_rect->property_fill_color_rgba() = fill_color;
+                       rec_rect->lower_to_bottom();
                        
                        RecBoxInfo recbox;
                        recbox.rectangle = rec_rect;
@@ -557,11 +556,9 @@ AudioStreamView::setup_rec_box ()
                } else if (rec_active &&
                           (_trackview.session().record_status() != Session::Recording ||
                            !_trackview.get_diskstream()->record_enabled())) {
-
                        screen_update_connection.disconnect();
                        rec_active = false;
                        rec_updating = false;
-
                }
                
        } else {
@@ -580,17 +577,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;
                        }
@@ -638,7 +634,7 @@ AudioStreamView::rec_peak_range_ready (nframes_t start, nframes_t cnt, boost::we
        
        rec_data_ready_map[src] = true;
        
-       if (rec_data_ready_map.size() == _trackview.get_diskstream()->n_channels().get(DataType::AUDIO)) {
+       if (rec_data_ready_map.size() == _trackview.get_diskstream()->n_channels().n_audio()) {
                this->update_rec_regions ();
                rec_data_ready_map.clear();
        }
@@ -648,12 +644,11 @@ void
 AudioStreamView::update_rec_regions ()
 {
        if (use_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;
@@ -664,14 +659,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()) {
 
@@ -686,12 +681,12 @@ AudioStreamView::update_rec_regions ()
 
                                                if (origlen == 1) {
                                                        /* our special initial length */
-                                                       add_region_view_internal (region, false);
+                                                       add_region_view_internal (region, false, true);
                                                }
 
                                                /* also update rect */
                                                ArdourCanvas::SimpleRect * rect = rec_rects[n].rectangle;
-                                               gdouble xend = _trackview.editor.frame_to_pixel (region->position() + region->length());
+                                               gdouble xend = _trackview.editor().frame_to_pixel (region->position() + region->length());
                                                rect->property_x2() = xend;
                                        }
                                }
@@ -702,7 +697,7 @@ AudioStreamView::update_rec_regions ()
 
                                if (nlen != region->length()) {
 
-                                       if (region->source(0)->length() >= region->start() + nlen) {
+                                       if (region->source_length(0) >= region->start() + nlen) {
 
                                                region->freeze ();
                                                region->set_position (_trackview.get_diskstream()->get_capture_start_frame(n), this);
@@ -711,7 +706,7 @@ AudioStreamView::update_rec_regions ()
                                                
                                                if (origlen == 1) {
                                                        /* our special initial length */
-                                                       add_region_view_internal (region, false);
+                                                       add_region_view_internal (region, false, true);
                                                }
                                                
                                                /* also hide rect */
@@ -727,6 +722,28 @@ AudioStreamView::update_rec_regions ()
        }
 }
 
+void
+AudioStreamView::show_all_fades ()
+{
+       for (list<RegionView*>::iterator i = region_views.begin(); i != region_views.end(); ++i) {
+               AudioRegionView* const arv = dynamic_cast<AudioRegionView*>(*i);
+               if (arv) {
+                       arv->set_fade_visibility (true);
+               }
+       }
+}
+
+void
+AudioStreamView::hide_all_fades ()
+{
+       for (list<RegionView*>::iterator i = region_views.begin(); i != region_views.end(); ++i) {
+               AudioRegionView* const arv = dynamic_cast<AudioRegionView*>(*i);
+               if (arv) {
+                       arv->set_fade_visibility (false);
+               }
+       }
+}
+
 void
 AudioStreamView::show_all_xfades ()
 {
@@ -755,32 +772,53 @@ void
 AudioStreamView::reveal_xfades_involving (AudioRegionView& rv)
 {
        for (list<CrossfadeView *>::iterator i = crossfade_views.begin(); i != crossfade_views.end(); ++i) {
-               if ((*i)->crossfade->involves (rv.audio_region()) && (*i)->visible()) {
+               if ((*i)->crossfade->involves (rv.audio_region()) && (*i)->visible() && _layer_display != Stacked) {
                        (*i)->show ();
                }
        }
 }
 
 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;
+       //case cAudioTrackBase:
+       if (_trackview.is_track()) {
+               canvas_rect->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_AudioTrackBase.get();
+       } 
+
+       //case cAudioBusBase:
+       if (!_trackview.is_track()) {
+               if (Profile->get_sae() && _trackview.route()->is_master()) {
+                       canvas_rect->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_AudioMasterBusBase.get();
+               } else {
+                       canvas_rect->property_fill_color_rgba() = ARDOUR_UI::config()->canvasvar_AudioBusBase.get();
                }
-               break;
-       case cAudioTrackOutline:
-               canvas_rect->property_outline_color_rgba() = val;
-               break;
+       }
+}
 
-       default:
-               break;
+void
+AudioStreamView::update_contents_height ()
+{
+       StreamView::update_contents_height ();
+       
+       for (CrossfadeViewList::iterator i = crossfade_views.begin(); i != crossfade_views.end(); ++i) {
+               if (_layer_display == Overlaid) {
+                       (*i)->show ();
+                       (*i)->set_height (height);
+               } else {
+                       (*i)->hide ();
+               }
        }
 }
 
+void
+AudioStreamView::parameter_changed (string const & p)
+{
+       if (p == "show-waveforms") {
+               set_show_waveforms (Config->get_show_waveforms ());
+       } else if (p == "waveform-scale") {
+               set_waveform_scale (Config->get_waveform_scale ());
+       } else if (p == "waveform-shape") {
+               set_waveform_shape (Config->get_waveform_shape ());
+       }
+}