g_vumeter.c 26.9 KB
Newer Older
Miller Puckette's avatar
Miller Puckette committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/* Copyright (c) 1997-1999 Miller Puckette.
 * For information on usage and redistribution, and for a DISCLAIMER OF ALL
 * WARRANTIES, see the file, "LICENSE.txt," in this distribution. */

/* g_7_guis.c written by Thomas Musil (c) IEM KUG Graz Austria 2000-2001 */
/* thanks to Miller Puckette, Guenther Geiger and Krzystof Czaja */

#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include "m_pd.h"
#include "g_canvas.h"
#include "g_all_guis.h"
#include <math.h>

16
17
18
19
20
21
22
23
#define IEM_VU_DEFAULTSIZE 3
#define IEM_VU_LARGESMALL  2
#define IEM_VU_MINSIZE     2
#define IEM_VU_MAXSIZE     25
#define IEM_VU_STEPS       40
#define IEM_VU_MINDB    -99.9
#define IEM_VU_MAXDB    12.0
#define IEM_VU_OFFSET   100.0
Hans-Christoph Steiner's avatar
Hans-Christoph Steiner committed
24

25
26
27
char *iemgui_vu_scale_str[]={
    "<-99", "-50", "-30", "-20", "-12", "-6",
      "-2", "-0dB",  "+2",  "+6",">+12",  "",
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
};

int iemgui_vu_db2i[]=
{
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
    1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
    2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
    2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
    2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
    2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
    3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
    4, 4, 4, 4, 4, 4, 4, 4, 4, 4,
    5, 5, 5, 5, 5, 5, 5, 5, 5, 5,
    6, 6, 6, 6, 6, 6, 6, 6, 6, 6,
    7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
    8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
    9, 9, 9, 9, 9,10,10,10,10,10,
    11,11,11,11,11,12,12,12,12,12,
    13,13,13,13,14,14,14,14,15,15,
    15,15,16,16,16,16,17,17,17,18,
    18,18,19,19,19,20,20,20,21,21,
    22,22,23,23,24,24,25,26,27,28,
    29,30,31,32,33,33,34,34,35,35,
    36,36,37,37,37,38,38,38,39,39,
    39,39,39,39,40,40
};

int iemgui_vu_col[]=
{
    0,17,16,16,16,16,16,16,16,16,16,16,16,16,16,16,16,
    15,15,15,15,15,15,15,15,15,15,14,14,13,13,13,13,13,13,13,13,13,13,13,19,19,19
};

63
64
extern int gfxstub_haveproperties(void *key);
void vu_check_height(t_vu *x, int h);
Miller Puckette's avatar
Miller Puckette committed
65
t_widgetbehavior vu_widgetbehavior;
66
t_class *vu_class;
Miller Puckette's avatar
Miller Puckette committed
67
68
69

static void vu_update_rms(t_vu *x, t_glist *glist)
{
70
    if (glist_isvisible(glist))
Miller Puckette's avatar
Miller Puckette committed
71
    {
72
        int w4 = x->x_gui.x_w / 4, off=text_ypix(&x->x_gui.x_obj, glist) - 1;
73
        int x1 = text_xpix(&x->x_gui.x_obj, glist),
Jonathan Wilkes's avatar
Jonathan Wilkes committed
74
            y1 = text_ypix(&x->x_gui.x_obj, glist),
75
            quad1 = x1 + w4 + 1, quad3 = x1 + x->x_gui.x_w-w4 - 1;
Miller Puckette's avatar
Miller Puckette committed
76

77
78
79
        //sys_vgui(".x%lx.c coords %lxRCOVER %d %d %d %d\n",
        //    glist_getcanvas(glist), x, quad1 + 1, off + 2, quad3 + 1,
        //    off + (x->x_led_size + 1) * (IEM_VU_STEPS - x->x_rms) + 2);
80
81
        gui_vmess("gui_vumeter_update_rms", "xxiiiiii",
            glist_getcanvas(glist), x,
82
            quad1 + 1, off + 2, quad3 + 1, off + (x->x_led_size + 1) *
Jonathan Wilkes's avatar
Jonathan Wilkes committed
83
            (IEM_VU_STEPS - x->x_rms) + 2, x1, y1);
Miller Puckette's avatar
Miller Puckette committed
84
85
86
87
88
89
90
91
92
    }
}

static void vu_update_peak(t_vu *x, t_glist *glist)
{
    t_canvas *canvas=glist_getcanvas(glist);

    if(glist_isvisible(glist))
    {
93
94
        int x1 = text_xpix(&x->x_gui.x_obj, glist);
        int y1 = text_ypix(&x->x_gui.x_obj, glist);
Miller Puckette's avatar
Miller Puckette committed
95
96
97

        if(x->x_peak)
        {
98
            int i = iemgui_vu_col[x->x_peak];
99
            int j = y1 + (x->x_led_size + 1) * (IEM_VU_STEPS + 1 - x->x_peak)
100
101
                - (x->x_led_size + 1) / 2;

102
103
            //sys_vgui(".x%lx.c coords %lxPLED %d %d %d %d\n",
            //    canvas, x, x1 + 1, j + 2, x1 + x->x_gui.x_w + 2, j + 2);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
104
105
            char colorbuf[MAXPDSTRING];
            sprintf(colorbuf, "#%6.6x", iemgui_color_hex[i]);
106
107
            gui_vmess("gui_vumeter_update_peak", "xxsiiiiii",
                canvas, x, colorbuf,
108
109
110
111
                x1 + 1, j + 2, x1 + x->x_gui.x_w + 2, j + 2, x1, y1);

            //sys_vgui(".x%lx.c itemconfigure %lxPLED -stroke #%6.6x\n",
            //    canvas, x, iemgui_color_hex[i]);
Miller Puckette's avatar
Miller Puckette committed
112
113
114
        }
        else
        {
115
            int mid = x1 + x->x_gui.x_w / 2;
Miller Puckette's avatar
Miller Puckette committed
116

117
118
119
120
121
122
123
124
            //sys_vgui(".x%lx.c itemconfigure %lxPLED -stroke #%6.6x\n",
            //         canvas, x, x->x_gui.x_bcol);
            //sys_vgui(".x%lx.c coords %lxPLED %d %d %d %d\n",
            //         canvas, x, mid+1, y1+22, mid+1, y1+22);

            char colorbuf[MAXPDSTRING];
            sprintf(colorbuf, "#%6.6x", x->x_gui.x_bcol);

125
126
            gui_vmess("gui_vumeter_update_peak", "xxsiiiiii",
                canvas, x, colorbuf,
127
                mid+1, y1+22, mid+1, y1+22, x1, y1);
Miller Puckette's avatar
Miller Puckette committed
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
        }
    }
}

static void vu_draw_update(t_gobj *client, t_glist *glist)
{
    t_vu *x = (t_vu *)client;
    if (x->x_updaterms)
    {
        vu_update_rms(x, glist);
        x->x_updaterms = 0;
    }
    if (x->x_updatepeak)
    {
        vu_update_peak(x, glist);
        x->x_updatepeak = 0;
    }
}
146

Miller Puckette's avatar
Miller Puckette committed
147
148
149
static void vu_draw_new(t_vu *x, t_glist *glist)
{
    t_canvas *canvas=glist_getcanvas(glist);
150
    int x1=text_xpix(&x->x_gui.x_obj, glist), x2=x1+x->x_gui.x_w;
151
    int y1=text_ypix(&x->x_gui.x_obj, glist);
152
153
    int w4=x->x_gui.x_w/4, mid=x1+x->x_gui.x_w/2, quad1=x1+w4+1;
    int quad3=x2-w4, end=x2+4;
Miller Puckette's avatar
Miller Puckette committed
154
    int k1=x->x_led_size+1, k2=IEM_VU_STEPS+1, k3=k1/2;
155
    int led_col, yyy, i, k4=y1-k3;
156
    iemgui_base_draw_new(&x->x_gui);
157

158
159
160
    for(i = 1; i <= IEM_VU_STEPS+1; i++)
    {
        yyy = k4 + k1 * (k2-i);
161
        if((i&3)==1 && (x->x_scale))
Jonathan Wilkes's avatar
Jonathan Wilkes committed
162
        {
163
164
165
166
167
            //sys_vgui(".x%lx.c create text %d %d -text {%s} -anchor w "
            //    "-font %s -fill #%6.6x "
            //    "-tags {%lxSCALEN %lxSCALE%d x%lx text iemgui}\n",
            //    canvas, end+1, yyy+k3+2, iemgui_vu_scale_str[i/4], 
            //    iemgui_font(&x->x_gui), x->x_gui.x_lcol, x, x, i, x);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
168
169
170
            char colorbuf[MAXPDSTRING];
            sprintf(colorbuf, "#%6.6x", x->x_gui.x_lcol);
            // not handling font size yet
171
172
            gui_vmess("gui_create_vumeter_text", "xxsiisiii",
                canvas, x,
173
                colorbuf, end+1, yyy+k3+2, iemgui_vu_scale_str[i/4],
Jonathan Wilkes's avatar
Jonathan Wilkes committed
174
175
                i, x1, y1);
        }
176
        led_col = iemgui_vu_col[i];
Jonathan Wilkes's avatar
Jonathan Wilkes committed
177
178
        if (i<=IEM_VU_STEPS)
        {
179
180
181
182
183
            //sys_vgui(".x%lx.c create polyline %d %d %d %d "
            //"-strokewidth %d -stroke #%6.6x "
            //"-tags {%lxRLED%d x%lx text iemgui}\n",
            //canvas, quad1+1, yyy+2, quad3, yyy+2,
            //x->x_led_size, iemgui_color_hex[led_col], x, i, x);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
184
185
            char colorbuf[MAXPDSTRING];
            sprintf(colorbuf, "#%6.6x", iemgui_color_hex[led_col]);
186
187
            gui_vmess("gui_create_vumeter_steps", "xxsiiiiiiiii",
                canvas, x, colorbuf, quad1+1,
188
                yyy+2, quad3, yyy+2, x->x_led_size, index, x1, y1, i);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
189
        }
190
    }
191
192
193
194
    //sys_vgui(".x%lx.c create prect %d %d %d %d -fill #%6.6x "
    //    "-stroke #%6.6x -tags {%lxRCOVER x%lx text iemgui}\n",
    //    canvas, quad1+1, y1+1, quad3, y1+1 + k1*IEM_VU_STEPS,
    //    x->x_gui.x_bcol, x->x_gui.x_bcol, x, x);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
195
196
    char colorbuf[MAXPDSTRING];
    sprintf(colorbuf, "#%6.6x", x->x_gui.x_bcol);
197
198
    gui_vmess("gui_create_vumeter_rect", "xxsiiiiii",
        canvas, x,
199
        colorbuf, quad1+1, y1+1, quad3, y1+1 + k1*IEM_VU_STEPS, x1, y1);
200
201
202
203
204
    //sys_vgui(".x%lx.c create polyline %d %d %d %d "
    //    "-strokewidth %d -fill #%6.6x "
    //    "-tags {%lxPLED x%lx text iemgui}\n",
    //    canvas, mid+1, y1+12,
    //    mid+1, y1+12, x->x_led_size, x->x_gui.x_bcol, x, x);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
205
    sprintf(colorbuf, "#%6.6x", x->x_gui.x_bcol);
206
207
    gui_vmess("gui_create_vumeter_peak", "xxsiiiiiii",
        canvas, x,
208
        colorbuf, mid+1, y1+12, mid+1, y1+12, x->x_led_size, x1, y1);
209
210
    x->x_updaterms = x->x_updatepeak = 1;
    sys_queuegui(x, x->x_gui.x_glist, vu_draw_update);
Miller Puckette's avatar
Miller Puckette committed
211
212
213
214
215
}

static void vu_draw_move(t_vu *x, t_glist *glist)
{
    t_canvas *canvas=glist_getcanvas(glist);
216
    if (!glist_isvisible(canvas)) return;
Miller Puckette's avatar
Miller Puckette committed
217

218
219
220
221
222
    int x1=text_xpix(&x->x_gui.x_obj, glist);
    int y1=text_ypix(&x->x_gui.x_obj, glist);
    int w4=x->x_gui.x_w/4, quad1=x1+w4+1;
    int quad3=x1+x->x_gui.x_w-w4,
          end=x1+x->x_gui.x_w+4;
223
    int k1=x->x_led_size+1, k2=IEM_VU_STEPS+1, k3=k1/2;
224
    int yyy, i, k4=y1-k3;
225
226
227

    //sys_vgui(".x%lx.c coords %lxBASE %d %d %d %d\n",
    //    canvas, x, x1, y1, x1+x->x_gui.x_w+2,y1+x->x_gui.x_h+4);
228
229
    gui_vmess("gui_vumeter_border_coords", "xxii",
        canvas, x, x->x_gui.x_w+2, x->x_gui.x_h+4);
230
    for(i=1; i<=IEM_VU_STEPS; i++)
231
    {
232
        yyy = k4 + k1*(k2-i);
233
234
        //sys_vgui(".x%lx.c coords %lxRLED%d %d %d %d %d\n",
        //    canvas, x, i, quad1+1, yyy+2, quad3, yyy+2);
235
236
        gui_vmess("gui_update_vumeter_step_coords", "xxiiiiiii",
            canvas, x, i, quad1+1, yyy+2, quad3, yyy+2,
237
238
239
240
241
242
            x1, y1);
//        if(((i+2)&3) && (x->x_scale))
        if((i&3)==1 && (x->x_scale))
        {
            //sys_vgui(".x%lx.c coords %lxSCALE%d %d %d\n",
            //    canvas, x, i, end+1, yyy+k3+2);
243
244
            gui_vmess("gui_vumeter_text_coords", "xxiiiii",
                canvas, x, i,
245
246
                end+1, yyy+k3+2, x1, y1);
        }
Miller Puckette's avatar
Miller Puckette committed
247
248
249
    }
    if(x->x_scale)
    {
250
251
        i=IEM_VU_STEPS+1;
        yyy = k4 + k1*(k2-i);
252
253
        //sys_vgui(".x%lx.c coords %lxSCALE%d %d %d\n",
        //    canvas, x, i, end+1, yyy+k3+2);
254
255
        gui_vmess("gui_vumeter_text_coords", "xxiiiii",
            canvas, x, i,
256
            end+1, yyy+k3+2, x1, y1);
Miller Puckette's avatar
Miller Puckette committed
257
    }
258
259
    x->x_updaterms = x->x_updatepeak = 1;
    sys_queuegui(x, glist, vu_draw_update);
Miller Puckette's avatar
Miller Puckette committed
260
261
}

262
static void vu_draw_config(t_vu* x, t_glist* glist)
Miller Puckette's avatar
Miller Puckette committed
263
{
264
    int i;
Miller Puckette's avatar
Miller Puckette committed
265
    t_canvas *canvas=glist_getcanvas(glist);
Mathieu L Bouchard's avatar
Mathieu L Bouchard committed
266
    char lcol[8]; sprintf(lcol, "#%6.6x", x->x_gui.x_lcol);
267
    for(i = 1; i <= IEM_VU_STEPS+1; i++)
268
    {
269
        if (i <= IEM_VU_STEPS)
270
271
272
        {
            //sys_vgui(".x%lx.c itemconfigure %lxRLED%d -strokewidth %d\n",
            //    canvas, x, i, x->x_led_size);
273
274
            gui_vmess("gui_update_vumeter_steps", "xxii",
                canvas, x, i, x->x_led_size);
275
276
277
278
279
280
281
282
283
284
285
        }
        //if((i&3)==1)
        if((i&3)==1)
        {
            //sys_vgui(".x%lx.c itemconfigure %lxSCALE%d -text {%s} "
            //"-font %s -fill %s\n", canvas, x, i, iemgui_vu_scale_str[i/4],
            //iemgui_font(&x->x_gui), x->x_gui.x_selected == canvas &&
            //x->x_gui.x_glist == canvas && x->x_scale ? selection_color : lcol);

            int isselected = x->x_gui.x_selected == canvas &&
                x->x_gui.x_glist == canvas && x->x_scale;
286
287
            gui_vmess("gui_update_vumeter_text", "xxssisi",
                canvas, x, iemgui_vu_scale_str[i/4],
288
289
                iemgui_font(&x->x_gui), isselected, lcol, i);
        }
290
    }
291
292
293
294
295
296
    //sys_vgui(".x%lx.c itemconfigure %lxRCOVER -fill #%6.6x -stroke #%6.6x\n",
    //         canvas, x, x->x_gui.x_bcol, x->x_gui.x_bcol);
    //sys_vgui(".x%lx.c itemconfigure %lxPLED -strokewidth %d\n",
    //         canvas, x, x->x_led_size);
    char bcol[8];
    sprintf(bcol, "#%6.6x", x->x_gui.x_bcol);
297
298
299
300
    gui_vmess("gui_update_vumeter_rect", "xxs",
        canvas, x, bcol);
    gui_vmess("gui_update_vumeter_peak", "xxi",
        canvas, x, x->x_led_size);
301
    iemgui_base_draw_config(&x->x_gui);
Miller Puckette's avatar
Miller Puckette committed
302
303
}

304
/* Not sure if this is necessary anymore */
Miller Puckette's avatar
Miller Puckette committed
305
306
307
static void vu_draw_select(t_vu* x,t_glist* glist)
{
    t_canvas *canvas=glist_getcanvas(glist);
308
    char lcol[8]; sprintf(lcol,"#%6.6x", x->x_gui.x_lcol);
309
    int issel = x->x_gui.x_selected == canvas && x->x_gui.x_glist == canvas;
310
311
    sys_vgui(".x%lx.c itemconfigure %lxSCALEN -fill %s\n",
        canvas, x, issel ? selection_color : lcol);
312
313
}

314
static void vu__clickhook(t_scalehandle *sh, int newstate)
315
{
316
    t_vu *x = (t_vu *)(sh->h_master);
317
    if (newstate)
Jonathan Wilkes's avatar
Jonathan Wilkes committed
318
319
    {
        canvas_apply_setundo(x->x_gui.x_glist, (t_gobj *)x);
320
321
        if (!sh->h_scale)
            scalehandle_click_label(sh);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
322
    }
323
    sh->h_dragon = newstate;
324
325
}

Jonathan Wilkes's avatar
Jonathan Wilkes committed
326
static void vu__motionhook(t_scalehandle *sh, t_floatarg mouse_x, t_floatarg mouse_y)
327
{
Jonathan Wilkes's avatar
Jonathan Wilkes committed
328
    if (sh->h_scale)
329
    {
330
        t_vu *x = (t_vu *)(sh->h_master);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
331
332
        int dx = (int)(mouse_x - sh->h_offset_x),
            dy = (int)(mouse_y - sh->h_offset_y);
333
        dx = maxi(dx, 8-x->x_gui.x_w);
Jonathan Wilkes's avatar
Jonathan Wilkes committed
334
        dy = maxi(dy,80-x->x_gui.x_h);
335
336
        sh->h_dragx = dx;
        sh->h_dragy = dy;
337
        scalehandle_drag_scale(sh);
338

Jonathan Wilkes's avatar
Jonathan Wilkes committed
339
340
341
342
343
344
345
346
347
348
349
350
351
        x->x_gui.x_w += dx;
        x->x_gui.x_h += dy;

        if (glist_isvisible(x->x_gui.x_glist))
        {
            /* draw_move doesn't seem to cut it for vu, so we
               just toggle visibility */
            //vu_draw_move(x, x->x_gui.x_glist);
            gobj_vis((t_gobj *)x, x->x_gui.x_glist, 0);
            gobj_vis((t_gobj *)x, x->x_gui.x_glist, 1);
            scalehandle_unclick_scale(sh);
        }

352
353
354
        int properties = gfxstub_haveproperties((void *)x);
        if (properties)
        {
355
356
            int new_w = x->x_gui.x_w + sh->h_dragx;
            int new_h = x->x_gui.x_h + sh->h_dragy;
357
358
            properties_set_field_int(properties,"dim.w_ent",new_w);
            properties_set_field_int(properties,"dim.h_ent",new_h);
359
        }
360
    }
361
    scalehandle_dragon_label(sh,mouse_x, mouse_y);
Miller Puckette's avatar
Miller Puckette committed
362
363
364
365
}

void vu_draw(t_vu *x, t_glist *glist, int mode)
{
366
367
368
369
370
    if(mode == IEM_GUI_DRAW_MODE_UPDATE)      sys_queuegui((t_gobj*)x, x->x_gui.x_glist, vu_draw_update);
    if(mode == IEM_GUI_DRAW_MODE_MOVE)        vu_draw_move(x, glist);
    else if(mode == IEM_GUI_DRAW_MODE_NEW)    vu_draw_new(x, glist);
    else if(mode == IEM_GUI_DRAW_MODE_SELECT) vu_draw_select(x, glist);
    else if(mode == IEM_GUI_DRAW_MODE_CONFIG) vu_draw_config(x, glist);
Miller Puckette's avatar
Miller Puckette committed
371
372
373
374
}

/* ------------------------ vu widgetbehaviour----------------------------- */

375
376
static void vu_scale_getrect(t_iemgui x_gui, t_glist *x, int *xp1, int *yp1,
    int *xp2, int *yp2, int scale_x, int scale_y)
377
{
378
379
380
381
382
383
384
385
386
387
388
    t_float width_multiplier;
    int scale_length;    
    int scale_x1;
    int scale_y1;
    int scale_x2;
    int scale_y2;
    int actual_fontsize; //seems tk does its own thing when it comes to rendering
    int actual_height;

    if (x->gl_isgraph && !glist_istoplevel(x))
    {
389
        switch(x_gui.x_font_style)
390
        {
391
392
393
            case 1:   width_multiplier = 0.83333; break;
            case 2:   width_multiplier = 0.735;   break;
            default:  width_multiplier = 1.0;     break;
394
        }
395
        actual_fontsize = x_gui.x_fontsize;
396
397
        actual_height = actual_fontsize;
        //exceptions
398
        if (x_gui.x_font_style == 0 &&
399
400
401
402
403
404
405
            (actual_fontsize == 8 || actual_fontsize == 13 ||
             actual_fontsize % 10 == 1 || actual_fontsize % 10 == 6 ||
             (actual_fontsize > 48 && actual_fontsize < 100 &&
              (actual_fontsize %10 == 4 || actual_fontsize %10 == 9))) )
        {
            actual_fontsize += 1;
        }
406
        else if (x_gui.x_font_style == 1 &&
407
408
409
410
411
412
            actual_fontsize >= 5 &&
            actual_fontsize < 13 &&
            actual_fontsize % 2 == 1)
        {
            actual_fontsize += 1;
        }
413
        else if (x_gui.x_font_style == 2 &&
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
            actual_fontsize >= 5 &&
            actual_fontsize % 2 == 1)
        {
            actual_fontsize += 1;
        }
        if (actual_height == 9)
            actual_height += 1;
        //done with exceptions

        width_multiplier = width_multiplier * (actual_fontsize * 0.6);

        scale_length = 4;
        scale_x1 = scale_x;
        scale_y1 = scale_y - actual_height/2;
        scale_x2 = scale_x1 + (scale_length * width_multiplier);
        scale_y2 = scale_y1 + actual_height*1.1;

        if (scale_x1 < *xp1) *xp1 = scale_x1;
        if (scale_x2 > *xp2) *xp2 = scale_x2;
        if (scale_y1 < *yp1) *yp1 = scale_y1;
        if (scale_y2 > *yp2) *yp2 = scale_y2;
    }
436
}
Miller Puckette's avatar
Miller Puckette committed
437
438

static void vu_getrect(t_gobj *z, t_glist *glist,
439
    int *xp1, int *yp1, int *xp2, int *yp2)
Miller Puckette's avatar
Miller Puckette committed
440
441
442
{
    t_vu* x = (t_vu*)z;

443
    int yyy, end;
444
445
446
    int x1=text_xpix(&x->x_gui.x_obj, glist);
    int y1=text_ypix(&x->x_gui.x_obj, glist);
    int k1=x->x_led_size+1, k2=IEM_VU_STEPS+1, k3=k1/2, k4=y1-k3;
447

448
    *xp1 = text_xpix(&x->x_gui.x_obj, glist);
449
    *yp1 = text_ypix(&x->x_gui.x_obj, glist);
450
    *xp2 = *xp1 + x->x_gui.x_w + 2;
451
    *yp2 = *yp1 + x->x_gui.x_h + 4;
452

453
    iemgui_label_getrect(x->x_gui, glist, xp1, yp1, xp2, yp2);
454

455
456
    /* In legacy mode we don't include the scale in the rect */
    if (x->x_scale && !sys_legacy)
457
458
    {
        //vu has custom scale all labels unlike other iemgui object
459
        end=x1+x->x_gui.x_w+4;
460
461
462
463
464
        yyy = k4 + k1*(k2-1);
        vu_scale_getrect(x->x_gui, glist, xp1, yp1, xp2, yp2, end+1, yyy+k3+2);
        yyy = k4;
        vu_scale_getrect(x->x_gui, glist, xp1, yp1, xp2, yp2, end+1, yyy+k3+2);
    }
Miller Puckette's avatar
Miller Puckette committed
465
466
467
468
469
470
471
472
}

static void vu_save(t_gobj *z, t_binbuf *b)
{
    t_vu *x = (t_vu *)z;
    int bflcol[3];
    t_symbol *srl[3];
    iemgui_save(&x->x_gui, srl, bflcol);
473
474
475
476
477
478
479
    binbuf_addv(b, "ssiisiissiiiiiiii;", gensym("#X"),gensym("obj"),
        (int)x->x_gui.x_obj.te_xpix, (int)x->x_gui.x_obj.te_ypix,
        gensym("vu"), x->x_gui.x_w, x->x_gui.x_h,
        srl[1], srl[2], x->x_gui.x_ldx, x->x_gui.x_ldy,
        iem_fstyletoint(&x->x_gui), x->x_gui.x_fontsize,
        bflcol[0], bflcol[2], x->x_scale,
        iem_symargstoint(&x->x_gui));
Miller Puckette's avatar
Miller Puckette committed
480
481
482
483
}

void vu_check_height(t_vu *x, int h)
{
484
    int n = maxi(h / IEM_VU_STEPS, IEM_VU_MINSIZE);
Miller Puckette's avatar
Miller Puckette committed
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
    x->x_led_size = n-1;
    x->x_gui.x_h = IEM_VU_STEPS * n;
}

static void vu_scale(t_vu *x, t_floatarg fscale)
{
    int i, scale = (int)fscale;

    if(scale != 0) scale = 1;
    if(x->x_scale && !scale)
    {
        t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist);

        x->x_scale = (int)scale;
        if(glist_isvisible(x->x_gui.x_glist))
        {
            for(i=1; i<=IEM_VU_STEPS; i++)
            {
503
504
505
506
                /* if((i+2)&3) */
                if((i&3)==1)
                {
                    //sys_vgui(".x%lx.c delete %lxSCALE%d\n", canvas, x, i);
507
508
                    gui_vmess("gui_erase_vumeter_text", "xxi",
                        canvas, x, i);
509
                }
Miller Puckette's avatar
Miller Puckette committed
510
511
            }
            i=IEM_VU_STEPS+1;
512
            //sys_vgui(".x%lx.c delete %lxSCALE%d\n", canvas, x, i);
513
514
            gui_vmess("gui_erase_vumeter_text", "xxi",
                canvas, x, i);
Miller Puckette's avatar
Miller Puckette committed
515
516
517
518
        }
    }
    if(!x->x_scale && scale)
    {
Mathieu L Bouchard's avatar
Mathieu L Bouchard committed
519
        int end=text_xpix(&x->x_gui.x_obj,
520
            x->x_gui.x_glist)+x->x_gui.x_w+4;
Miller Puckette's avatar
Miller Puckette committed
521
522
523
524
525
526
527
528
529
530
        int k1=x->x_led_size+1, k2=IEM_VU_STEPS+1, k3=k1/2;
        int yyy, k4=text_ypix(&x->x_gui.x_obj, x->x_gui.x_glist)-k3;
        t_canvas *canvas=glist_getcanvas(x->x_gui.x_glist);

        x->x_scale = (int)scale;
        if(glist_isvisible(x->x_gui.x_glist))
        {
            for(i=1; i<=IEM_VU_STEPS; i++)
            {
                yyy = k4 + k1*(k2-i);
531
                if((i&3)==1)
532
533
534
535
536
537
538
                {
                    //sys_vgui(".x%lx.c create text %d %d -text {%s} -anchor w "
                    //         "-font %s -fill #%6.6x -tags {%lxSCALE%d x%lx}\n",
                    //    canvas, end+1, yyy+k3+2, iemgui_vu_scale_str[i/4], 
                    //    iemgui_font(&x->x_gui), x->x_gui.x_lcol, x, i, x);
                    char colorbuf[MAXPDSTRING];
                    sprintf(colorbuf, "#%6.6x", x->x_gui.x_lcol);
539
540
                    gui_vmess("gui_create_vumeter_text", "xxsiisiii",
                        canvas, x, colorbuf,
541
542
543
                        end+1, yyy+k3+2, iemgui_vu_scale_str[i/4],
                        i, end, yyy);
                }
Miller Puckette's avatar
Miller Puckette committed
544
            }
545
            i = IEM_VU_STEPS + 1;
Miller Puckette's avatar
Miller Puckette committed
546
            yyy = k4 + k1*(k2-i);
547
548
549
550
551
552
            //sys_vgui(".x%lx.c create text %d %d -text {%s} -anchor w "
            //         "-font %s -fill #%6.6x -tags {%lxSCALE%d x%lx}\n",
            //    canvas, end+1, yyy+k3+2, iemgui_vu_scale_str[i/4], 
            //    iemgui_font(&x->x_gui), x->x_gui.x_lcol, x, i, x);
            char colorbuf[MAXPDSTRING];
            sprintf(colorbuf, "#%6.6x", x->x_gui.x_lcol);
553
554
            gui_vmess("gui_create_vumeter_text", "xxsiisiii",
                canvas, x, colorbuf,
555
556
                end+1, yyy+k3+2, iemgui_vu_scale_str[i/4],
                i, end, yyy);
Miller Puckette's avatar
Miller Puckette committed
557
558
559
560
561
562
563
        }
    }
}

static void vu_properties(t_gobj *z, t_glist *owner)
{
    t_vu *x = (t_vu *)z;
564
    char buf[800], *gfx_tag;
Miller Puckette's avatar
Miller Puckette committed
565
566
567
    t_symbol *srl[3];

    iemgui_properties(&x->x_gui, srl);
Miller Puckette's avatar
Miller Puckette committed
568
    sprintf(buf, "pdtk_iemgui_dialog %%s |vu| \
569
570
571
572
573
574
575
576
577
578
579
580
        --------dimensions(pix)(pix):-------- %d %d width: %d %d height: \
        empty 0.0 empty 0.0 empty %d %d no_scale scale %d %d empty %d \
        {%s} {%s} {%s} %d %d %d %d %d %d %d\n",
        x->x_gui.x_w, IEM_GUI_MINSIZE, x->x_gui.x_h,
        IEM_VU_STEPS*IEM_VU_MINSIZE,
        0,/*no_schedule*/
        x->x_scale, -1, -1, -1,/*no linlog, no init, no multi*/
        "nosndno", srl[1]->s_name,/*no send*/
        srl[2]->s_name, x->x_gui.x_ldx, x->x_gui.x_ldy,
        x->x_gui.x_font_style, x->x_gui.x_fontsize,
        0xffffff & x->x_gui.x_bcol, -1/*no front-color*/,
        0xffffff & x->x_gui.x_lcol);
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
    //gfxstub_new(&x->x_gui.x_obj.ob_pd, x, buf);
    gfx_tag = gfxstub_new2(&x->x_gui.x_obj.ob_pd, x);

    gui_start_vmess("gui_iemgui_dialog", "s", gfx_tag);

    gui_start_array();

    gui_s("type");
    gui_s("vu");

    gui_s("width"); gui_i(x->x_gui.x_w);
    gui_s("height"); gui_i(x->x_gui.x_h);
    
    gui_s("vu-scale"); gui_i(x->x_scale);

    gui_s("minimum-size"); gui_i(IEM_GUI_MINSIZE);
    
    gui_s("range-schedule"); // no idea what this is...
    gui_i(2);

    gui_s("receive-symbol");   gui_s(srl[1]->s_name);
    gui_s("label");            gui_s(srl[2]->s_name);
    gui_s("x-offset");         gui_i(x->x_gui.x_ldx);
    gui_s("y-offset");         gui_i(x->x_gui.x_ldy);
    gui_s("font-style");       gui_i(x->x_gui.x_font_style);
    gui_s("font-size");        gui_i(x->x_gui.x_fontsize);
    gui_s("background-color"); gui_i(0xffffff & x->x_gui.x_bcol);
    gui_s("label-color");      gui_i(0xffffff & x->x_gui.x_lcol);
    
    gui_end_array();
    gui_end_vmess();
Miller Puckette's avatar
Miller Puckette committed
612
613
614
615
}

static void vu_dialog(t_vu *x, t_symbol *s, int argc, t_atom *argv)
{
616
    canvas_apply_setundo(x->x_gui.x_glist, (t_gobj *)x);
617
618
619
    int w = atom_getintarg(0, argc, argv);
    int h = atom_getintarg(1, argc, argv);
    int scale = !!atom_getintarg(4, argc, argv);
Mathieu L Bouchard's avatar
Mathieu L Bouchard committed
620
    int sr_flags = iemgui_dialog(&x->x_gui, argc, argv);
621
    x->x_gui.x_loadinit = 0;
Miller Puckette's avatar
Miller Puckette committed
622
623
    x->x_gui.x_w = iemgui_clip_size(w);
    vu_check_height(x, h);
624
    vu_scale(x, (t_float)scale);
625
626
    iemgui_draw_config(&x->x_gui);
    iemgui_draw_io(&x->x_gui, sr_flags);
627
    iemgui_shouldvis(&x->x_gui, IEM_GUI_DRAW_MODE_MOVE);
628
    scalehandle_draw(&x->x_gui);
629
    scrollbar_update(x->x_gui.x_glist);
Miller Puckette's avatar
Miller Puckette committed
630
631
632
633
634
635
636
637
638
}

static void vu_size(t_vu *x, t_symbol *s, int ac, t_atom *av)
{
    x->x_gui.x_w = iemgui_clip_size((int)atom_getintarg(0, ac, av));
    if(ac > 1)
        vu_check_height(x, (int)atom_getintarg(1, ac, av));
    if(glist_isvisible(x->x_gui.x_glist))
    {
639
640
        iemgui_draw_move(&x->x_gui);
        iemgui_draw_config(&x->x_gui);
Miller Puckette's avatar
Miller Puckette committed
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
        canvas_fixlinesfor(glist_getcanvas(x->x_gui.x_glist), (t_text*)x);
    }
}

static void vu_float(t_vu *x, t_floatarg rms)
{
    int i;

    if(rms <= IEM_VU_MINDB)
        x->x_rms = 0;
    else if(rms >= IEM_VU_MAXDB)
        x->x_rms = IEM_VU_STEPS;
    else
    {
        int i = (int)(2.0*(rms + IEM_VU_OFFSET));
        x->x_rms = iemgui_vu_db2i[i];
    }
    i = (int)(100.0*rms + 10000.5);
659
    rms = 0.01*(t_float)(i - 10000);
Miller Puckette's avatar
Miller Puckette committed
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
    x->x_fr = rms;
    outlet_float(x->x_out_rms, rms);
    x->x_updaterms = 1;
    sys_queuegui(x, x->x_gui.x_glist, vu_draw_update);
}

static void vu_ft1(t_vu *x, t_floatarg peak)
{
    int i;

    if(peak <= IEM_VU_MINDB)
        x->x_peak = 0;
    else if(peak >= IEM_VU_MAXDB)
        x->x_peak = IEM_VU_STEPS;
    else
    {
        int i = (int)(2.0*(peak + IEM_VU_OFFSET));
        x->x_peak = iemgui_vu_db2i[i];
    }
    i = (int)(100.0*peak + 10000.5);
680
    peak = 0.01*(t_float)(i - 10000);
Miller Puckette's avatar
Miller Puckette committed
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
    x->x_fp = peak;
    x->x_updatepeak = 1;
    sys_queuegui(x, x->x_gui.x_glist, vu_draw_update);
    outlet_float(x->x_out_peak, peak);
}

static void vu_bang(t_vu *x)
{
    outlet_float(x->x_out_peak, x->x_fp);
    outlet_float(x->x_out_rms, x->x_fr);
    x->x_updaterms = x->x_updatepeak = 1;
    sys_queuegui(x, x->x_gui.x_glist, vu_draw_update);
}

static void *vu_new(t_symbol *s, int argc, t_atom *argv)
{
    t_vu *x = (t_vu *)pd_new(vu_class);
698
699
    int bflcol[] = {-66577, -1, -1};
    int w = IEM_GUI_DEFAULTSIZE, h = IEM_VU_STEPS*IEM_VU_DEFAULTSIZE;
Mathieu L Bouchard's avatar
Mathieu L Bouchard committed
700
    int ldx = -1, ldy = -8, fs = 10, scale = 1;
Miller Puckette's avatar
Miller Puckette committed
701

702
703
    iem_inttosymargs(&x->x_gui, 0);
    iem_inttofstyle(&x->x_gui, 0);
Miller Puckette's avatar
Miller Puckette committed
704
705
706
707
708
709
710
711

    if((argc >= 11)&&IS_A_FLOAT(argv,0)&&IS_A_FLOAT(argv,1)
       &&(IS_A_SYMBOL(argv,2)||IS_A_FLOAT(argv,2))
       &&(IS_A_SYMBOL(argv,3)||IS_A_FLOAT(argv,3))
       &&IS_A_FLOAT(argv,4)&&IS_A_FLOAT(argv,5)
       &&IS_A_FLOAT(argv,6)&&IS_A_FLOAT(argv,7)
       &&IS_A_FLOAT(argv,8)&&IS_A_FLOAT(argv,9)&&IS_A_FLOAT(argv,10))
    {
712
713
        w = atom_getintarg(0, argc, argv);
        h = atom_getintarg(1, argc, argv);
Miller Puckette's avatar
Miller Puckette committed
714
        iemgui_new_getnames(&x->x_gui, 1, argv);
715
716
        ldx = atom_getintarg(4, argc, argv);
        ldy = atom_getintarg(5, argc, argv);
717
        iem_inttofstyle(&x->x_gui, atom_getintarg(6, argc, argv));
718
719
720
721
        fs = maxi(atom_getintarg(7, argc, argv),4);
        bflcol[0] = atom_getintarg(8, argc, argv);
        bflcol[2] = atom_getintarg(9, argc, argv);
        scale = !!atom_getintarg(10, argc, argv);
Miller Puckette's avatar
Miller Puckette committed
722
723
724
    }
    else iemgui_new_getnames(&x->x_gui, 1, 0);
    if((argc == 12)&&IS_A_FLOAT(argv,11))
725
        iem_inttosymargs(&x->x_gui, atom_getintarg(11, argc, argv));
Miller Puckette's avatar
Miller Puckette committed
726
727
728
    x->x_gui.x_draw = (t_iemfunptr)vu_draw;

    x->x_gui.x_glist = (t_glist *)canvas_getcurrent();
729
730
    if (x->x_gui.x_font_style<0 || x->x_gui.x_font_style>2) x->x_gui.x_font_style=0;
    if(iemgui_has_rcv(&x->x_gui))
Miller Puckette's avatar
Miller Puckette committed
731
732
733
        pd_bind(&x->x_gui.x_obj.ob_pd, x->x_gui.x_rcv);
    x->x_gui.x_ldx = ldx;
    x->x_gui.x_ldy = ldy;
734
    x->x_gui.x_snd = s_empty;
Miller Puckette's avatar
Miller Puckette committed
735
736
737
738
739
740
741
742
743
    x->x_gui.x_fontsize = fs;
    x->x_gui.x_w = iemgui_clip_size(w);
    vu_check_height(x, h);
    iemgui_all_colfromload(&x->x_gui, bflcol);
    x->x_scale = scale;
    x->x_peak = 0;
    x->x_rms = 0;
    x->x_fp = -101.0;
    x->x_fr = -101.0;
Mathieu L Bouchard's avatar
Mathieu L Bouchard committed
744
    iemgui_verify_snd_ne_rcv(&x->x_gui); // makes no sense, because snd is unused
Miller Puckette's avatar
Miller Puckette committed
745
746
747
    inlet_new(&x->x_gui.x_obj, &x->x_gui.x_obj.ob_pd, &s_float, gensym("ft1"));
    x->x_out_rms = outlet_new(&x->x_gui.x_obj, &s_float);
    x->x_out_peak = outlet_new(&x->x_gui.x_obj, &s_float);
748

749
    x->x_gui.x_handle = scalehandle_new((t_object *)x,x->x_gui.x_glist,1,vu__clickhook,vu__motionhook);
750
    x->x_gui.x_lhandle = scalehandle_new((t_object *)x,x->x_gui.x_glist,0,vu__clickhook,vu__motionhook);
751
    x->x_gui.x_obj.te_iemgui = 1;
752

753
    x->x_gui.legacy_x = -1;
754
    x->x_gui.legacy_y = -1;
755

Miller Puckette's avatar
Miller Puckette committed
756
757
758
759
760
    return (x);
}

static void vu_free(t_vu *x)
{
761
    if(iemgui_has_rcv(&x->x_gui))
Miller Puckette's avatar
Miller Puckette committed
762
763
        pd_unbind(&x->x_gui.x_obj.ob_pd, x->x_gui.x_rcv);
    gfxstub_deleteforkey(x);
764

765
766
    if (x->x_gui. x_handle) scalehandle_free(x->x_gui. x_handle);
    if (x->x_gui.x_lhandle) scalehandle_free(x->x_gui.x_lhandle);
Miller Puckette's avatar
Miller Puckette committed
767
768
769
770
771
}

void g_vumeter_setup(void)
{
    vu_class = class_new(gensym("vu"), (t_newmethod)vu_new, (t_method)vu_free,
772
        sizeof(t_vu), 0, A_GIMME, 0);
Miller Puckette's avatar
Miller Puckette committed
773
774
775
776
    class_addbang(vu_class,vu_bang);
    class_addfloat(vu_class,vu_float);
    class_addmethod(vu_class, (t_method)vu_ft1, gensym("ft1"), A_FLOAT, 0);
    class_addmethod(vu_class, (t_method)vu_dialog, gensym("dialog"),
777
        A_GIMME, 0);
Miller Puckette's avatar
Miller Puckette committed
778
    class_addmethod(vu_class, (t_method)vu_size, gensym("size"), A_GIMME, 0);
779
780
    class_addmethod(vu_class, (t_method)vu_scale,
        gensym("scale"), A_DEFFLOAT, 0);
781
    iemgui_class_addmethods(vu_class);
782

783
    wb_init(&vu_widgetbehavior,vu_getrect,0);
Miller Puckette's avatar
Miller Puckette committed
784
785
786
787
788
    class_setwidget(vu_class,&vu_widgetbehavior);
    class_sethelpsymbol(vu_class, gensym("vu"));
    class_setsavefn(vu_class, vu_save);
    class_setpropertiesfn(vu_class, vu_properties);
}