minor peak-meter performance tweak
[ardour.git] / libs / gtkmm2ext / gtkmm2ext / fastmeter.h
1 /*
2     Copyright (C) 2003 Paul Davis
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #ifndef __gtkmm2ext_fastmeter_h__
21 #define __gtkmm2ext_fastmeter_h__
22
23 #include <map>
24 #include <boost/tuple/tuple.hpp>
25 #include <boost/tuple/tuple_comparison.hpp>
26 #include <cairomm/pattern.h>
27 #include <gtkmm/drawingarea.h>
28 #include <gdkmm/pixbuf.h>
29
30 namespace Gtkmm2ext {
31
32 class FastMeter : public Gtk::DrawingArea {
33   public:
34         enum Orientation {
35                 Horizontal,
36                 Vertical
37         };
38
39         FastMeter (long hold_cnt, unsigned long width, Orientation, int len=0,
40                         int clr0=0x008800ff, int clr1=0x008800ff,
41                         int clr2=0x00ff00ff, int clr3=0x00ff00ff,
42                         int clr4=0xffaa00ff, int clr5=0xffaa00ff,
43                         int clr6=0xffff00ff, int clr7=0xffff00ff,
44                         int clr8=0xff0000ff, int clr9=0xff0000ff,
45                         int bgc0=0x333333ff, int bgc1=0x444444ff,
46                         int bgh0=0x991122ff, int bgh1=0x551111ff,
47                         float stp0 = 55.0, // log_meter(-18);
48                         float stp1 = 77.5, // log_meter(-9);
49                         float stp2 = 92.5, // log_meter(-3); // 95.0, // log_meter(-2);
50                         float stp3 = 100.0
51                         );
52         virtual ~FastMeter ();
53
54         void set (float level, float peak = -1);
55         void clear ();
56
57         float get_level() { return current_level; }
58         float get_user_level() { return current_user_level; }
59         float get_peak() { return current_peak; }
60
61         long hold_count() { return hold_cnt; }
62         void set_hold_count (long);
63         void set_highlight (bool);
64         bool get_highlight () { return highlight; }
65
66 protected:
67         bool on_expose_event (GdkEventExpose*);
68         void on_size_request (GtkRequisition*);
69         void on_size_allocate (Gtk::Allocation&);
70
71 private:
72
73         Cairo::RefPtr<Cairo::Pattern> fgpattern;
74         Cairo::RefPtr<Cairo::Pattern> bgpattern;
75         gint pixheight;
76         gint pixwidth;
77
78         float _stp[4];
79         int _clr[10];
80         int _bgc[2];
81         int _bgh[2];
82
83         Orientation orientation;
84         GdkRectangle pixrect;
85         GdkRectangle last_peak_rect;
86         gint request_width;
87         gint request_height;
88         unsigned long hold_cnt;
89         unsigned long hold_state;
90         bool bright_hold;
91         float current_level;
92         float current_peak;
93         float current_user_level;
94         bool highlight;
95
96         bool vertical_expose (GdkEventExpose*);
97         void queue_vertical_redraw (const Glib::RefPtr<Gdk::Window>&, float);
98
99         static Cairo::RefPtr<Cairo::Pattern> generate_meter_pattern (
100                 int w, int h, int *clr, float *stp, bool shade);
101         static Cairo::RefPtr<Cairo::Pattern> request_vertical_meter (
102                 int w, int h, int *clr, float *stp, bool shade);
103
104         static Cairo::RefPtr<Cairo::Pattern> generate_meter_background (
105                 int w, int h, int *bgc, bool shade);
106         static Cairo::RefPtr<Cairo::Pattern> request_vertical_background (
107                 int w, int h, int *bgc, bool shade);
108
109         struct Pattern10MapKey {
110                 Pattern10MapKey (
111                                 int w, int h,
112                                 float stp0, float stp1, float stp2, float stp3,
113                                 int c0, int c1, int c2, int c3,
114                                 int c4, int c5, int c6, int c7,
115                                 int c8, int c9
116                                 )
117                         : dim(w, h)
118                         , stp(stp0, stp1, stp2, stp3)
119                         , cols(c0, c1, c2, c3, c4, c5, c6, c7, c8, c9)
120                 {}
121                 inline bool operator<(const Pattern10MapKey& rhs) const {
122                         return (dim < rhs.dim)
123                                 || (dim == rhs.dim && stp < rhs.stp)
124                                 || (dim == rhs.dim && stp == rhs.stp && cols < rhs.cols);
125                 }
126                 boost::tuple<int, int> dim;
127                 boost::tuple<float, float, float, float> stp;
128                 boost::tuple<int, int, int, int, int, int, int, int, int, int> cols;
129         };
130         typedef std::map<Pattern10MapKey, Cairo::RefPtr<Cairo::Pattern> > Pattern10Map;
131
132         struct PatternBgMapKey {
133                 PatternBgMapKey (int w, int h, int c0, int c1)
134                         : dim(w, h)
135                         , cols(c0, c1)
136                 {}
137                 inline bool operator<(const PatternBgMapKey& rhs) const {
138                         return (dim < rhs.dim) || (dim == rhs.dim && cols < rhs.cols);
139                 }
140                 boost::tuple<int, int> dim;
141                 boost::tuple<int, int> cols;
142         };
143         typedef std::map<PatternBgMapKey, Cairo::RefPtr<Cairo::Pattern> > PatternBgMap;
144
145         static Pattern10Map vm_pattern_cache;
146         static PatternBgMap vb_pattern_cache;
147         static int min_pattern_metric_size; // min dimension for axis that displays the meter level
148         static int max_pattern_metric_size; // max dimension for axis that displays the meter level
149 };
150
151
152 } /* namespace */
153
154  #endif /* __gtkmm2ext_fastmeter_h__ */