Mercurial > audlegacy-plugins
view src/paranormal/wave.c @ 192:a7c823478180 trunk
[svn] - add fast linedrawing code
author | nenolod |
---|---|
date | Thu, 02 Nov 2006 22:19:05 -0800 |
parents | 0d826917c56f |
children | 4b48e6e9b3cb |
line wrap: on
line source
#ifdef HAVE_CONFIG_H # include <config.h> #endif #include "paranormal.h" #include "actuators.h" #include "pn_utils.h" /* **************** wave_horizontal **************** */ struct pn_actuator_option_desc wave_horizontal_opts[] = { {"channels", "Which sound channels to use: negative = channel 1, \npositive = channel 2, " "zero = both (two wave-forms.)", OPT_TYPE_INT, {ival: -1} }, {"value", "The colour value to use.", OPT_TYPE_INT, {ival: 255} }, { NULL } }; static void wave_horizontal_exec (const struct pn_actuator_option *opts, gpointer data) { int i; int channel = ( opts[0].val.ival < 0 ) ? 0 : 1; guchar value = (opts[1].val.ival < 0 || opts[1].val.ival > 255) ? 255 : opts[1].val.ival; for (i=0; i<pn_image_data->width; i++) { /*single channel, centered horz.*/ if ( opts[0].val.ival ) { pn_image_data->surface[0][PN_IMG_INDEX (i, (pn_image_data->height>>1) - CAP (pn_sound_data->pcm_data[channel] [i*512/pn_image_data->width]>>8, (pn_image_data->height>>1)-1))] = value; } /*both channels, at 1/4 and 3/4 of the screen*/ else { pn_image_data->surface[0][PN_IMG_INDEX( i, (pn_image_data->height>>2) - CAP( (pn_sound_data->pcm_data[0] [i*512/pn_image_data->width]>>9), (pn_image_data->height>>2)-1))] = value; pn_image_data->surface[0][PN_IMG_INDEX( i, 3*(pn_image_data->height>>2) - CAP( (pn_sound_data->pcm_data[1] [i*512/pn_image_data->width]>>9), (pn_image_data->height>>2)-1))] = value; } } } struct pn_actuator_desc builtin_wave_horizontal = { "wave_horizontal", "Horizontal Waveform", "Draws one or two waveforms horizontally across " "the drawing surface", 0, wave_horizontal_opts, NULL, NULL, wave_horizontal_exec }; /* **************** wave_vertical **************** */ struct pn_actuator_option_desc wave_vertical_opts[] = { {"channels", "Which sound channels to use: negative = channel 1, \npositive = channel 2, " "zero = both (two wave-forms.)", OPT_TYPE_INT, {ival: -1} }, {"value", "The colour value to use.", OPT_TYPE_INT, {ival: 255} }, { NULL } }; static void wave_vertical_exec (const struct pn_actuator_option *opts, gpointer data) { int i; int channel = ( opts[0].val.ival < 0 ) ? 0 : 1; guchar value = (opts[1].val.ival < 0 || opts[1].val.ival > 255) ? 255 : opts[1].val.ival; for (i=0; i<pn_image_data->height; i++) { if ( opts[0].val.ival ) { pn_image_data->surface[0][PN_IMG_INDEX ((pn_image_data->width>>1) - CAP (pn_sound_data->pcm_data[channel] [i*512/pn_image_data->height]>>8, (pn_image_data->width>>1)-1), i)] = value; } else { pn_image_data->surface[0][PN_IMG_INDEX ((pn_image_data->width>>2) - CAP (pn_sound_data->pcm_data[0] [i*512/pn_image_data->height]>>9, (pn_image_data->width>>2)-1), i)] = value; pn_image_data->surface[0][PN_IMG_INDEX ((3*pn_image_data->width>>2) -CAP (pn_sound_data->pcm_data[1] [i*512/pn_image_data->height]>>9, (pn_image_data->width>>2)-1), i)] = value; } } } struct pn_actuator_desc builtin_wave_vertical = { "wave_vertical", "Vertical Waveform", "Draws one or two waveforms vertically across " "the drawing surface", 0, wave_vertical_opts, NULL, NULL, wave_vertical_exec }; /* FIXME: allow for only 1 channel for wave_normalize & wave_smooth */ /* **************** wave_normalize **************** */ static struct pn_actuator_option_desc wave_normalize_opts[] = { { "height", "If positive, the height, in pixels, to which the waveform will be " "normalized; if negative, hfrac is used", OPT_TYPE_INT, { ival: -1 } }, { "hfrac", "If positive, the fraction of the horizontal image size to which the " "waveform will be normalized; if negative, vfrac is used", OPT_TYPE_FLOAT, { fval: -1 } }, { "vfrac", "If positive, the fraction of the vertical image size to which the " "waveform will be normalized", OPT_TYPE_FLOAT, { fval: .125 } }, { "channels", "Which sound channel(s) to normalize: negative = channel 1,\n" "\tpositive = channel 2, 0 = both channels.", OPT_TYPE_INT, { ival: 0 } }, { NULL } }; static void wave_normalize_exec (const struct pn_actuator_option *opts, gpointer data) { int i, j, max=0; float denom; for (j=0; j<2; j++) { if ( !(opts[3].val.ival) || (opts[3].val.ival < 0 && j == 0) || (opts[3].val.ival > 0 && j == 1) ) { for (i=0; i<512; i++) if (abs(pn_sound_data->pcm_data[j][i]) > max) max = abs(pn_sound_data->pcm_data[j][i]); if (opts[0].val.ival > 0) denom = max/(opts[0].val.ival<<8); else if (opts[1].val.fval > 0) denom = max/(opts[1].val.fval * (pn_image_data->width<<8)); else denom = max/(opts[2].val.fval * (pn_image_data->height<<8)); if (denom > 0) for (i=0; i<512; i++) pn_sound_data->pcm_data[j][i] /= denom; } } } struct pn_actuator_desc builtin_wave_normalize = { "wave_normalize", "Normalize Waveform Data", "Normalizes the waveform data used by the wave_* actuators", 0, wave_normalize_opts, NULL, NULL, wave_normalize_exec }; /* **************** wave_smooth **************** */ struct pn_actuator_option_desc wave_smooth_opts[] = { { "channels", "Which sound channel(s) to smooth: negative = channel 1, \n" "\tpositive = channel 2, 0 = both channels.", OPT_TYPE_INT, { ival: 0 } }, {0} }; static void wave_smooth_exec (const struct pn_actuator_option *opts, gpointer data) { int i, j, k; gint16 tmp[512]; for (j=0; j<2; j++) { if ( !(opts[0].val.ival) || (opts[0].val.ival < 0 && j == 0) || (opts[0].val.ival > 0 && j == 1) ) { for (i=4; i<508; i++) { k = (pn_sound_data->pcm_data[j][i]<<3) + (pn_sound_data->pcm_data[j][i+1]<<2) + (pn_sound_data->pcm_data[j][i-1]<<2) + (pn_sound_data->pcm_data[j][i+2]<<2) + (pn_sound_data->pcm_data[j][i-2]<<2) + (pn_sound_data->pcm_data[j][i+3]<<1) + (pn_sound_data->pcm_data[j][i-3]<<1) + (pn_sound_data->pcm_data[j][i+4]<<1) + (pn_sound_data->pcm_data[j][i-4]<<1); tmp[i] = k >> 5; } memcpy (pn_sound_data->pcm_data[j]+4, tmp, sizeof (gint16) * 504); } } } struct pn_actuator_desc builtin_wave_smooth = { "wave_smooth", "Smooth Waveform Data", "Smooth out the waveform data used by the wave_* actuators", 0, wave_smooth_opts, NULL, NULL, wave_smooth_exec }; /* **************** wave_radial **************** */ static struct pn_actuator_option_desc wave_radial_opts[] = { { "base_radius", " ", OPT_TYPE_FLOAT, { fval: 0 } }, {"value", "The colour value to use.", OPT_TYPE_INT, {ival: 255} }, { NULL } }; static void wave_radial_exec (const struct pn_actuator_option *opts, gpointer data) { int i, x, y; guchar value = (opts[1].val.ival < 0 || opts[1].val.ival > 255) ? 255 : opts[1].val.ival; for(i=0; i<360; i++) { x = (pn_image_data->width>>1) + (opts[0].val.fval + (pn_sound_data->pcm_data[0][(int)(i*(512.0/360.0))]>>8)) * cos_val[i]; y = (pn_image_data->height>>1) + (opts[0].val.fval + (pn_sound_data->pcm_data[0][(int)(i*(512.0/360.0))]>>8)) * sin_val[i]; pn_image_data->surface[0][PN_IMG_INDEX (CAPHILO(x,pn_image_data->width,0), CAPHILO(y,pn_image_data->height,0))] = value; } }; struct pn_actuator_desc builtin_wave_radial = { "wave_radial", "Radial Waveform", "Draws a single waveform varying" " radially from the center of the image", 0, wave_radial_opts, NULL, NULL, wave_radial_exec };