fix crash when copy'ing latent plugins
[ardour.git] / libs / gtkmm2ext / ardour_icon.cc
1 /*
2     Copyright (C) 2009 Paul Davis
3     Copyright (C) 2015 Robin Gareus <robin@gareus.org>
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 along
16     with this program; if not, write to the Free Software Foundation, Inc.,
17     51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18
19 */
20
21 #include <math.h> // M_PI
22 #include <assert.h>
23 #include <algorithm> // std:min
24 #include "gtkmm2ext/ardour_icon.h"
25
26 using namespace Gtkmm2ext::ArdourIcon;
27
28 /* general style info:
29  *
30  * - geometry: icons should be centered, spanning
31  *   wh = std::min (width * .5, height *.5) * .55;
32  *
33  * - all shapes should have a contrasting outline
34  *   (usually white foreground, black outline)
35  */
36
37 #define OUTLINEWIDTH 1.5 // px
38
39 #define VECTORICONSTROKEFILL(fillalpha)                    \
40         cairo_set_line_width (cr, OUTLINEWIDTH);           \
41         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);          \
42         cairo_stroke_preserve (cr);                        \
43         cairo_set_source_rgba (cr, 1, 1, 1, (fillalpha));  \
44         cairo_fill (cr);
45
46 #define VECTORICONSTROKEOUTLINE(LW, color)                 \
47         cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);     \
48         cairo_set_line_width (cr, (LW) + OUTLINEWIDTH);    \
49         ardour_icon_set_source_inv_rgba (cr, color);       \
50         cairo_stroke_preserve (cr);                        \
51         ardour_icon_set_source_rgba (cr, color);           \
52         cairo_set_line_width (cr, (LW));                   \
53         cairo_stroke (cr);
54
55
56 /** convert 32bit 'RRGGBBAA' to cairo doubles
57  * from libs/canvas/utils.cc and  canvas/types.h: typedef uint32_t Color;
58  */
59 static void ardour_icon_set_source_rgba (cairo_t *cr, uint32_t color)
60 {
61         cairo_set_source_rgba (cr,
62                         ((color >> 24) & 0xff) / 255.0,
63                         ((color >> 16) & 0xff) / 255.0,
64                         ((color >>  8) & 0xff) / 255.0,
65                         ((color >>  0) & 0xff) / 255.0
66                         );
67 }
68
69 /** inverse color */
70 static void ardour_icon_set_source_inv_rgba (cairo_t *cr, uint32_t color)
71 {
72         cairo_set_source_rgba (cr,
73                         1.0 - ((color >> 24) & 0xff) / 255.0,
74                         1.0 - ((color >> 16) & 0xff) / 255.0,
75                         1.0 - ((color >>  8) & 0xff) / 255.0,
76                         ((color >>  0) & 0xff) / 255.0
77                         );
78 }
79
80 /*****************************************************************************
81  * Tool Icons.
82  * Foreground is always white, compatible with small un-blurred rendering.
83  */
84
85 /** internal edit icon */
86 static void icon_tool_content (cairo_t *cr, const int width, const int height) {
87 #define EM_POINT(X,Y) round (x + (X) * em) + .5, round (y + (Y) * em) + .5
88
89                 const double x  = width * .5;
90                 const double y  = height * .5;
91                 const double em = std::min (x, y) * .1; // 1px at 20x20
92
93                 // draw dot outlines (control-points)
94                 cairo_move_to (cr, EM_POINT(-6.0,  0.0));
95                 cairo_close_path (cr);
96                 cairo_move_to (cr, EM_POINT(-2.5,  4.0));
97                 cairo_close_path (cr);
98                 cairo_move_to (cr, EM_POINT( 5.0, -5.0));
99                 cairo_close_path (cr);
100
101                 cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
102                 ardour_icon_set_source_inv_rgba (cr, 0xffffffff);
103                 cairo_set_line_width (cr, 3 * em + OUTLINEWIDTH);
104                 cairo_stroke (cr);
105
106                 // "midi note" lines
107                 cairo_move_to (cr, EM_POINT(-7.0, -5.0));
108                 cairo_line_to (cr, EM_POINT( 0.0, -5.0));
109
110                 cairo_move_to (cr, EM_POINT( 2.0,  4.0));
111                 cairo_line_to (cr, EM_POINT( 6.0,  4.0));
112
113                 // automation line (connect control-points)
114                 cairo_move_to (cr, EM_POINT(-6.0,  0.0));
115                 cairo_line_to (cr, EM_POINT(-2.5,  4.0));
116                 cairo_line_to (cr, EM_POINT( 5.0, -5.0));
117
118                 cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
119                 VECTORICONSTROKEOUTLINE(1 * em, 0xffffffff);
120
121                 // remove automation line outline at control-points
122                 cairo_move_to (cr, EM_POINT(-6.0,  0.0));
123                 cairo_close_path (cr);
124                 cairo_move_to (cr, EM_POINT(-2.5,  4.0));
125                 cairo_close_path (cr);
126                 cairo_move_to (cr, EM_POINT( 5.0, -5.0));
127                 cairo_close_path (cr);
128
129                 ardour_icon_set_source_rgba (cr, 0xffffffff);
130                 cairo_set_line_width (cr, 3 * em);
131                 cairo_stroke (cr);
132 #undef EM_POINT
133 }
134
135 /** range tool |<->| */
136 static void icon_tool_range (cairo_t *cr, const int width, const int height)
137 {
138         const double x  = width * .5;
139         const double y  = height * .5;
140         const double wh = std::min (x, y) * .55;
141         const double lw = rint (wh / 6.0); // line width
142         const double ar = wh * .6; // arrow
143
144         const double bw = ceil (wh) - .5;
145         const double y0 = ceil (y);
146         const double ym = rint (y0 - wh * .1) + .5; // arrow-horizontal; slightly to the top, on a px
147         const double x0 = rint (x) - bw; // left arrow tip
148         const double x1 = rint (x) + bw; // right arrow tip
149
150         // left and right box
151         cairo_move_to (cr, x0, y0 - bw);
152         cairo_line_to (cr, x0, y0 + bw);
153         VECTORICONSTROKEOUTLINE(lw, 0xffffffff);
154         cairo_move_to (cr, x1, y0 - bw);
155         cairo_line_to (cr, x1, y0 + bw);
156         VECTORICONSTROKEOUTLINE(lw, 0xffffffff);
157
158         // arrows
159         cairo_move_to (cr, x0 + ar, ym - ar);
160         cairo_line_to (cr, x0 + .5, ym);
161         cairo_line_to (cr, x0 + ar, ym + ar);
162
163         cairo_move_to (cr, x1 - ar, ym - ar);
164         cairo_line_to (cr, x1 - .5, ym);
165         cairo_line_to (cr, x1 - ar, ym + ar);
166
167         // line connecting the arrows
168         cairo_move_to (cr, x0, ym);
169         cairo_line_to (cr, x1, ym);
170         VECTORICONSTROKEOUTLINE(lw, 0xffffffff);
171
172         cairo_set_source_rgba (cr, 1, 1, 1, 1.0);
173         cairo_set_line_width (cr, lw);
174
175         cairo_move_to (cr, x0, y0 - bw);
176         cairo_line_to (cr, x0, y0 + bw);
177         cairo_stroke (cr);
178
179         cairo_move_to (cr, x1, y0 - bw);
180         cairo_line_to (cr, x1, y0 + bw);
181         cairo_stroke (cr);
182
183
184 }
185
186 /** Grab/Object tool - 6x8em "hand", with 'em' wide index finger. */
187 static void icon_tool_grab (cairo_t *cr, const int width, const int height)
188 {
189         const double x  = width * .5;
190         const double y  = height * .5;
191         const double em = std::min (x, y) * .15; // 1.5px at 20x20
192
193 #define EM_POINT(X,Y) x + (X) * em, y + (Y) * em
194
195         // wrist
196         cairo_move_to (cr, EM_POINT( 2.0,  4.0));
197         cairo_line_to (cr, EM_POINT(-1.5,  4.0));
198         cairo_line_to (cr, EM_POINT(-2.5,  2.0));
199         // thumb
200         cairo_line_to (cr, EM_POINT(-3.0,  1.0));
201
202         // index finger
203         cairo_line_to (cr, EM_POINT(-2.0,  0.0));
204         cairo_line_to (cr, EM_POINT(-2.1, -4.0));
205         cairo_line_to (cr, EM_POINT(-1.5, -4.5));
206         cairo_line_to (cr, EM_POINT(-1.1, -4.0));
207         cairo_line_to (cr, EM_POINT(-1.0,  0.1));
208
209         // middle finger knuckle
210         cairo_line_to (cr, EM_POINT(-0.6,  0.3));
211         cairo_line_to (cr, EM_POINT(-0.3,  0.0));
212         cairo_line_to (cr, EM_POINT(-0.2, -0.2));
213         cairo_line_to (cr, EM_POINT( 0.1, -0.3));
214         cairo_line_to (cr, EM_POINT( 0.4, -0.2));
215         cairo_line_to (cr, EM_POINT( 0.5,  0.1));
216
217         // ring finger knuckle
218         cairo_line_to (cr, EM_POINT( 0.8,  0.4));
219         cairo_line_to (cr, EM_POINT( 1.1,  0.2));
220         cairo_line_to (cr, EM_POINT( 1.2,  0.0));
221         cairo_line_to (cr, EM_POINT( 1.5, -0.1));
222         cairo_line_to (cr, EM_POINT( 1.8,  0.0));
223         cairo_line_to (cr, EM_POINT( 1.9,  0.4));
224
225         // pinky
226         cairo_line_to (cr, EM_POINT( 2.0,  0.6));
227         cairo_line_to (cr, EM_POINT( 2.4,  0.4));
228         cairo_line_to (cr, EM_POINT( 2.8,  0.5));
229         cairo_line_to (cr, EM_POINT( 3.0,  1.0));
230
231         // wrist
232         cairo_line_to (cr, EM_POINT( 3.0,  1.5));
233         cairo_line_to (cr, EM_POINT( 2.0,  4.0));
234
235         cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
236         cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
237         VECTORICONSTROKEFILL(1.0);
238 #undef EM_POINT
239 }
240
241 /** cut icon - scissors */
242 static void icon_tool_cut (cairo_t *cr, const int width, const int height)
243 {
244         const double x  = width * .5;
245         const double y  = height * .5;
246         const double em = std::min (x, y) * .1; // 1px at 20x20
247
248 #define EM_POINT(X,Y) x + (X) * em, y + (Y) * em
249
250         cairo_save (cr);
251         cairo_translate (cr, EM_POINT(4, -3));
252         cairo_scale (cr, 1.6, 1.0); // ellipse
253         cairo_arc (cr, 0., 0., 1.5 * em, 0., 2 * M_PI);
254         cairo_restore (cr);
255
256         cairo_move_to (cr, EM_POINT(-6.0,  2.5));
257         cairo_line_to (cr, EM_POINT( 5.5, -2.0));
258
259         cairo_move_to (cr, EM_POINT(-6.0, -2.5));
260         cairo_line_to (cr, EM_POINT( 5.5,  2.0));
261
262         cairo_save (cr);
263         cairo_translate (cr, EM_POINT(4,  3));
264         cairo_scale (cr, 1.6, 1.0); // ellipse
265         cairo_arc (cr, 0., 0., 1.5 * em, 0., 2 * M_PI);
266         cairo_restore (cr);
267
268         cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
269         cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
270
271         VECTORICONSTROKEOUTLINE (1.5 * em, 0xffffffff);
272 #undef EM_POINT
273 }
274
275 /** time stretch icon */
276 static void icon_tool_stretch (cairo_t *cr, const int width, const int height)
277 {
278         const double x  = width * .5;
279         const double y  = height * .5;
280         const double wh = std::min (x, y) * .55;
281
282         const double y0 = ceil (y);
283         const double bw = rint (wh);
284         const double lw = rint (wh / 3.0) / 2.0;
285         const double x0 = rint (x + lw) + .5;
286
287         // box indication region
288         cairo_rectangle (cr, x0 - lw - bw - .5, y0 - bw, lw + bw, 2 * bw);
289         VECTORICONSTROKEFILL (0.75);
290
291         cairo_set_line_width (cr, 1.0);
292
293         // inside/left arrow
294         cairo_move_to (cr, x0,          y);
295         cairo_line_to (cr, x0 - lw * 2, y);
296         cairo_line_to (cr, x0 - lw * 2, y - lw * 3.5);
297         cairo_line_to (cr, x0 - lw * 6, y);
298         cairo_line_to (cr, x0 - lw * 2, y + lw * 3.5);
299         cairo_line_to (cr, x0 - lw * 2, y);
300
301         cairo_set_source_rgba (cr, 0, 0, 0, .5);
302         cairo_stroke_preserve (cr);
303         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
304         cairo_fill (cr);
305
306         // outside/right arrow
307         cairo_move_to (cr, x0,          y);
308         cairo_line_to (cr, x0 + lw * 2, y);
309         cairo_line_to (cr, x0 + lw * 2, y - lw * 4);
310         cairo_line_to (cr, x0 + lw * 6, y);
311         cairo_line_to (cr, x0 + lw * 2, y + lw * 4);
312         cairo_line_to (cr, x0 + lw * 2, y);
313
314         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
315         cairo_stroke_preserve (cr);
316         cairo_set_source_rgba (cr, 1, 1, 1, 1.0);
317         cairo_fill (cr);
318 }
319
320 /** audition - small speaker with sound-waves*/
321 static void icon_tool_audition (cairo_t *cr, const int width, const int height)
322 {
323         const double x  = width * .5;
324         const double y  = height * .5;
325         const double em = std::min (x, y) * .1; // 1px at 20x20
326
327 #define EM_POINT(X,Y) x + (X) * em, y + (Y) * em
328
329         cairo_move_to (cr, EM_POINT(-7.0, -2.0));
330         cairo_line_to (cr, EM_POINT(-7.0,  2.0));
331         cairo_line_to (cr, EM_POINT(-6.0,  3.0));
332         cairo_line_to (cr, EM_POINT(-3.0,  3.0));
333         cairo_line_to (cr, EM_POINT( 2.0,  6.0));
334         cairo_line_to (cr, EM_POINT( 2.0, -6.0));
335         cairo_line_to (cr, EM_POINT(-3.0, -3.0));
336         cairo_line_to (cr, EM_POINT(-6.0, -3.0));
337         cairo_close_path (cr);
338
339         cairo_pattern_t *speaker;
340         speaker = cairo_pattern_create_linear (EM_POINT(0, -3.0), EM_POINT(0, 3.0));
341         cairo_pattern_add_color_stop_rgba (speaker, 0.0,  0.8, 0.8, 0.8, 1.0);
342         cairo_pattern_add_color_stop_rgba (speaker, 0.25, 1.0, 1.0, 1.0, 1.0);
343         cairo_pattern_add_color_stop_rgba (speaker, 1.0,  0.6, 0.6, 0.6, 1.0);
344
345         cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
346         cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
347         cairo_set_line_width (cr, 1.5);
348         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
349         cairo_stroke_preserve (cr);
350         cairo_set_source (cr, speaker);
351         cairo_fill (cr);
352         cairo_pattern_destroy (speaker);
353
354         // TODO use a slight curve
355         cairo_move_to (cr, EM_POINT(-3.0, -3.0));
356         cairo_line_to (cr, EM_POINT(-3.5,  0.0));
357         cairo_line_to (cr, EM_POINT(-3.0,  3.0));
358         cairo_set_source_rgba (cr, 0, 0, 0, 0.7);
359         cairo_set_line_width (cr, 1.0);
360         cairo_stroke (cr);
361
362
363         cairo_save (cr);
364         cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
365         cairo_set_source_rgba (cr, 1, 1, 1, 1);
366
367         cairo_translate (cr, EM_POINT (4.0, 0));
368         cairo_scale (cr, 0.8, 1.25); // ellipse
369
370         cairo_arc (cr, 0, 0, 4 * em, -.5 * M_PI, .5 * M_PI);
371         cairo_set_line_width (cr, .8 * em);
372         cairo_stroke (cr);
373
374         cairo_arc (cr, 0, 0, 2 * em, -.5 * M_PI, .5 * M_PI);
375         cairo_set_line_width (cr, .5 * em);
376         cairo_stroke (cr);
377         cairo_restore (cr);
378 #undef EM_POINT
379 }
380
381 /** pen top-left to bottom right */
382 static void icon_tool_draw (cairo_t *cr, const int width, const int height)
383 {
384         const double x  = width * .5;
385         const double y  = height * .5;
386         const double em = std::min (x, y) * .1; // 1px at 20x20
387
388 #define EM_POINT(X,Y) x + (X) * em, y + (Y) * em
389
390         // pen [6,-5] to [-3, 3]
391         // y = -8 * x / 9 + 1/3
392
393         // top-right end
394         cairo_move_to (cr, EM_POINT( 5.0, -6.11));
395         cairo_line_to (cr, EM_POINT( 6.4, -5.35)); // todo round properly.
396         cairo_line_to (cr, EM_POINT( 7.0, -3.88));
397
398         // bottom-left w/tip
399         cairo_line_to (cr, EM_POINT(-2.0,  4.11));
400         cairo_line_to (cr, EM_POINT(-6.0,  5.66)); // pen tip
401         cairo_line_to (cr, EM_POINT(-4.0,  1.88));
402         cairo_close_path (cr);
403
404         cairo_pattern_t *pen;
405         pen = cairo_pattern_create_linear (EM_POINT(-3.0, -6.0), EM_POINT(6.0, 4.0));
406         cairo_pattern_add_color_stop_rgba (pen, 0.4, 0.6, 0.6, 0.6, 1.0);
407         cairo_pattern_add_color_stop_rgba (pen, 0.5, 1.0, 1.0, 1.0, 1.0);
408         cairo_pattern_add_color_stop_rgba (pen, 0.6, 0.1, 0.1, 0.1, 1.0);
409
410         cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
411         cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
412         cairo_set_line_width (cr, em + .5);
413         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
414         cairo_stroke_preserve (cr);
415         cairo_set_source (cr, pen);
416         cairo_fill (cr);
417
418         // separate the tip
419         cairo_move_to (cr, EM_POINT(-2.0,  4.11));
420         cairo_line_to (cr, EM_POINT(-3.0,  2.8)); // slight curve [-3,3]
421         cairo_line_to (cr, EM_POINT(-4.0,  2.0));
422         cairo_set_line_width (cr, OUTLINEWIDTH);
423         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
424         cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
425         cairo_stroke (cr);
426
427         // pen tip
428         cairo_move_to (cr, EM_POINT(-5.0, 3.9));
429         cairo_line_to (cr, EM_POINT(-6.0, 5.66));
430         cairo_line_to (cr, EM_POINT(-4.1, 4.9));
431         cairo_close_path (cr);
432         cairo_set_source_rgba (cr, 0, 0, 0, 0.7);
433         cairo_set_line_width (cr, em);
434         cairo_stroke_preserve (cr);
435         cairo_fill (cr);
436
437         cairo_pattern_destroy (pen);
438 #undef EM_POINT
439 }
440
441 /** Toolbar icon - Time Axis View reduce height */
442 static void icon_tav_shrink (cairo_t *cr, const int width, const int height)
443 {
444         const double x = width * .5;
445         const double y = height * .5;
446         const double wh = std::min (x, y) * .66;
447         const double ar = std::min (x, y) * .15;
448         const double tri = .7 * (wh - ar);
449
450         cairo_rectangle (cr, x - wh, y - ar, 2 * wh, 2 * ar);
451         VECTORICONSTROKEFILL(.75);
452
453         cairo_set_line_width (cr, 1.0);
454
455         cairo_move_to (cr, x,       y - ar - 0.5);
456         cairo_line_to (cr, x - tri, y - wh + 0.5);
457         cairo_line_to (cr, x + tri, y - wh + 0.5);
458         cairo_close_path (cr);
459
460         cairo_set_source_rgba (cr, 1, 1, 1, .75);
461         cairo_stroke_preserve (cr);
462         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
463         cairo_fill (cr);
464
465         cairo_move_to (cr, x,       y + ar + 0.5);
466         cairo_line_to (cr, x - tri, y + wh - 0.5);
467         cairo_line_to (cr, x + tri, y + wh - 0.5);
468         cairo_close_path (cr);
469
470         cairo_set_source_rgba (cr, 1, 1, 1, .75);
471         cairo_stroke_preserve (cr);
472         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
473         cairo_fill (cr);
474 }
475
476 /** Toolbar icon - Time Axis View increase height */
477 static void icon_tav_expand (cairo_t *cr, const int width, const int height)
478 {
479         const double x = width * .5;
480         const double y = height * .5;
481         const double wh = std::min (x, y) * .66;
482         const double ar = std::min (x, y) * .15;
483         const double tri = .7 * (wh - ar);
484
485         cairo_rectangle (cr, x - wh, y - wh, 2 * wh, 2 * wh);
486         VECTORICONSTROKEFILL(.75);
487
488         cairo_set_line_width (cr, 1.0);
489
490         cairo_move_to (cr, x,       y - wh + 0.5);
491         cairo_line_to (cr, x - tri, y - ar - 0.5);
492         cairo_line_to (cr, x + tri, y - ar - 0.5);
493         cairo_close_path (cr);
494
495         cairo_set_source_rgba (cr, 1, 1, 1, .5);
496         cairo_stroke_preserve (cr);
497         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
498         cairo_fill (cr);
499
500         cairo_move_to (cr, x      , y + wh - 0.5);
501         cairo_line_to (cr, x - tri, y + ar + 0.5);
502         cairo_line_to (cr, x + tri, y + ar + 0.5);
503         cairo_close_path (cr);
504
505         cairo_set_source_rgba (cr, 1, 1, 1, .5);
506         cairo_stroke_preserve (cr);
507         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
508         cairo_fill (cr);
509 }
510
511
512 /*****************************************************************************
513  * Record enable (transport & track header).
514  *
515  * hardcoded "red" #f46f6f
516  */
517
518 /** standard rec-enable circle */
519 static void icon_rec_enable (cairo_t *cr, const int width, const int height, const Gtkmm2ext::ActiveState state)
520 {
521         const double x = width * .5;
522         const double y = height * .5;
523         const double r = std::min (x, y) * .55;
524         cairo_arc (cr, x, y, r, 0, 2 * M_PI);
525         if (state == Gtkmm2ext::ExplicitActive) {
526                 cairo_set_source_rgba (cr, 1.0, .1, .1, 1.0);
527         }
528         else if (state == Gtkmm2ext::ImplicitActive) {
529                 cairo_set_source_rgba (cr, .9, .3, .3, 1.0);
530         }
531         else {
532                 cairo_set_source_rgba (cr, .4, .3, .3, 1.0);
533         }
534         cairo_fill_preserve (cr);
535         cairo_set_source_rgba (cr, 0.0, 0.0, 0.0, 0.8); // outline
536         cairo_set_line_width (cr, 1);
537         cairo_stroke (cr);
538 }
539
540 /** tape-mode, "reel" */
541 static void icon_rec_tape (cairo_t *cr, const int width, const int height, const Gtkmm2ext::ActiveState state)
542 {
543         const double x = width * .5;
544         const double y = height * .5;
545         const double r = std::min (x, y) * .6;
546         const double slit = .11 * M_PI;
547         cairo_translate (cr, x, y);
548
549         cairo_arc (cr, 0, 0, r, 0, 2 * M_PI);
550         if (state == Gtkmm2ext::ExplicitActive) {
551                 cairo_set_source_rgba (cr, 1.0, .1, .1, 1.0);
552         }
553         else if (state == Gtkmm2ext::ImplicitActive) {
554                 cairo_set_source_rgba (cr, .9, .3, .3, 1.0);
555         }
556         else {
557                 cairo_set_source_rgba (cr, .4, .3, .3, 1.0);
558         }
559         cairo_fill_preserve (cr);
560         cairo_set_source_rgba (cr, .0, .0, .0, .5);
561         cairo_set_line_width (cr, 1);
562         cairo_stroke (cr);
563
564         cairo_save (cr);
565         cairo_set_source_rgba (cr, .15, .07, .07, 1.0);
566
567         cairo_rotate (cr, -.5 * M_PI);
568         cairo_move_to (cr, 0, 0);
569         cairo_arc (cr, 0, 0, r *.85, -slit, slit);
570         cairo_line_to (cr, 0, 0);
571         cairo_close_path (cr);
572
573         cairo_fill (cr);
574         cairo_rotate (cr, 2. * M_PI / 3.);
575
576         cairo_move_to (cr, 0, 0);
577         cairo_arc (cr, 0, 0, r *.85, -slit, slit);
578         cairo_line_to (cr, 0, 0);
579         cairo_close_path (cr);
580         cairo_fill (cr);
581
582         cairo_rotate (cr, 2. * M_PI / 3.);
583         cairo_move_to (cr, 0, 0);
584         cairo_arc (cr, 0, 0, r *.85, -slit, slit);
585         cairo_line_to (cr, 0, 0);
586         cairo_close_path (cr);
587         cairo_fill (cr);
588
589         cairo_restore (cr);
590
591         cairo_arc (cr, 0, 0, r * .3, 0, 2 * M_PI);
592         if (state == Gtkmm2ext::ExplicitActive) {
593                 cairo_set_source_rgba (cr, 1.0, .1, .1, 1.0);
594         }
595         else if (state == Gtkmm2ext::ImplicitActive) {
596                 cairo_set_source_rgba (cr, .9, .3, .3, 1.0);
597         }
598         else {
599                 cairo_set_source_rgba (cr, .4, .3, .3, 1.0);
600         }
601         cairo_fill (cr);
602         cairo_set_source_rgba (cr, .0, .0, .0, 1.0);
603         cairo_arc (cr, 0, 0, r *.15, 0, 2 * M_PI); // hole in the middle
604         cairo_fill (cr);
605 }
606
607
608 /*****************************************************************************
609  * Transport buttons, foreground is always white
610  */
611
612 /** stop square box */
613 static void icon_transport_stop (cairo_t *cr, const int width, const int height)
614 {
615         const int wh = std::min (width, height);
616         cairo_rectangle (cr,
617                         (width - wh) * .5 + wh * .25,
618                         (height - wh) * .5 + wh * .25,
619                         wh * .5, wh * .5);
620         VECTORICONSTROKEFILL(0.9); // small 'shine'
621 }
622
623 /** play triangle */
624 static void icon_transport_play (cairo_t *cr, const int width, const int height)
625 {
626         const int wh = std::min (width, height) * .5;
627         const double y = height * .5;
628         const double x = width - wh;
629
630         const double tri = ceil (.577 * wh); // 1/sqrt(3)
631
632         cairo_move_to (cr,  x + wh * .5, y);
633         cairo_line_to (cr,  x - wh * .5, y - tri);
634         cairo_line_to (cr,  x - wh * .5, y + tri);
635         cairo_close_path (cr);
636
637         VECTORICONSTROKEFILL(0.9);
638 }
639
640 /** Midi Panic "!" */
641 static void icon_transport_panic (cairo_t *cr, const int width, const int height)
642 {
643         const int wh = std::min (width, height) * .1;
644         const double xc = width * .5;
645         const double yh = height;
646         cairo_rectangle (cr,
647                          xc - wh, yh *.19,
648                          wh * 2,  yh *.41);
649         VECTORICONSTROKEFILL(0.9);
650
651         cairo_arc (cr, xc, yh *.75, wh, 0, 2 * M_PI);
652         VECTORICONSTROKEFILL(0.9);
653 }
654
655 /** various combinations of lines and triangles "|>|", ">|" "|>" */
656 static void icon_transport_ck (cairo_t *cr,
657                 const enum Gtkmm2ext::ArdourIcon::Icon icon,
658                 const int width, const int height)
659 {
660         // small play triangle
661         int wh = std::min (width, height);
662         const double y = height * .5;
663         const double x = width - wh * .5;
664         wh *= .18;
665         const double tri = ceil (.577 * wh * 2); // 1/sqrt(3)
666
667         const float ln = std::min (width, height) * .07;
668
669         if (icon == TransportStart || icon == TransportRange) {
670                 cairo_rectangle (cr,
671                                 x - wh - ln, y  - tri * 1.7,
672                                 ln * 2,  tri * 3.4);
673
674                 VECTORICONSTROKEFILL(1.0);
675         }
676
677         if (icon == TransportEnd || icon == TransportRange) {
678                 cairo_rectangle (cr,
679                                 x + wh - ln, y  - tri * 1.7,
680                                 ln * 2,  tri * 3.4);
681
682                 VECTORICONSTROKEFILL(1.0);
683         }
684
685         if (icon == TransportStart) {
686                 cairo_move_to (cr,  x - wh, y);
687                 cairo_line_to (cr,  x + wh, y - tri);
688                 cairo_line_to (cr,  x + wh, y + tri);
689         } else {
690                 cairo_move_to (cr,  x + wh, y);
691                 cairo_line_to (cr,  x - wh, y - tri);
692                 cairo_line_to (cr,  x - wh, y + tri);
693         }
694
695         cairo_close_path (cr);
696         VECTORICONSTROKEFILL(1.0);
697 }
698
699 /** loop spiral */
700 static void icon_transport_loop (cairo_t *cr, const int width, const int height)
701 {
702         const double x = width * .5;
703         const double y = height * .5;
704         const double r = std::min (x, y);
705
706         cairo_arc          (cr, x, y, r * .62, 0, 2 * M_PI);
707         cairo_arc_negative (cr, x, y, r * .35, 2 * M_PI, 0);
708
709         VECTORICONSTROKEFILL(1.0);
710
711 #define ARCARROW(rad, ang) \
712         x + (rad) * sin ((ang) * 2.0 * M_PI), y + (rad) * cos ((ang) * 2.0 * M_PI)
713
714         cairo_move_to (cr, ARCARROW(r * .35, .72));
715         cairo_line_to (cr, ARCARROW(r * .15, .72));
716         cairo_line_to (cr, ARCARROW(r * .56, .60));
717         cairo_line_to (cr, ARCARROW(r * .75, .72));
718         cairo_line_to (cr, ARCARROW(r * .62, .72));
719
720         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
721         cairo_stroke_preserve (cr);
722         cairo_close_path (cr);
723         cairo_set_source_rgba (cr, 1, 1, 1, 1.0);
724         cairo_fill (cr);
725 #undef ARCARROW
726 }
727
728 /** de-construct thorwil's metronom */
729 static void icon_transport_metronom (cairo_t *cr, const int width, const int height)
730 {
731         const double x  = width * .5;
732         const double y  = height * .5;
733         const double wh = std::min (x, y);
734         const double h  = wh * .85;
735         const double w  = wh * .55;
736         const double lw = w  * .34;
737
738         cairo_rectangle (cr,
739                         x - w * .7, y + h * .25,
740                         w * 1.4, lw);
741
742         VECTORICONSTROKEFILL(1.0);
743
744         cairo_move_to (cr,  x - w,       y + h);
745         cairo_line_to (cr,  x + w,       y + h);
746         cairo_line_to (cr,  x + w * .35, y - h);
747         cairo_line_to (cr,  x - w * .35, y - h);
748         cairo_line_to (cr,  x - w,       y + h);
749
750         cairo_move_to (cr,  x - w + lw,       y + h -lw);
751         cairo_line_to (cr,  x - w * .35 + lw, y - h + lw);
752         cairo_line_to (cr,  x + w * .35 - lw, y - h + lw);
753         cairo_line_to (cr,  x + w - lw,       y + h -lw);
754         cairo_line_to (cr,  x - w + lw,       y + h -lw);
755
756         VECTORICONSTROKEFILL(1.0);
757
758         // Pendulum
759         // ddx = .70 w      = .75 * .5 wh              = .375 wh
760         // ddy = .75 h - lw = .75 * .8 wh - wh .5 * .2 = .5 wh
761         // ang = (ddx/ddy):
762         // -> angle = atan (ang) = atan (375 / .5) ~= 36deg
763         const double dx = lw * .2;  // 1 - cos(tan^-1(ang))
764         const double dy = lw * .4;  // 1 - sin(tan^-1(ang))
765         cairo_move_to (cr,  x - w * .3     , y + h * .25 + lw * .5);
766         cairo_line_to (cr,  x - w + dx     , y - h + lw + dy);
767         cairo_line_to (cr,  x - w + lw     , y - h + lw);
768         cairo_line_to (cr,  x - w * .3 + lw, y + h * .25 + lw * .5);
769         cairo_close_path (cr);
770
771         VECTORICONSTROKEFILL(1.0);
772
773         cairo_rectangle (cr,
774                         x - w * .7, y + h * .25,
775                         w * 1.4, lw);
776         cairo_fill (cr);
777 }
778
779
780 /*****************************************************************************
781  * Zoom: In "+", Out "-" and Full "[]"
782  */
783 static void icon_zoom (cairo_t *cr, const enum Gtkmm2ext::ArdourIcon::Icon icon, const int width, const int height, const uint32_t fg_color)
784 {
785         const double x = width * .5;
786         const double y = height * .5;
787         const double r = std::min (x, y) * .7;
788         const double wh = std::min (x, y) * .45;
789
790         // draw handle first
791 #define LINE45DEG(rad) \
792         x + r * (rad) * .707, y + r * (rad) * .707 // sin(45deg) = cos(45deg) = .707
793         cairo_move_to (cr, LINE45DEG(.9));
794         cairo_line_to (cr, LINE45DEG(1.3));
795         cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
796         cairo_set_line_width (cr, 3.0);
797         cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
798         cairo_stroke (cr);
799 #undef LINE45DEG
800
801         // lens
802         ardour_icon_set_source_rgba (cr, fg_color);
803         cairo_arc (cr, x, y, r, 0, 2 * M_PI);
804         cairo_fill_preserve (cr);
805
806         // add a lens gradient
807         cairo_pattern_t *lens;
808         lens = cairo_pattern_create_radial (x - r, y - r, r * .5, x - r, y - r, r * 2);
809         cairo_pattern_add_color_stop_rgba (lens, 0, 1, 1, 1, .4);
810         cairo_pattern_add_color_stop_rgba (lens, 1, 0, 0, 0, .4);
811         cairo_set_source (cr, lens);
812         cairo_fill_preserve (cr);
813         cairo_pattern_destroy (lens);
814
815         // outline
816         cairo_set_line_width (cr, 1.5);
817         //ardour_icon_set_source_inv_rgba (cr, fg_color); // alpha
818         cairo_set_source_rgba (cr, .0, .0, .0, .8);
819         cairo_stroke (cr);
820
821         // add "+", "-" or "[]"
822         cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
823         cairo_set_line_width (cr, 1.5);
824         ardour_icon_set_source_inv_rgba (cr, fg_color);
825
826         if (icon == ZoomIn || icon == ZoomOut) {
827                 cairo_move_to (cr, x - wh, y);
828                 cairo_line_to (cr, x + wh, y);
829                 cairo_stroke (cr);
830         }
831         if (icon == ZoomIn) {
832                 cairo_move_to (cr, x, y - wh);
833                 cairo_line_to (cr, x, y + wh);
834                 cairo_stroke (cr);
835         }
836         if (icon == ZoomFull) {
837                 const double br0 = std::min (x, y) * .1;
838                 const double br1 = std::min (x, y) * .3;
839                 const double bry = std::min (x, y) * .3;
840                 cairo_move_to (cr, x - br0, y - bry);
841                 cairo_line_to (cr, x - br1, y - bry);
842                 cairo_line_to (cr, x - br1, y + bry);
843                 cairo_line_to (cr, x - br0, y + bry);
844                 cairo_stroke (cr);
845
846                 cairo_move_to (cr, x + br0, y - bry);
847                 cairo_line_to (cr, x + br1, y - bry);
848                 cairo_line_to (cr, x + br1, y + bry);
849                 cairo_line_to (cr, x + br0, y + bry);
850                 cairo_stroke (cr);
851         }
852 }
853
854
855 /*****************************************************************************
856  * Misc buttons
857  */
858
859 /** "close" - "X" , no outline */
860 static void icon_close_cross (cairo_t *cr, const int width, const int height, const uint32_t fg_color)
861 {
862         const double x = width * .5;
863         const double y = height * .5;
864         const double o = .5 + std::min (x, y) * .4;
865         ardour_icon_set_source_rgba (cr, fg_color);
866         cairo_set_line_width (cr, 1.0);
867         cairo_move_to (cr, x-o, y-o);
868         cairo_line_to (cr, x+o, y+o);
869         cairo_move_to (cr, x+o, y-o);
870         cairo_line_to (cr, x-o, y+o);
871         cairo_stroke (cr);
872 }
873
874 /** "<" */
875 static void icon_nudge_left (cairo_t *cr, const int width, const int height, const uint32_t fg_color)
876 {
877         const double x = width * .5;
878         const double y = height * .5;
879         const double wh = std::min (x, y);
880
881         const double tri_x = .3 * wh;
882         const double tri_y = .6 * wh;
883
884         cairo_move_to (cr, x + tri_x, y - tri_y);
885         cairo_line_to (cr, x - tri_x, y);
886         cairo_line_to (cr, x + tri_x, y + tri_y);
887         VECTORICONSTROKEOUTLINE(1.5, fg_color);
888 }
889
890 /** ">" */
891 static void icon_nudge_right (cairo_t *cr, const int width, const int height, const uint32_t fg_color)
892 {
893
894         const double x = width * .5;
895         const double y = height * .5;
896         const double wh = std::min (x, y);
897
898         const double tri_x = .3 * wh;
899         const double tri_y = .6 * wh;
900
901         cairo_move_to (cr, x - tri_x, y - tri_y);
902         cairo_line_to (cr, x + tri_x, y);
903         cairo_line_to (cr, x - tri_x, y + tri_y);
904         VECTORICONSTROKEOUTLINE(1.5, fg_color);
905
906 }
907
908 /** mixer strip narrow/wide */
909 static void icon_strip_width (cairo_t *cr, const int width, const int height, const uint32_t fg_color)
910 {
911         const double x0 = width   * .2;
912         const double x1 = width   * .8;
913
914         const double y0 = height  * .25;
915         const double y1 = height  * .75;
916
917         const double ym = height  * .5;
918
919         // arrow
920         const double xa0= width  * .39;
921         const double xa1= width  * .61;
922         const double ya0= height * .35;
923         const double ya1= height * .65;
924
925         ardour_icon_set_source_rgba (cr, fg_color);
926         cairo_set_line_width (cr, 1);
927
928         // left + right
929         cairo_move_to (cr, x0, y0);
930         cairo_line_to (cr, x0, y1);
931         cairo_move_to (cr, x1, y0);
932         cairo_line_to (cr, x1, y1);
933
934         // horiz center line
935         cairo_move_to (cr, x0, ym);
936         cairo_line_to (cr, x1, ym);
937
938         // arrow left
939         cairo_move_to (cr,  x0, ym);
940         cairo_line_to (cr, xa0, ya0);
941         cairo_move_to (cr,  x0, ym);
942         cairo_line_to (cr, xa0, ya1);
943
944         // arrow right
945         cairo_move_to (cr,  x1,  ym);
946         cairo_line_to (cr, xa1, ya0);
947         cairo_move_to (cr,  x1,  ym);
948         cairo_line_to (cr, xa1, ya1);
949         cairo_stroke (cr);
950 }
951
952 /** 5-pin DIN MIDI socket */
953 static void icon_din_midi (cairo_t *cr, const int width, const int height, const uint32_t fg_color)
954 {
955         const double x = width * .5;
956         const double y = height * .5;
957         const double r = std::min (x, y) * .75;
958         ardour_icon_set_source_rgba (cr, fg_color);
959         cairo_set_line_width (cr, 1);
960         cairo_arc (cr, x, y, r, .57 * M_PI, 2.43 * M_PI);
961         cairo_stroke (cr);
962
963         // pins equally spaced 45deg
964         cairo_arc (cr, x, y * 0.5, r * .15, 0, 2 * M_PI);
965         cairo_fill (cr);
966         cairo_arc (cr, x * 0.5, y, r * .15, 0, 2 * M_PI);
967         cairo_fill (cr);
968         cairo_arc (cr, x * 1.5, y, r * .15, 0, 2 * M_PI);
969         cairo_fill (cr);
970         //  .5 + .5 * .5 * sin(45deg),  1.5 - .5 * .5 * cos(45deg)
971         cairo_arc (cr, x * 0.677, y * .677, r * .15, 0, 2 * M_PI);
972         cairo_fill (cr);
973         cairo_arc (cr, x * 1.323, y * .677, r * .15, 0, 2 * M_PI);
974         cairo_fill (cr);
975
976         // bottom notch
977         cairo_arc (cr, x, y+r, r * .26, 1.05 * M_PI, 1.95 * M_PI);
978         cairo_stroke (cr);
979 }
980
981
982 /*****************************************************************************/
983
984 bool
985 Gtkmm2ext::ArdourIcon::render (cairo_t *cr,
986                                const enum Gtkmm2ext::ArdourIcon::Icon icon,
987                                const int width, const int height,
988                                const Gtkmm2ext::ActiveState state,
989                                const uint32_t fg_color)
990 {
991         bool rv = true;
992         cairo_save (cr);
993
994         if (width < 6 || height < 6) {
995                 return false;
996         }
997
998         switch (icon) {
999                 case TransportStop:
1000                         icon_transport_stop (cr, width, height);
1001                         break;
1002                 case TransportPlay:
1003                         icon_transport_play (cr, width, height);
1004                         break;
1005                 case TransportLoop:
1006                         icon_transport_loop (cr, width, height);
1007                         break;
1008                 case TransportMetronom:
1009                         icon_transport_metronom (cr, width, height);
1010                         break;
1011                 case TransportPanic:
1012                         icon_transport_panic (cr, width, height);
1013                         break;
1014                 case TransportStart: // no break
1015                 case TransportEnd:   // no break
1016                 case TransportRange:
1017                         icon_transport_ck (cr, icon, width, height);
1018                         break;
1019                 case RecTapeMode:
1020                         icon_rec_tape (cr, width, height, state);
1021                         break;
1022                 case RecButton:
1023                         icon_rec_enable (cr, width, height, state);
1024                         break;
1025                 case CloseCross:
1026                         icon_close_cross (cr, width, height, fg_color);
1027                         break;
1028                 case StripWidth:
1029                         icon_strip_width (cr, width, height, fg_color);
1030                         break;
1031                 case DinMidi:
1032                         icon_din_midi (cr, width, height, fg_color);
1033                         break;
1034                 case NudgeLeft:
1035                         icon_nudge_left (cr, width, height, fg_color);
1036                         break;
1037                 case NudgeRight:
1038                         icon_nudge_right (cr, width, height, fg_color);
1039                         break;
1040                 case ZoomIn:  // no break
1041                 case ZoomOut: // no break
1042                 case ZoomFull:
1043                         icon_zoom (cr, icon, width, height, fg_color);
1044                         break;
1045                 case TimeAxisShrink:
1046                         icon_tav_shrink (cr, width, height);
1047                         break;
1048                 case TimeAxisExpand:
1049                         icon_tav_expand (cr, width, height);
1050                         break;
1051                 case ToolRange:
1052                         icon_tool_range (cr, width, height);
1053                         break;
1054                 case ToolGrab:
1055                         icon_tool_grab (cr, width, height);
1056                         break;
1057                 case ToolCut:
1058                         icon_tool_cut (cr, width, height);
1059                         break;
1060                 case ToolStretch:
1061                         icon_tool_stretch (cr, width, height);
1062                         break;
1063                 case ToolAudition:
1064                         icon_tool_audition (cr, width, height);
1065                         break;
1066                 case ToolDraw:
1067                         icon_tool_draw (cr, width, height);
1068                         break;
1069                 case ToolContent:
1070                         icon_tool_content (cr, width, height);
1071                         break;
1072                 default:
1073                         rv = false;
1074                         break;
1075         }
1076         cairo_restore (cr);
1077         return rv;
1078 }
1079
1080 #undef VECTORICONSTROKEFILL
1081 #undef VECTORICONSTROKEOUTLINE