358 lines
8 KiB
C
358 lines
8 KiB
C
/* xscreensaver, Copyright (c) 1999 Jamie Zawinski <jwz@jwz.org>
|
|
*
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
* the above copyright notice appear in all copies and that both that
|
|
* copyright notice and this permission notice appear in supporting
|
|
* documentation. No representations are made about the suitability of this
|
|
* software for any purpose. It is provided "as is" without express or
|
|
* implied warranty.
|
|
*
|
|
* Matrix -- simulate the text scrolls from the movie "The Matrix".
|
|
*
|
|
* The movie people distribute their own Windows/Mac screensaver that does
|
|
* a similar thing, so I wrote one for Unix. However, that version (the
|
|
* Windows/Mac version at http://www.whatisthematrix.com/) doesn't match my
|
|
* memory of what the screens in the movie looked like, so my `xmatrix'
|
|
* does things differently.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include "images/small.xpm"
|
|
#include "images/medium.xpm"
|
|
#include "images/large.xpm"
|
|
#include "images/matrix.xbm"
|
|
/*
|
|
#define CHAR_HEIGHT 4
|
|
*/
|
|
|
|
#include "matrix.h"
|
|
/*
|
|
#include "resources.h"
|
|
*/
|
|
|
|
extern GC NormalGC;
|
|
extern GC EraseGC;
|
|
extern Pixel back_pix, fore_pix;
|
|
extern int PixmapSize;
|
|
int CHAR_HEIGHT;
|
|
|
|
static void load_images(m_state * state)
|
|
{
|
|
|
|
if (state->xgwa.depth > 1) {
|
|
|
|
XpmAttributes xpmattrs;
|
|
int result;
|
|
|
|
xpmattrs.valuemask = 0;
|
|
xpmattrs.valuemask |= XpmCloseness;
|
|
xpmattrs.closeness = 40000;
|
|
xpmattrs.valuemask |= XpmVisual;
|
|
xpmattrs.visual = state->xgwa.visual;
|
|
xpmattrs.valuemask |= XpmDepth;
|
|
xpmattrs.depth = state->xgwa.depth;
|
|
xpmattrs.valuemask |= XpmColormap;
|
|
xpmattrs.colormap = state->xgwa.colormap;
|
|
|
|
if (PixmapSize == 1) {
|
|
|
|
CHAR_HEIGHT = 4;
|
|
result = XpmCreatePixmapFromData(state->dpy, state->window, small, &state->images, 0 /* mask */ ,
|
|
&xpmattrs);
|
|
} else if (PixmapSize == 2) {
|
|
|
|
CHAR_HEIGHT = 6;
|
|
result = XpmCreatePixmapFromData(state->dpy, state->window, medium, &state->images, 0 /* mask */ ,
|
|
&xpmattrs);
|
|
} else {
|
|
|
|
CHAR_HEIGHT = 8;
|
|
result = XpmCreatePixmapFromData(state->dpy, state->window, large, &state->images, 0 /* mask */ ,
|
|
&xpmattrs);
|
|
}
|
|
|
|
if (!state->images || (result != XpmSuccess && result != XpmColorError))
|
|
state->images = 0;
|
|
|
|
state->image_width = xpmattrs.width;
|
|
state->image_height = xpmattrs.height;
|
|
state->nglyphs = state->image_height / CHAR_HEIGHT;
|
|
|
|
} else {
|
|
|
|
state->image_width = matrix_width;
|
|
state->image_height = matrix_height;
|
|
state->nglyphs = state->image_height / CHAR_HEIGHT;
|
|
|
|
state->images = XCreatePixmapFromBitmapData(state->dpy, state->window,
|
|
(char *)matrix_bits,
|
|
state->image_width, state->image_height, back_pix, fore_pix, state->xgwa.depth);
|
|
}
|
|
|
|
}
|
|
|
|
m_state *init_matrix(Display * dpy, Window window)
|
|
{
|
|
|
|
m_state *state = (m_state *) calloc(sizeof(*state), 1);
|
|
|
|
state->dpy = dpy;
|
|
state->window = window;
|
|
|
|
XGetWindowAttributes(dpy, window, &state->xgwa);
|
|
load_images(state);
|
|
|
|
state->draw_gc = NormalGC;
|
|
state->erase_gc = EraseGC;
|
|
|
|
state->char_width = state->image_width / 2;
|
|
state->char_height = CHAR_HEIGHT;
|
|
|
|
state->grid_width = state->xgwa.width / state->char_width;
|
|
state->grid_height = state->xgwa.height / state->char_height;
|
|
state->grid_width++;
|
|
state->grid_height++;
|
|
|
|
state->cells = (m_cell *) calloc(sizeof(m_cell), state->grid_width * state->grid_height);
|
|
state->feeders = (m_feeder *) calloc(sizeof(m_feeder), state->grid_width);
|
|
|
|
state->density = 40;
|
|
|
|
state->insert_top_p = False;
|
|
state->insert_bottom_p = True;
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
static void insert_glyph(m_state * state, int glyph, int x, int y)
|
|
{
|
|
|
|
Bool bottom_feeder_p = (y >= 0);
|
|
m_cell *from, *to;
|
|
|
|
if (y >= state->grid_height)
|
|
return;
|
|
|
|
if (bottom_feeder_p) {
|
|
|
|
to = &state->cells[state->grid_width * y + x];
|
|
|
|
} else {
|
|
|
|
for (y = state->grid_height - 1; y > 0; y--) {
|
|
|
|
from = &state->cells[state->grid_width * (y - 1) + x];
|
|
to = &state->cells[state->grid_width * y + x];
|
|
*to = *from;
|
|
to->changed = True;
|
|
|
|
}
|
|
to = &state->cells[x];
|
|
|
|
}
|
|
|
|
to->glyph = glyph;
|
|
to->changed = True;
|
|
|
|
if (!to->glyph) ;
|
|
else if (bottom_feeder_p)
|
|
to->glow = 1 + (random() % 2);
|
|
else
|
|
to->glow = 0;
|
|
|
|
}
|
|
|
|
static void feed_matrix(m_state * state)
|
|
{
|
|
|
|
int x;
|
|
|
|
/*
|
|
* Update according to current feeders.
|
|
*/
|
|
for (x = 0; x < state->grid_width; x++) {
|
|
|
|
m_feeder *f = &state->feeders[x];
|
|
|
|
if (f->throttle) { /* this is a delay tick, synced to frame. */
|
|
|
|
f->throttle--;
|
|
|
|
} else if (f->remaining > 0) { /* how many items are in the pipe */
|
|
|
|
int g = (random() % state->nglyphs) + 1;
|
|
insert_glyph(state, g, x, f->y);
|
|
f->remaining--;
|
|
if (f->y >= 0)
|
|
f->y++; /* bottom_feeder_p */
|
|
|
|
} else { /* if pipe is empty, insert spaces */
|
|
|
|
insert_glyph(state, 0, x, f->y);
|
|
if (f->y >= 0)
|
|
f->y++; /* bottom_feeder_p */
|
|
|
|
}
|
|
|
|
if ((random() % 10) == 0) { /* randomly change throttle speed */
|
|
|
|
f->throttle = ((random() % 5) + (random() % 5));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static int densitizer(m_state * state)
|
|
{
|
|
|
|
/* Horrid kludge that converts percentages (density of screen coverage)
|
|
to the parameter that actually controls this. I got this mapping
|
|
empirically, on a 1024x768 screen. Sue me. */
|
|
if (state->density < 10)
|
|
return 85;
|
|
else if (state->density < 15)
|
|
return 60;
|
|
else if (state->density < 20)
|
|
return 45;
|
|
else if (state->density < 25)
|
|
return 25;
|
|
else if (state->density < 30)
|
|
return 20;
|
|
else if (state->density < 35)
|
|
return 15;
|
|
else if (state->density < 45)
|
|
return 10;
|
|
else if (state->density < 50)
|
|
return 8;
|
|
else if (state->density < 55)
|
|
return 7;
|
|
else if (state->density < 65)
|
|
return 5;
|
|
else if (state->density < 80)
|
|
return 3;
|
|
else if (state->density < 90)
|
|
return 2;
|
|
else
|
|
return 1;
|
|
|
|
}
|
|
|
|
static void hack_matrix(m_state * state)
|
|
{
|
|
|
|
int x;
|
|
|
|
/* Glow some characters. */
|
|
if (!state->insert_bottom_p) {
|
|
|
|
int i = random() % (state->grid_width / 2);
|
|
while (--i > 0) {
|
|
|
|
int x = random() % state->grid_width;
|
|
int y = random() % state->grid_height;
|
|
m_cell *cell = &state->cells[state->grid_width * y + x];
|
|
if (cell->glyph && cell->glow == 0) {
|
|
|
|
cell->glow = random() % 10;
|
|
cell->changed = True;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
/* Change some of the feeders. */
|
|
for (x = 0; x < state->grid_width; x++) {
|
|
|
|
m_feeder *f = &state->feeders[x];
|
|
Bool bottom_feeder_p;
|
|
|
|
if (f->remaining > 0) /* never change if pipe isn't empty */
|
|
continue;
|
|
|
|
if ((random() % densitizer(state)) != 0) /* then change N% of the time */
|
|
continue;
|
|
|
|
f->remaining = 3 + (random() % state->grid_height);
|
|
f->throttle = ((random() % 5) + (random() % 5));
|
|
|
|
if ((random() % 4) != 0)
|
|
f->remaining = 0;
|
|
|
|
if (state->insert_top_p && state->insert_bottom_p)
|
|
bottom_feeder_p = (random() & 1);
|
|
else
|
|
bottom_feeder_p = state->insert_bottom_p;
|
|
|
|
if (bottom_feeder_p)
|
|
f->y = random() % (state->grid_height / 2);
|
|
else
|
|
f->y = -1;
|
|
}
|
|
}
|
|
|
|
void draw_matrix(m_state * state, int d)
|
|
{
|
|
|
|
int x, y;
|
|
int count = 0;
|
|
|
|
state->density = d;
|
|
feed_matrix(state);
|
|
hack_matrix(state);
|
|
|
|
for (y = 0; y < state->grid_height; y++) {
|
|
for (x = 0; x < state->grid_width; x++) {
|
|
|
|
m_cell *cell = &state->cells[state->grid_width * y + x];
|
|
|
|
if (cell->glyph)
|
|
count++;
|
|
|
|
if (!cell->changed)
|
|
continue;
|
|
|
|
if (cell->glyph == 0) {
|
|
|
|
XFillRectangle(state->dpy, state->window, state->erase_gc,
|
|
x * state->char_width, y * state->char_height, state->char_width, state->char_height);
|
|
} else {
|
|
|
|
XCopyArea(state->dpy, state->images, state->window, state->draw_gc,
|
|
(cell->glow ? state->char_width : 0), (cell->glyph - 1) * state->char_height,
|
|
state->char_width, state->char_height, x * state->char_width, y * state->char_height);
|
|
|
|
}
|
|
|
|
cell->changed = False;
|
|
|
|
if (cell->glow > 0) {
|
|
|
|
cell->glow--;
|
|
cell->changed = True;
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
{
|
|
static int i = 0;
|
|
static int ndens = 0;
|
|
static int tdens = 0;
|
|
i++;
|
|
if (i > 50) {
|
|
int dens = (100.0 * (((double)count) / ((double)(state->grid_width * state->grid_height))));
|
|
tdens += dens;
|
|
ndens++;
|
|
printf("density: %d%% (%d%%)\n", dens, (tdens / ndens));
|
|
i = 0;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
}
|