/* -*- Mode: C; tab-width: 4 -*- */ /* fadeplot --- a fading plot of sine squared */ #if !defined( lint ) && !defined( SABER ) static const char sccsid[] = "@(#)fadeplot.c 5.00 2000/11/01 xlockmore"; #endif /*- * Some easy plotting stuff, by Bas van Gaalen, Holland, PD * * Copyright (c) 1996 by Charles Vidal * * Permission to use, copy, modify, and distribute this software and its * documentation for any purpose and without fee is hereby granted, * provided that the above copyright notice appear in all copies and that * both that copyright notice and this permission notice appear in * supporting documentation. * * This file is provided AS IS with no warranties of any kind. The author * shall have no liability with respect to the infringement of copyrights, * trade secrets or any patents by this file or any part thereof. In no * event will the author be liable for any lost revenue or profits or * other special, indirect and consequential damages. * * Revision History: * 01-Nov-2000: Allocation checks * 10-May-1997: Compatible with screensaver * 1996: Written by Charles Vidal based on work by Bas van Gaalen */ #ifdef STANDALONE #define MODE_fadeplot #define PROGCLASS "Fadeplot" #define HACK_INIT init_fadeplot #define HACK_DRAW draw_fadeplot #define fadeplot_opts xlockmore_opts #define DEFAULTS "*delay: 30000 \n" \ "*count: 10 \n" \ "*cycles: 1500 \n" \ "*ncolors: 64 \n" #define BRIGHT_COLORS #define UNIFORM_COLORS #include "xlockmore.h" /* in xscreensaver distribution */ #else /* STANDALONE */ #include "xlock.h" /* in xlockmore distribution */ #endif /* STANDALONE */ #ifdef MODE_fadeplot ModeSpecOpt fadeplot_opts = {0, (XrmOptionDescRec *) NULL, 0, (argtype *) NULL, (OptionStruct *) NULL}; #ifdef USE_MODULES ModStruct fadeplot_description = {"fadeplot", "init_fadeplot", "draw_fadeplot", "release_fadeplot", "refresh_fadeplot", "init_fadeplot", (char *) NULL, &fadeplot_opts, 30000, 10, 1500, 1, 64, 0.6, "", "Shows a fading plot of sine squared", 0, NULL}; #endif #define MINSTEPS 1 typedef struct { XPoint speed, step, factor, st; int temps, maxpts, nbstep; int min; int width, height; int pix; int angles; int *stab; XPoint *pts; } fadeplotstruct; static fadeplotstruct *fadeplots = (fadeplotstruct *) NULL; static void free_fadeplot(fadeplotstruct *fp) { if (fp->pts != NULL) { free(fp->pts); fp->pts = (XPoint *) NULL; } if (fp->stab != NULL) { free(fp->stab); fp->stab = (int *) NULL; } } static Bool initSintab(ModeInfo * mi) { fadeplotstruct *fp = &fadeplots[MI_SCREEN(mi)]; int i; float x; fp->angles = NRAND(950) + 250; if ((fp->stab = (int *) malloc(fp->angles * sizeof (int))) == NULL) { free_fadeplot(fp); return False; } for (i = 0; i < fp->angles; i++) { x = SINF(2.0 * M_PI * i / fp->angles); fp->stab[i] = (int) (x * ABS(x) * fp->min) + fp->min; } return True; } void init_fadeplot(ModeInfo * mi) { fadeplotstruct *fp; if (fadeplots == NULL) { if ((fadeplots = (fadeplotstruct *) calloc(MI_NUM_SCREENS(mi), sizeof (fadeplotstruct))) == NULL) return; } fp = &fadeplots[MI_SCREEN(mi)]; fp->width = MI_WIDTH(mi); fp->height = MI_HEIGHT(mi); fp->min = MAX(MIN(fp->width, fp->height) / 2, 1); fp->speed.x = 8; fp->speed.y = 10; fp->step.x = 1; fp->step.y = 1; fp->temps = 0; fp->factor.x = MAX(fp->width / (2 * fp->min), 1); fp->factor.y = MAX(fp->height / (2 * fp->min), 1); fp->nbstep = MI_COUNT(mi); if (fp->nbstep < -MINSTEPS) { fp->nbstep = NRAND(-fp->nbstep - MINSTEPS + 1) + MINSTEPS; } else if (fp->nbstep < MINSTEPS) fp->nbstep = MINSTEPS; fp->maxpts = MI_CYCLES(mi); if (fp->maxpts < 1) fp->maxpts = 1; if (fp->pts == NULL) { if ((fp->pts = (XPoint *) calloc( fp->maxpts, sizeof (XPoint))) == NULL) { free_fadeplot(fp); return; } } if (MI_NPIXELS(mi) > 2) fp->pix = NRAND(MI_NPIXELS(mi)); if (fp->stab != NULL) free(fp->stab); if (!initSintab(mi)) return; MI_CLEARWINDOW(mi); } void draw_fadeplot(ModeInfo * mi) { Display *display = MI_DISPLAY(mi); Window window = MI_WINDOW(mi); GC gc = MI_GC(mi); int i, j, temp; fadeplotstruct *fp; if (fadeplots == NULL) return; fp = &fadeplots[MI_SCREEN(mi)]; if (fp->stab == NULL) return; MI_IS_DRAWN(mi) = True; XSetForeground(display, gc, MI_BLACK_PIXEL(mi)); XDrawPoints(display, window, gc, fp->pts, fp->maxpts, CoordModeOrigin); if (MI_NPIXELS(mi) > 2) { XSetForeground(display, gc, MI_PIXEL(mi, fp->pix)); if (++fp->pix >= MI_NPIXELS(mi)) fp->pix = 0; } else XSetForeground(display, gc, MI_WHITE_PIXEL(mi)); temp = 0; for (j = 0; j < fp->nbstep; j++) { for (i = 0; i < fp->maxpts / fp->nbstep; i++) { fp->pts[temp].x = fp->stab[(fp->st.x + fp->speed.x * j + i * fp->step.x) % fp->angles] * fp->factor.x + fp->width / 2 - fp->min; fp->pts[temp].y = fp->stab[(fp->st.y + fp->speed.y * j + i * fp->step.y) % fp->angles] * fp->factor.y + fp->height / 2 - fp->min; temp++; } } XDrawPoints(display, window, gc, fp->pts, temp, CoordModeOrigin); XFlush(display); fp->st.x = (fp->st.x + fp->speed.x) % fp->angles; fp->st.y = (fp->st.y + fp->speed.y) % fp->angles; fp->temps++; if ((fp->temps % (fp->angles / 2)) == 0) { fp->temps = fp->temps % fp->angles * 5; if ((fp->temps % (fp->angles)) == 0) fp->speed.y = (fp->speed.y++) % 30 + 1; if ((fp->temps % (fp->angles * 2)) == 0) fp->speed.x = (fp->speed.x) % 20; if ((fp->temps % (fp->angles * 3)) == 0) fp->step.y = (fp->step.y++) % 2 + 1; MI_CLEARWINDOWCOLORMAPFAST(mi, MI_GC(mi), MI_BLACK_PIXEL(mi)); } } void refresh_fadeplot(ModeInfo * mi) { MI_CLEARWINDOW(mi); } void release_fadeplot(ModeInfo * mi) { if (fadeplots != NULL) { int screen; for (screen = 0; screen < MI_NUM_SCREENS(mi); screen++) free_fadeplot(&fadeplots[screen]); free(fadeplots); fadeplots = (fadeplotstruct *) NULL; } } #endif /* MODE_fadeplot */