summaryrefslogtreecommitdiffstats
path: root/hacks/filmleader.c
diff options
context:
space:
mode:
authorSimon Rettberg2018-10-16 10:08:48 +0200
committerSimon Rettberg2018-10-16 10:08:48 +0200
commitd3a98cf6cbc3bd0b9efc570f58e8812c03931c18 (patch)
treecbddf8e50f35a9c6e878a5bfe3c6d625d99e12ba /hacks/filmleader.c
downloadxscreensaver-d3a98cf6cbc3bd0b9efc570f58e8812c03931c18.tar.gz
xscreensaver-d3a98cf6cbc3bd0b9efc570f58e8812c03931c18.tar.xz
xscreensaver-d3a98cf6cbc3bd0b9efc570f58e8812c03931c18.zip
Original 5.40
Diffstat (limited to 'hacks/filmleader.c')
-rw-r--r--hacks/filmleader.c571
1 files changed, 571 insertions, 0 deletions
diff --git a/hacks/filmleader.c b/hacks/filmleader.c
new file mode 100644
index 0000000..ef021e1
--- /dev/null
+++ b/hacks/filmleader.c
@@ -0,0 +1,571 @@
+/* filmleader, Copyright (c) 2018 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.
+ *
+ * Simulate an SMPTE Universal Film Leader playing on an analog television.
+ */
+
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif /* HAVE_CONFIG_H */
+
+#include "xft.h" /* before screenhack.h */
+
+#include "screenhack.h"
+#include "analogtv.h"
+
+#include <time.h>
+
+#undef countof
+#define countof(x) (sizeof((x))/sizeof((*x)))
+
+struct state {
+ Display *dpy;
+ Window window;
+ XWindowAttributes xgwa;
+ int w, h;
+ unsigned long bg, text_color, ring_color, trace_color;
+ XftColor xft_text_color_1, xft_text_color_2;
+
+ XftFont *font, *font2, *font3;
+ XftDraw *xftdraw;
+ Pixmap pix;
+ GC gc;
+ double start, last_time;
+ double value;
+ int stop;
+ double noise;
+ analogtv *tv;
+ analogtv_input *inp;
+ analogtv_reception rec;
+ Bool button_down_p;
+};
+
+
+static void *
+filmleader_init (Display *dpy, Window window)
+{
+ struct state *st = (struct state *) calloc (1, sizeof(*st));
+ XGCValues gcv;
+
+ st->dpy = dpy;
+ st->window = window;
+ st->tv = analogtv_allocate (st->dpy, st->window);
+ analogtv_set_defaults (st->tv, "");
+ st->tv->need_clear = 1;
+ st->inp = analogtv_input_allocate();
+ analogtv_setup_sync (st->inp, 1, 0);
+ st->rec.input = st->inp;
+ st->rec.level = 2.0;
+ st->tv->use_color = 1;
+ st->rec.level = pow(frand(1.0), 3.0) * 2.0 + 0.05;
+ st->rec.ofs = random() % ANALOGTV_SIGNAL_LEN;
+ st->tv->powerup = 0;
+
+ st->rec.multipath = 0.0;
+ st->tv->color_control += frand(0.3);
+ st->noise = get_float_resource (st->dpy, "noise", "Float");
+ st->value = 18; /* Leave time for powerup */
+ st->stop = 2 + (random() % 5);
+ XGetWindowAttributes (dpy, window, &st->xgwa);
+
+ /* Let's render it into a 16:9 pixmap, since that's what most screens are
+ these days. That means the circle will be squashed on 4:3 screens. */
+ {
+ double r = 16/9.0;
+
+# ifdef HAVE_MOBILE
+ /* analogtv.c always fills whole screen on mobile, so use screen aspect. */
+ r = st->xgwa.width / (double) st->xgwa.height;
+ if (r < 1) r = 1/r;
+# endif
+
+ st->w = 712;
+ st->h = st->w / r;
+ }
+
+ if (st->xgwa.width < st->xgwa.height)
+ {
+ int swap = st->w;
+ st->w = st->h;
+ st->h = swap;
+ }
+
+ st->pix = XCreatePixmap (dpy, window,
+ st->w > st->h ? st->w : st->h,
+ st->w > st->h ? st->w : st->h,
+ st->xgwa.depth);
+ st->gc = XCreateGC (dpy, st->pix, 0, &gcv);
+
+ st->xftdraw = XftDrawCreate (dpy, st->pix, st->xgwa.visual,
+ st->xgwa.colormap);
+ st->font = load_xft_font_retry (dpy, screen_number (st->xgwa.screen),
+ get_string_resource (dpy, "numberFont",
+ "Font"));
+ st->font2 = load_xft_font_retry (dpy, screen_number (st->xgwa.screen),
+ get_string_resource (dpy, "numberFont2",
+ "Font"));
+ st->font3 = load_xft_font_retry (dpy, screen_number (st->xgwa.screen),
+ get_string_resource (dpy, "numberFont3",
+ "Font"));
+
+ st->bg = get_pixel_resource (dpy, st->xgwa.colormap,
+ "textBackground", "Background");
+ st->text_color = get_pixel_resource (dpy, st->xgwa.colormap,
+ "textColor", "Foreground");
+ st->ring_color = get_pixel_resource (dpy, st->xgwa.colormap,
+ "ringColor", "Foreground");
+ st->trace_color = get_pixel_resource (dpy, st->xgwa.colormap,
+ "traceColor", "Foreground");
+
+ XftColorAllocName (dpy, st->xgwa.visual, st->xgwa.colormap,
+ get_string_resource (dpy, "textColor", "Foreground"),
+ &st->xft_text_color_1);
+ XftColorAllocName (dpy, st->xgwa.visual, st->xgwa.colormap,
+ get_string_resource (dpy, "textBackground", "Background"),
+ &st->xft_text_color_2);
+
+ return st;
+}
+
+
+static double
+double_time (void)
+{
+ struct timeval now;
+# ifdef GETTIMEOFDAY_TWO_ARGS
+ struct timezone tzp;
+ gettimeofday(&now, &tzp);
+# else
+ gettimeofday(&now);
+# endif
+
+ return (now.tv_sec + ((double) now.tv_usec * 0.000001));
+}
+
+
+static unsigned long
+filmleader_draw (Display *dpy, Window window, void *closure)
+{
+ struct state *st = (struct state *) closure;
+ const analogtv_reception *rec = &st->rec;
+ double then = double_time(), now, timedelta;
+ XImage *img;
+ int i, x, y, w2, h2;
+ XGlyphInfo extents;
+ int lbearing, rbearing, ascent, descent;
+ char s[20];
+ double r = 1 - (st->value - (int) st->value);
+ int ivalue = st->value;
+ XftFont *xftfont;
+ XftColor *xftcolor;
+
+ /* You may ask, why use Xft for this instead of the much simpler XDrawString?
+ Well, for some reason, XLoadQueryFont is giving me horribly-scaled bitmap
+ fonts, but Xft works properly. So perhaps in This Modern World, if one
+ expects large fonts to work, one must use Xft instead of Xlib?
+
+ Everything is terrible.
+ */
+
+ const struct { double t; int k, f; const char * const s[4]; } blurbs[] = {
+ { 9.1, 3, 1, { "PICTURE", " START", 0, 0 }},
+ { 10.0, 2, 1, { " 16", "SOUND", "START", 0 }},
+ { 10.5, 2, 1, { " 32", "SOUND", "START", 0 }},
+ { 11.6, 2, 0, { "PICTURE", "COMPANY", "SERIES", 0 }},
+ { 11.7, 2, 0, { "XSCRNSAVER", 0, 0, 0 }},
+ { 11.9, 2, 0, { "REEL No.", "PROD No.", "PLAY DATE", 0 }},
+ { 12.2, 0, 0, { " SMPTE ", "UNIVERSAL", " LEADER", 0 }},
+ { 12.3, 0, 1, { "X ", "X", "X", "X" }},
+ { 12.4, 0, 0, { " SMPTE ", "UNIVERSAL", " LEADER", 0 }},
+ { 12.5, 3, 1, { "PICTURE", 0, 0, 0 }},
+ { 12.7, 3, 1, { "HEAD", 0, 0, 0 }},
+ { 12.8, 2, 1, { "OOOO", 0, "ASPECT", "TYPE OF" }},
+ { 12.9, 2, 0, { "SOUND", 0, "RATIO", 0 }},
+ { 13.2, 1, 1, { " ", "PICTURE", 0, 0 }},
+ { 13.3, 1, 0, { "REEL No. ", "COLOR", 0, 0 }},
+ { 13.4, 1, 0, { "LENGTH ", 0, 0, "ROLL" }},
+ { 13.5, 1, 0, { "SUBJECT", 0, 0, 0 }},
+ { 13.9, 1, 1, { " \342\206\221", "SPLICE", " HERE", 0 }},
+ };
+
+ for (i = 0; i < countof(blurbs); i++)
+ {
+ if (st->value >= blurbs[i].t && st->value <= blurbs[i].t + 1/15.0)
+ {
+ int line_height;
+ int j;
+ xftfont = (blurbs[i].f == 1 ? st->font2 :
+ blurbs[i].f == 2 ? st->font : st->font3);
+
+ XSetForeground (dpy, st->gc,
+ blurbs[i].k == 3 ? st->bg : st->text_color);
+ XFillRectangle (dpy, st->pix, st->gc, 0, 0, st->w, st->h);
+ XSetForeground (dpy, st->gc,
+ blurbs[i].k == 3 ? st->text_color : st->bg);
+ xftcolor = (blurbs[i].k == 3 ?
+ &st->xft_text_color_1 : &st->xft_text_color_2);
+
+ /* The height of a string of spaces is 0... */
+ XftTextExtentsUtf8 (dpy, xftfont, (FcChar8 *) "My", 2, &extents);
+ line_height = extents.height;
+
+ XftTextExtentsUtf8 (dpy, xftfont, (FcChar8 *)
+ blurbs[i].s[0], strlen(blurbs[i].s[0]),
+ &extents);
+ lbearing = -extents.x;
+ rbearing = extents.width - extents.x;
+ ascent = extents.y;
+ descent = extents.height - extents.y;
+
+ x = (st->w - rbearing) / 2;
+ y = st->h * 0.1 + ascent;
+
+ for (j = 0; j < countof(blurbs[i].s); j++)
+ {
+ if (blurbs[i].s[j])
+ XftDrawStringUtf8 (st->xftdraw, xftcolor, xftfont, x, y,
+ (FcChar8 *) blurbs[i].s[j],
+ strlen(blurbs[i].s[j]));
+
+ y += line_height * 1.5;
+
+ if (blurbs[i].s[j])
+ {
+ XftTextExtentsUtf8 (dpy, xftfont, (FcChar8 *)
+ blurbs[i].s[0], strlen(blurbs[i].s[j]),
+ &extents);
+ lbearing = -extents.x;
+ rbearing = extents.width - extents.x;
+ ascent = extents.y;
+ descent = extents.height - extents.y;
+ }
+ }
+
+ if (blurbs[i].k == 2) /* Rotate clockwise and flip */
+ {
+ int wh = st->w < st->h ? st->w : st->h;
+ XImage *img1 = XGetImage (dpy, st->pix,
+ (st->w - wh) / 2,
+ (st->h - wh) / 2,
+ wh, wh, ~0L, ZPixmap);
+ XImage *img2 = XCreateImage (dpy, st->xgwa.visual,
+ st->xgwa.depth, ZPixmap, 0, 0,
+ wh, wh, 32, 0);
+ img2->data = malloc (img2->bytes_per_line * img2->height);
+ for (y = 0; y < wh; y++)
+ for (x = 0; x < wh; x++)
+ XPutPixel (img2, y, x, XGetPixel (img1, x, y));
+ XSetForeground (dpy, st->gc,
+ blurbs[i].k == 3 ? st->bg : st->text_color);
+ XFillRectangle (dpy, st->pix, st->gc, 0, 0, st->w, st->h);
+ XPutImage (dpy, st->pix, st->gc, img2,
+ 0, 0,
+ (st->w - wh) / 2,
+ (st->h - wh) / 2,
+ wh, wh);
+ XDestroyImage (img1);
+ XDestroyImage (img2);
+ }
+ else if (blurbs[i].k == 1) /* Flip vertically */
+ {
+ XImage *img1 = XGetImage (dpy, st->pix, 0, 0,
+ st->w, st->h, ~0L, ZPixmap);
+ XImage *img2 = XCreateImage (dpy, st->xgwa.visual,
+ st->xgwa.depth, ZPixmap, 0, 0,
+ st->w, st->h, 32, 0);
+ img2->data = malloc (img2->bytes_per_line * img2->height);
+ for (y = 0; y < img2->height; y++)
+ for (x = 0; x < img2->width; x++)
+ XPutPixel (img2, x, img2->height-y-1,
+ XGetPixel (img1, x, y));
+ XPutImage (dpy, st->pix, st->gc, img2, 0, 0, 0, 0, st->w, st->h);
+ XDestroyImage (img1);
+ XDestroyImage (img2);
+ }
+
+ goto DONE;
+ }
+ }
+
+ if (st->value < 2.0 || st->value >= 9.0) /* Black screen */
+ {
+ XSetForeground (dpy, st->gc, st->text_color);
+ XFillRectangle (dpy, st->pix, st->gc, 0, 0, st->w, st->h);
+ goto DONE;
+ }
+
+ XSetForeground (dpy, st->gc, st->bg);
+ XFillRectangle (dpy, st->pix, st->gc, 0, 0, st->w, st->h);
+
+ if (r > 1/30.0) /* Sweep line and background */
+ {
+ x = st->w/2 + st->w * cos (M_PI * 2 * r - M_PI/2);
+ y = st->h/2 + st->h * sin (M_PI * 2 * r - M_PI/2);
+
+ XSetForeground (dpy, st->gc, st->trace_color);
+ XFillArc (dpy, st->pix, st->gc,
+ -st->w, -st->h, st->w*3, st->h*3,
+ 90*64,
+ 90*64 - ((r + 0.25) * 360*64));
+
+ XSetForeground (dpy, st->gc, st->text_color);
+ XSetLineAttributes (dpy, st->gc, 1, LineSolid, CapRound, JoinRound);
+ XDrawLine (dpy, st->pix, st->gc, st->w/2, st->h/2, x, y);
+
+ XSetForeground (dpy, st->gc, st->text_color);
+ XSetLineAttributes (dpy, st->gc, 2, LineSolid, CapRound, JoinRound);
+ XDrawLine (dpy, st->pix, st->gc, st->w/2, 0, st->w/2, st->h);
+ XDrawLine (dpy, st->pix, st->gc, 0, st->h/2, st->w, st->h/2);
+ }
+
+ /* Big number */
+
+ s[0] = (char) (ivalue + '0');
+ xftfont = st->font;
+ xftcolor = &st->xft_text_color_1;
+ XftTextExtentsUtf8 (dpy, xftfont, (FcChar8 *) s, 1, &extents);
+ lbearing = -extents.x;
+ rbearing = extents.width - extents.x;
+ ascent = extents.y;
+ descent = extents.height - extents.y;
+
+ x = (st->w - (rbearing + lbearing)) / 2;
+ y = (st->h + (ascent - descent)) / 2;
+ XftDrawStringUtf8 (st->xftdraw, xftcolor, xftfont, x, y, (FcChar8 *) s, 1);
+
+ /* Annotations on 7 and 4 */
+
+ if ((st->value >= 7.75 && st->value <= 7.85) ||
+ (st->value >= 4.00 && st->value <= 4.25))
+ {
+ XSetForeground (dpy, st->gc, st->ring_color);
+ xftcolor = &st->xft_text_color_2;
+ xftfont = st->font2;
+
+ s[0] = (ivalue == 4 ? 'C' : 'M');
+ s[1] = 0;
+
+ XftTextExtentsUtf8 (dpy, xftfont, (FcChar8 *) s, strlen(s), &extents);
+ lbearing = -extents.x;
+ rbearing = extents.width - extents.x;
+ ascent = extents.y;
+ descent = extents.height - extents.y;
+
+ x = st->w * 0.1;
+ y = st->h * 0.1 + ascent;
+ XftDrawStringUtf8 (st->xftdraw, xftcolor, xftfont, x, y,
+ (FcChar8 *) s, strlen(s));
+ x = st->w * 0.9 - (rbearing + lbearing);
+ XftDrawStringUtf8 (st->xftdraw, xftcolor, xftfont, x, y,
+ (FcChar8 *) s, strlen(s));
+
+ s[0] = (ivalue == 4 ? 'F' : '3');
+ s[1] = (ivalue == 4 ? 0 : '5');
+ s[2] = 0;
+
+ XftTextExtentsUtf8 (dpy, xftfont, (FcChar8 *) s, strlen(s), &extents);
+ lbearing = -extents.x;
+ rbearing = extents.width - extents.x;
+ ascent = extents.y;
+ descent = extents.height - extents.y;
+
+ x = st->w * 0.1;
+ y = st->h * 0.95;
+ XftDrawStringUtf8 (st->xftdraw, xftcolor, xftfont, x, y,
+ (FcChar8 *) s, strlen(s));
+ x = st->w * 0.9 - (rbearing + lbearing);
+ XftDrawStringUtf8 (st->xftdraw, xftcolor, xftfont, x, y,
+ (FcChar8 *) s, strlen(s));
+ }
+
+ if (r > 1/30.0) /* Two rings around number */
+ {
+ double r2 = st->w / (double) st->h;
+ double ss = 1;
+
+ if (st->xgwa.width < st->xgwa.height)
+ ss = 0.5;
+
+ XSetForeground (dpy, st->gc, st->ring_color);
+ XSetLineAttributes (dpy, st->gc, st->w * 0.025,
+ LineSolid, CapRound, JoinRound);
+
+ w2 = st->w * 0.8 * ss / r2;
+ h2 = st->h * 0.8 * ss;
+ x = (st->w - w2) / 2;
+ y = (st->h - h2) / 2;
+ XDrawArc (dpy, st->pix, st->gc, x, y, w2, h2, 0, 360*64);
+
+ w2 = w2 * 0.8;
+ h2 = h2 * 0.8;
+ x = (st->w - w2) / 2;
+ y = (st->h - h2) / 2;
+ XDrawArc (dpy, st->pix, st->gc, x, y, w2, h2, 0, 360*64);
+ }
+
+ DONE:
+
+ img = XGetImage (dpy, st->pix, 0, 0, st->w, st->h, ~0L, ZPixmap);
+
+ analogtv_load_ximage (st->tv, st->rec.input, img, 0, 0, 0, 0, 0);
+ analogtv_reception_update (&st->rec);
+ analogtv_draw (st->tv, st->noise, &rec, 1);
+
+ XDestroyImage (img);
+
+ now = double_time();
+ timedelta = (1 / 29.97) - (now - then);
+
+ if (! st->button_down_p)
+ {
+ if (st->last_time == 0)
+ st->start = then;
+ else
+ st->value -= then - st->last_time;
+
+ if (st->value <= 0 ||
+ (r > 0.9 && st->value <= st->stop))
+ {
+ st->value = (random() % 20) ? 8.9 : 15;
+ st->stop = ((random() % 50) ? 2 : 1) + (random() % 5);
+
+ if (st->value > 9) /* Spin the knobs again */
+ {
+ st->rec.level = pow(frand(1.0), 3.0) * 2.0 + 0.05;
+ st->rec.ofs = random() % ANALOGTV_SIGNAL_LEN;
+ st->tv->color_control += frand(0.3) - 0.15;
+ }
+ }
+ }
+
+ st->tv->powerup = then - st->start;
+ st->last_time = then;
+
+ return timedelta > 0 ? timedelta * 1000000 : 0;
+}
+
+
+static void
+filmleader_reshape (Display *dpy, Window window, void *closure,
+ unsigned int w, unsigned int h)
+{
+ struct state *st = (struct state *) closure;
+ analogtv_reconfigure (st->tv);
+ XGetWindowAttributes (dpy, window, &st->xgwa);
+
+ if ((st->w > st->h) != (st->xgwa.width > st->xgwa.height))
+ {
+ int swap = st->w;
+ st->w = st->h;
+ st->h = swap;
+ }
+}
+
+
+static Bool
+filmleader_event (Display *dpy, Window window, void *closure, XEvent *event)
+{
+ struct state *st = (struct state *) closure;
+ if (event->xany.type == ButtonPress)
+ {
+ st->button_down_p = True;
+ return True;
+ }
+ else if (event->xany.type == ButtonRelease)
+ {
+ st->button_down_p = False;
+ return True;
+ }
+ else if (screenhack_event_helper (dpy, window, event))
+ {
+ st->value = 15;
+ st->rec.level = pow(frand(1.0), 3.0) * 2.0 + 0.05;
+ st->rec.ofs = random() % ANALOGTV_SIGNAL_LEN;
+ st->tv->color_control += frand(0.3) - 0.15;
+ return True;
+ }
+ else if (event->xany.type == KeyPress)
+ {
+ KeySym keysym;
+ char c = 0;
+ XLookupString (&event->xkey, &c, 1, &keysym, 0);
+ if (c >= '2' && c <= '8')
+ {
+ st->value = (c - '0') + (st->value - (int) st->value);
+ return True;
+ }
+ }
+
+ return False;
+}
+
+
+static void
+filmleader_free (Display *dpy, Window window, void *closure)
+{
+ struct state *st = (struct state *) closure;
+ analogtv_release (st->tv);
+ XftDrawDestroy (st->xftdraw);
+ XftColorFree(dpy, st->xgwa.visual, st->xgwa.colormap, &st->xft_text_color_1);
+ XftColorFree(dpy, st->xgwa.visual, st->xgwa.colormap, &st->xft_text_color_2);
+ XFreePixmap (dpy, st->pix);
+ XFreeGC (dpy, st->gc);
+ free (st);
+}
+
+
+static const char *filmleader_defaults [] = {
+
+ ".background: #000000",
+
+# ifdef HAVE_MOBILE
+
+ "*textBackground: #444488", /* Need much higher contrast for some reason */
+ "*textColor: #000033",
+ "*ringColor: #DDDDFF",
+ "*traceColor: #222244",
+
+# else /* X11 or Cocoa */
+
+ "*textBackground: #9999DD",
+ "*textColor: #000015",
+ "*ringColor: #DDDDFF",
+ "*traceColor: #555577",
+
+# endif
+
+# ifdef USE_IPHONE
+
+ "*numberFont: Helvetica Bold 120",
+ "*numberFont2: Helvetica 36",
+ "*numberFont3: Helvetica 28",
+
+# else /* X11, Cocoa or Android */
+
+ "*numberFont: -*-helvetica-bold-r-*-*-*-1700-*-*-*-*-*-*",
+ "*numberFont2: -*-helvetica-medium-r-*-*-*-500-*-*-*-*-*-*",
+ "*numberFont3: -*-helvetica-medium-r-*-*-*-360-*-*-*-*-*-*",
+
+# endif
+
+
+ "*noise: 0.04",
+ ANALOGTV_DEFAULTS
+ "*geometry: 1280x720",
+ 0
+};
+
+static XrmOptionDescRec filmleader_options [] = {
+ { "-noise", ".noise", XrmoptionSepArg, 0 },
+ ANALOGTV_OPTIONS
+ { 0, 0, 0, 0 }
+};
+
+XSCREENSAVER_MODULE ("FilmLeader", filmleader)