Mercurial > audlegacy
view src/audacious/widgets/svis.c @ 2883:4117f05f7054 trunk
branch merge
author | William Pitcock <nenolod@atheme.org> |
---|---|
date | Sun, 24 Jun 2007 05:31:48 -0500 |
parents | ddd127429fc6 |
children | 6065d70cb790 |
line wrap: on
line source
/* BMP - Cross-platform multimedia player * Copyright (C) 2003-2004 BMP development team. * * Based on XMMS: * Copyright (C) 1998-2003 XMMS development team. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "widgetcore.h" #include <glib.h> #include <gdk/gdk.h> #include <string.h> #include "main.h" #include "ui_main.h" #include "plugin.h" #include "widget.h" #include "playback.h" #include "vis.h" #include "ui_skinned_window.h" static gint svis_redraw_delays[] = { 1, 2, 4, 8 }; /* FIXME: Are the svis_scope_colors correct? */ static guint8 svis_scope_colors[] = { 20, 19, 18, 19, 20 }; static guint8 svis_vu_normal_colors[] = { 17, 17, 17, 12, 12, 12, 2, 2 }; #define DRAW_DS_PIXEL(ptr,value) \ *(ptr) = (value); \ *((ptr) + 1) = (value); \ *((ptr) + 76) = (value); \ *((ptr) + 77) = (value); #define SVIS_HEIGHT 5 #define SVIS_WIDTH 38 void svis_timeout_func(SVis * svis, guchar * data) { static GTimer *timer = NULL; gulong micros = 9999999; gboolean falloff = FALSE; gint i; if (!timer) { timer = g_timer_new(); g_timer_start(timer); } else { g_timer_elapsed(timer, µs); if (micros > 14000) g_timer_reset(timer); } if (cfg.vis_type == VIS_VOICEPRINT) { if (micros > 14000) falloff = TRUE; for (i = 0; i < 2; i++) { if (falloff || data) { if (data && data[i] > svis->vs_data[i]) svis->vs_data[i] = data[i]; else if (falloff) { if (svis->vs_data[i] >= 2) svis->vs_data[i] -= 2; else svis->vs_data[i] = 0; } } } } else if (data) { for (i = 0; i < 75; i++) svis->vs_data[i] = data[i]; } if (micros > 14000) { if (!svis->vs_refresh_delay) { svis_draw((Widget *) svis); svis->vs_refresh_delay = svis_redraw_delays[cfg.vis_refresh]; } svis->vs_refresh_delay--; } } void svis_draw(Widget * w) { SVis *svis = (SVis *) w; gint x, y, h; guchar svis_color[24][3]; guchar rgb_data[SVIS_WIDTH * 2 * SVIS_HEIGHT * 2], *ptr, c; guint32 colors[24]; GdkRgbCmap *cmap; GDK_THREADS_ENTER(); skin_get_viscolor(bmp_active_skin, svis_color); for (y = 0; y < 24; y++) { colors[y] = svis_color[y][0] << 16 | svis_color[y][1] << 8 | svis_color[y][2]; } cmap = gdk_rgb_cmap_new(colors, 24); if (!cfg.doublesize) { memset(rgb_data, 0, SVIS_WIDTH * SVIS_HEIGHT); if (cfg.vis_type == VIS_ANALYZER && !playback_get_paused() && playback_get_playing()){ for(y=0; y < SVIS_HEIGHT; y++){ if (cfg.analyzer_type == ANALYZER_BARS){ for(x=0;x< SVIS_WIDTH; x++){ if(svis->vs_data[x] > y << 1) { rgb_data[x*3+ (SVIS_HEIGHT - y) * SVIS_WIDTH] = 23; rgb_data[x*3+1 + (SVIS_HEIGHT - y) * SVIS_WIDTH] = 23; } } } else{ for(x=0;x< SVIS_WIDTH; x++){ if(svis->vs_data[x] > y << 1) { rgb_data[x + (SVIS_HEIGHT - y) * SVIS_WIDTH] = 23; } } } } } else if (cfg.vis_type == VIS_VOICEPRINT){ switch (cfg.vu_mode) { case VU_NORMAL: for (y = 0; y < 2; y++) { ptr = rgb_data + ((y * 3) * 38); h = (svis->vs_data[y] * 7) / 37; for (x = 0; x < h; x++, ptr += 5) { c = svis_vu_normal_colors[x]; *(ptr) = c; *(ptr + 1) = c; *(ptr + 2) = c; *(ptr + 38) = c; *(ptr + 39) = c; *(ptr + 40) = c; } } break; case VU_SMOOTH: for (y = 0; y < 2; y++) { ptr = rgb_data + ((y * 3) * SVIS_WIDTH); for (x = 0; x < svis->vs_data[y]; x++, ptr++) { c = 17 - ((x * 15) / 37); *(ptr) = c; *(ptr + 38) = c; } } break; } } else if (cfg.vis_type == VIS_SCOPE) { for (x = 0; x < 38; x++) { h = svis->vs_data[x << 1] / 3; ptr = rgb_data + ((4 - h) * 38) + x; *ptr = svis_scope_colors[h]; } } gdk_draw_indexed_image(mainwin->window, SKINNED_WINDOW(mainwin)->gc, svis->vs_widget.x, svis->vs_widget.y, svis->vs_widget.width, svis->vs_widget.height, GDK_RGB_DITHER_NORMAL, (guchar *) rgb_data, 38, cmap); } else { /* doublesize */ memset(rgb_data, 0, SVIS_WIDTH * 2 * SVIS_HEIGHT * 2); if (cfg.vis_type == VIS_ANALYZER && !playback_get_paused() && playback_get_playing()){ for(y=0; y < SVIS_HEIGHT; y++){ if (cfg.analyzer_type == ANALYZER_BARS){ for(x=0;x< SVIS_WIDTH; x++){ if(svis->vs_data[x] > y << 1) { ptr = rgb_data + x * 6 + (SVIS_HEIGHT * 2 - y * 2) * SVIS_WIDTH * 2; DRAW_DS_PIXEL(ptr, 23); DRAW_DS_PIXEL(ptr + 2, 23); } } } else{ for(x=0;x< SVIS_WIDTH; x++){ if(svis->vs_data[x] > y << 1) { ptr = rgb_data + x * 2 + (SVIS_HEIGHT * 2 - y * 2) * SVIS_WIDTH * 2; DRAW_DS_PIXEL(ptr, 23); } } } } } else if (cfg.vis_type == VIS_VOICEPRINT){ switch (cfg.vu_mode) { case VU_NORMAL: for (y = 0; y < 2; y++) { ptr = rgb_data + ((y * 3) * 152); h = (svis->vs_data[y] * 8) / 37; for (x = 0; x < h; x++, ptr += 10) { c = svis_vu_normal_colors[x]; DRAW_DS_PIXEL(ptr, c); DRAW_DS_PIXEL(ptr + 2, c); DRAW_DS_PIXEL(ptr + 4, c); DRAW_DS_PIXEL(ptr + 152, c); DRAW_DS_PIXEL(ptr + 154, c); DRAW_DS_PIXEL(ptr + 156, c); } } break; case VU_SMOOTH: for (y = 0; y < 2; y++) { ptr = rgb_data + ((y * 3) * 152); for (x = 0; x < svis->vs_data[y]; x++, ptr += 2) { c = 17 - ((x * 15) / 37); DRAW_DS_PIXEL(ptr, c); DRAW_DS_PIXEL(ptr + 152, c); } } break; } } else if (cfg.vis_type == VIS_SCOPE) { for (x = 0; x < 38; x++) { h = svis->vs_data[x << 1] / 3; ptr = rgb_data + ((4 - h) * 152) + (x << 1); *ptr = svis_scope_colors[h]; *(ptr + 1) = svis_scope_colors[h]; *(ptr + 76) = svis_scope_colors[h]; *(ptr + 77) = svis_scope_colors[h]; } } gdk_draw_indexed_image(mainwin->window, SKINNED_WINDOW(mainwin)->gc, svis->vs_widget.x << 1, svis->vs_widget.y << 1, svis->vs_widget.width << 1, svis->vs_widget.height << 1, GDK_RGB_DITHER_NONE, (guchar *) rgb_data, 76, cmap); } gdk_rgb_cmap_free(cmap); GDK_THREADS_LEAVE(); } void svis_clear_data(SVis * svis) { gint i; if (!svis) return; for (i = 0; i < 75; i++) { svis->vs_data[i] = (cfg.vis_type == VIS_SCOPE) ? 6 : 0; } } void svis_clear(SVis * svis) { if (!cfg.doublesize) gdk_window_clear_area(mainwin->window, svis->vs_widget.x, svis->vs_widget.y, svis->vs_widget.width, svis->vs_widget.height); else gdk_window_clear_area(mainwin->window, svis->vs_widget.x << 1, svis->vs_widget.y << 1, svis->vs_widget.width << 1, svis->vs_widget.height << 1); } SVis * create_svis(GList ** wlist, GdkPixmap * parent, GdkGC * gc, gint x, gint y) { SVis *svis; svis = g_new0(SVis, 1); widget_init(&svis->vs_widget, parent, gc, x, y, SVIS_WIDTH, SVIS_HEIGHT, 1); widget_list_add(wlist, WIDGET(svis)); return svis; }