changes to waveform clipping display
[ardour.git] / libs / canvas / canvas / wave_view.h
1 /*
2     Copyright (C) 2011-2013 Paul Davis
3     Author: Carl Hetherington <cth@carlh.net>
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18
19 */
20
21 #include <boost/shared_ptr.hpp>
22 #include <boost/shared_array.hpp>
23 #include <boost/scoped_array.hpp>
24
25 #include "pbd/properties.h"
26
27 #include "ardour/types.h"
28
29 #include <glibmm/refptr.h>
30
31 #include "canvas/item.h"
32 #include "canvas/fill.h"
33 #include "canvas/outline.h"
34
35 namespace ARDOUR {
36         class AudioRegion;
37 }
38
39 namespace Gdk {
40         class Pixbuf;
41 }
42
43 class WaveViewTest;
44         
45 namespace ArdourCanvas {
46
47 class WaveView : virtual public Item, public Outline, public Fill
48 {
49 public:
50         enum Shape { 
51                 Normal,
52                 Rectified,
53         };
54
55     /* Displays a single channel of waveform data for the given Region.
56
57        x = 0 in the waveview corresponds to the first waveform datum taken
58        from region->start() samples into the source data.
59
60        x = N in the waveview corresponds to the (N * spp)'th sample 
61        measured from region->start() into the source data.
62
63        when drawing, we will map the zeroth-pixel of the waveview
64        into a window. 
65
66        The waveview itself contains a set of pre-rendered Cairo::ImageSurfaces
67        that cache sections of the display. This is filled on-demand and
68        never cleared until something explicitly marks the cache invalid
69        (such as a change in samples_per_pixel, the log scaling, rectified or
70        other view parameters).
71     */
72
73
74         WaveView (Group *, boost::shared_ptr<ARDOUR::AudioRegion>);
75        ~WaveView ();
76
77         void render (Rect const & area, Cairo::RefPtr<Cairo::Context>) const;
78         void compute_bounding_box () const;
79     
80         void set_samples_per_pixel (double);
81         void set_height (Distance);
82         void set_channel (int);
83         void set_region_start (ARDOUR::frameoffset_t);
84
85         void set_fill_color (Color);
86         void set_outline_color (Color);
87         
88         void region_resized ();
89         void gain_changed ();
90
91         void set_show_zero_line (bool);
92         bool show_zero_line() const { return _show_zero; }
93         void set_zero_color (Color);
94         void set_clip_color (Color);
95         void set_logscaled (bool);
96         void set_gradient_depth (double);
97         double gradient_depth() const { return _gradient_depth; }
98         void set_shape (Shape);
99
100         
101         /* currently missing because we don't need them (yet):
102            set_shape_independent();
103            set_logscaled_independent()
104         */
105
106         static void set_global_gradient_depth (double);
107         static void set_global_logscaled (bool);
108         static void set_global_shape (Shape);
109         static void set_global_show_waveform_clipping (bool);
110     
111         static double  global_gradient_depth()  { return _global_gradient_depth; }
112         static bool    global_logscaled()  { return _global_logscaled; }
113         static Shape   global_shape()  { return _global_shape; }
114
115         void set_amplitude_above_axis (double v);
116         double amplitude_above_axis () const { return _amplitude_above_axis; }
117
118 #ifdef CANVAS_COMPATIBILITY     
119         void*& property_gain_src () {
120                 return _foo_void;
121         }
122         void*& property_gain_function () {
123                 return _foo_void;
124         }
125 private:
126         void* _foo_void;
127
128 #endif
129
130         friend class ::WaveViewTest;
131
132         void invalidate_image ();
133
134         boost::shared_ptr<ARDOUR::AudioRegion> _region;
135         int    _channel;
136         double _samples_per_pixel;
137         Coord  _height;
138         Color  _wave_color;
139         bool   _show_zero;
140         Color  _zero_color;
141         Color  _clip_color;
142         bool   _logscaled;
143         Shape  _shape;
144         double _gradient_depth;
145         bool   _shape_independent;
146         bool   _logscaled_independent;
147         bool   _gradient_depth_independent;
148         double _amplitude_above_axis;
149
150         /** The `start' value to use for the region; we can't use the region's
151          *  value as the crossfade editor needs to alter it.
152          */
153         ARDOUR::frameoffset_t _region_start;
154     
155     
156         mutable ARDOUR::framepos_t _sample_start; 
157         mutable ARDOUR::framepos_t _sample_end; 
158         mutable Cairo::RefPtr<Cairo::ImageSurface> _image;
159
160         PBD::ScopedConnection invalidation_connection;
161
162         static double _global_gradient_depth;
163         static bool   _global_logscaled;
164         static Shape  _global_shape;
165         static bool   _global_show_waveform_clipping;
166
167         static PBD::Signal0<void> VisualPropertiesChanged;
168
169         void handle_visual_property_change ();
170
171         void ensure_cache (ARDOUR::framepos_t sample_start, ARDOUR::framepos_t sample_end) const;
172         ArdourCanvas::Coord position (double) const;
173         void draw_image (ARDOUR::PeakData*, int npeaks) const;
174 };
175
176 }