1 /* xscreensaver, Copyright (c) 1997-2013 Jamie Zawinski <jwz@jwz.org>
3 * Permission to use, copy, modify, distribute, and sell this software and its
4 * documentation for any purpose is hereby granted without fee, provided that
5 * the above copyright notice appear in all copies and that both that
6 * copyright notice and this permission notice appear in supporting
7 * documentation. No representations are made about the suitability of this
8 * software for any purpose. It is provided "as is" without express or
11 * Wendy, let me explain something to you. Whenever you come in here and
12 * interrupt me, you're BREAKING my CONCENTRATION. You're DISTRACTING me!
13 * And it will then take me time to get back to where I was. You understand?
14 * Now, we're going to make a new rule. When you come in here and you hear
15 * me typing, or whether you DON'T hear me typing, or whatever the FUCK you
16 * hear me doing; when I'm in here, it means that I am working, THAT means
17 * don't come in! Now, do you think you can handle that?
21 #include "screenhack.h"
23 static const char *source = "All work and no play makes Jack a dull boy. ";
24 /* If you're here because you're thinking about making the above string be
25 customizable, then you don't get the joke. You loser. */
30 XWindowAttributes xgwa;
35 int columns, rows; /* characters */
36 int left, right; /* characters */
37 int char_width, line_height; /* pixels */
38 int x, y; /* characters */
40 int hspace; /* pixels */
41 int vspace; /* pixels */
55 xjack_reshape (Display *dpy, Window window, void *closure,
56 unsigned int w, unsigned int h)
58 struct state *st = (struct state *) closure;
59 XGetWindowAttributes (st->dpy, st->window, &st->xgwa);
60 st->columns = (st->xgwa.width - st->hspace - st->hspace) / st->char_width;
61 st->rows = (st->xgwa.height - st->vspace - st->vspace) / st->line_height;
65 /* If the window is stupidly small, just truncate. */
66 if (st->rows < 4) st->rows = 4;
67 if (st->columns < 12) st->columns = 12;
69 if (st->y > st->rows) st->y = st->rows-1;
70 if (st->x > st->columns) st->x = st->columns-2;
72 if (st->right > st->columns) st->right = st->columns;
73 if (st->left > st->columns-20) st->left = st->columns-20;
74 if (st->left < 0) st->left = 0;
76 XClearWindow (st->dpy, st->window);
81 xjack_init (Display *dpy, Window window)
83 struct state *st = (struct state *) calloc (1, sizeof(*st));
90 st->delay = get_integer_resource (st->dpy, "delay", "Integer");
91 fontname = get_string_resource (st->dpy, "font", "Font");
93 XGetWindowAttributes (st->dpy, st->window, &st->xgwa);
95 if (st->xgwa.width <= 480)
96 fontname = "-*-courier-medium-r-*-*-*-180-*-*-m-*-*-*";
98 st->font = load_font_retry (st->dpy, fontname);
99 if (!st->font) abort();
101 gcv.font = st->font->fid;
102 gcv.foreground = get_pixel_resource (st->dpy, st->xgwa.colormap,
103 "foreground", "Foreground");
104 gcv.background = get_pixel_resource (st->dpy, st->xgwa.colormap,
105 "background", "Background");
106 st->gc = XCreateGC (st->dpy, st->window,
107 (GCFont | GCForeground | GCBackground), &gcv);
111 ? st->font->per_char['n'-st->font->min_char_or_byte2].rbearing
112 : st->font->min_bounds.rbearing);
113 st->line_height = st->font->ascent + st->font->descent + 1;
115 xjack_reshape (dpy, window, st, st->xgwa.width, st->xgwa.height);
117 st->left = 0xFF & (random() % ((st->columns / 2)+1));
118 st->right = st->left + (0xFF & (random() % (st->columns - st->left)
120 if (st->right < st->left + 10) st->right = st->left + 10;
121 if (st->right > st->columns) st->right = st->columns;
126 if (st->xgwa.width > 200 && st->xgwa.height > 200)
127 st->hspace = st->vspace = 40;
133 xjack_scroll (struct state *st)
136 if (st->subscrolling)
138 int inc = st->line_height / 7;
139 XCopyArea (st->dpy, st->window, st->window, st->gc,
141 st->xgwa.width, st->xgwa.height - inc,
144 /* See? It's OK. He saw it on the television. */
145 XClearArea (st->dpy, st->window,
146 0, st->xgwa.height - inc, st->xgwa.width, inc,
149 st->subscrolling -= inc;
150 if (st->subscrolling <= 0)
151 st->subscrolling = 0;
152 if (st->subscrolling == 0)
154 if (st->scrolling > 0)
158 return st->delay / 1000;
160 else if (st->scrolling)
161 st->subscrolling = st->line_height;
165 else if (st->y >= st->rows-1)
172 xjack_pine (struct state *st)
174 /* See also http://catalog.com/hopkins/unix-haters/login.html */
175 const char *n1 = "NFS server overlook not responding, still trying...";
176 const char *n2 = "NFS server overlook ok.";
177 int prev = st->pining;
180 st->pining = 1 + (random() % 3);
185 XDrawString (st->dpy, st->window, st->gc,
186 (st->x * st->char_width) + st->hspace,
187 ((st->y * st->line_height) + st->vspace
191 if (st->x >= st->columns) st->x = 0, st->y++;
201 XDrawString (st->dpy, st->window, st->gc,
202 (st->x * st->char_width) + st->hspace,
203 ((st->y * st->line_height) + st->vspace
207 if (st->x >= st->columns) st->x = 0, st->y++;
216 xjack_draw (Display *dpy, Window window, void *closure)
218 struct state *st = (struct state *) closure;
219 int this_delay = st->delay;
224 return xjack_scroll (st);
226 return xjack_pine (st);
228 for (s2 = st->s; *s2 && *s2 != ' '; s2++)
231 if (st->break_para ||
233 (st->x + word_length) >= st->right))
243 if (st->mode == 1 || st->mode == 2)
245 /* 1 = left margin goes southwest; 2 = southeast */
246 st->left += (st->mode == 1 ? 1 : -1);
247 if (st->left >= st->right - 10)
249 if ((st->right < (st->columns - 10)) && (random() & 1))
250 st->right += (0xFF & (random() % (st->columns - st->right)));
254 else if (st->left <= 0)
260 else if (st->mode == 3 || st->mode == 4)
262 /* 3 = right margin goes southeast; 4 = southwest */
263 st->right += (st->mode == 3 ? 1 : -1);
264 if (st->right >= st->columns)
266 st->right = st->columns;
269 else if (st->right <= st->left + 10)
273 if (st->y >= st->rows-1) /* bottom of page */
275 # if 0 /* Nah, this looks bad. */
277 /* scroll by 1-5 lines */
278 st->scrolling = (random() % 5 ? 0 : (0xFF & (random() % 5))) + 1;
283 /* but sometimes scroll by a whole page */
284 if (0 == (random() % 100))
285 st->scrolling += st->rows;
290 return xjack_scroll (st);
297 int xshift = 0, yshift = 0;
298 if (0 == random() % 50)
300 xshift = random() % ((st->char_width / 3) + 1); /* mis-strike */
301 yshift = random() % ((st->line_height / 6) + 1);
302 if (0 == (random() % 3))
310 if (0 == (random() % 250)) /* introduce adjascent-key typo */
312 static const char * const typo[] = {
313 "asqw", "ASQW", "bgvhn", "cxdfv", "dserfcx", "ewsdrf",
314 "Jhuikmn", "kjiol,m", "lkop;.,", "mnjk,", "nbhjm", "oiklp09",
315 "pol;(-0", "redft54", "sawedxz", "uyhji87", "wqase32",
316 "yuhgt67", ".,l;/", 0 };
318 while (typo[i] && typo[i][0] != c)
321 c = typo[i][0xFF & ((random() % strlen(typo[i]+1)) + 1)];
325 if (c >= 'a' && c <= 'z' && (st->caps || 0 == (random() % 350)))
328 if (c == 'O' && random() & 1)
333 XDrawString (st->dpy, st->window, st->gc,
334 (st->x * st->char_width) + st->hspace + xshift,
335 ((st->y * st->line_height) + st->vspace + st->font->ascent
338 if (xshift == 0 && yshift == 0 && (0 == (random() & 3000)))
347 if ((tolower(c) != tolower(*st->s))
348 ? (0 == (random() % 10)) /* backup to correct */
349 : (0 == (random() % 400))) /* fail to advance */
354 st->delay += (0xFFFF & (st->delay +
355 (random() % (st->delay * 10))));
362 if (0 == random() % 200)
364 if (random() & 1 && st->s != source)
365 st->s--; /* duplicate character */
367 st->s++; /* skip character */
373 st->caps = (0 == random() % 40); /* capitalize sentence */
375 if (0 == (random() % 10) || /* randomly break paragraph */
377 ((0 == (random() % 10)) || st->sentences > 20)))
379 st->break_para = True;
383 if (random() & 1) /* mode=0 50% of the time */
386 st->mode = (0xFF & (random() % 5));
388 if (0 == (random() % 2)) /* re-pick margins */
390 st->left = 0xFF & (random() % (st->columns / 3));
391 st->right = (st->columns -
392 (0xFF & (random() % (st->columns / 3))));
394 if (0 == random() % 3) /* sometimes be wide */
395 st->right = st->left + ((st->right - st->left) / 2);
398 if (st->right - st->left <= 10) /* introduce sanity */
401 st->right = st->columns;
404 if (st->right - st->left > 50) /* if wide, shrink and move */
406 st->left += (0xFF & (random() % ((st->columns - 50) + 1)));
407 st->right = st->left + (0xFF & ((random() % 40) + 10));
412 st->right - st->left < 25 &&
416 if (st->right > st->columns)
417 st->left -= (st->right - st->columns);
420 if (st->right - st->left < 5)
421 st->left = st->right - 5;
424 if (st->right - st->left < 5)
425 st->right = st->left + 5;
432 if (0 == random() % 3)
433 this_delay += (0xFFFFFF & ((random() % (st->delay * 5)) + 1));
436 this_delay += (0xFFFFFF & ((random() % (st->delay * 15)) + 1));
440 (0 == (random() % 1000)) &&
442 return xjack_pine (st);
448 xjack_event (Display *dpy, Window window, void *closure, XEvent *event)
450 struct state *st = (struct state *) closure;
451 if (event->xany.type == ButtonPress)
461 xjack_free (Display *dpy, Window window, void *closure)
463 struct state *st = (struct state *) closure;
468 static const char *xjack_defaults [] = {
469 ".background: #FFF0B4",
470 ".foreground: #000000",
473 ".font: American Typewriter 24",
475 ".font: -*-courier-medium-r-*-*-*-240-*-*-m-*-*-*",
481 static XrmOptionDescRec xjack_options [] = {
482 { "-delay", ".delay", XrmoptionSepArg, 0 },
483 { "-font", ".font", XrmoptionSepArg, 0 },
487 XSCREENSAVER_MODULE ("XJack", xjack)