diff options
-rw-r--r-- | config.h | 9 | ||||
-rw-r--r-- | config.mk | 2 | ||||
-rw-r--r-- | slock.1 | 7 | ||||
-rw-r--r-- | slock.c | 119 |
4 files changed, 132 insertions, 5 deletions
@@ -39,3 +39,12 @@ static const int blurRadius=5; | |||
39 | //#define PIXELATION | 39 | //#define PIXELATION |
40 | /*Set pixelation radius*/ | 40 | /*Set pixelation radius*/ |
41 | static const int pixelSize=0; | 41 | static const int pixelSize=0; |
42 | |||
43 | /* default message */ | ||
44 | static const char * message = "Suckless: Software that sucks less."; | ||
45 | |||
46 | /* text color */ | ||
47 | static const char * text_color = "#ffffff"; | ||
48 | |||
49 | /* text size (must be a valid size) */ | ||
50 | static const char * font_name = "6x10"; | ||
@@ -37,4 +37,4 @@ COMPATSRC = explicit_bzero.c | |||
37 | #COMPATSRC = | 37 | #COMPATSRC = |
38 | 38 | ||
39 | # compiler and linker | 39 | # compiler and linker |
40 | CC = cc | 40 | CC = cc -w |
@@ -6,6 +6,8 @@ | |||
6 | .Sh SYNOPSIS | 6 | .Sh SYNOPSIS |
7 | .Nm | 7 | .Nm |
8 | .Op Fl v | 8 | .Op Fl v |
9 | .Op Fl f | ||
10 | .Op Fl m Ar message | ||
9 | .Op Ar cmd Op Ar arg ... | 11 | .Op Ar cmd Op Ar arg ... |
10 | .Sh DESCRIPTION | 12 | .Sh DESCRIPTION |
11 | .Nm | 13 | .Nm |
@@ -16,6 +18,11 @@ is executed after the screen has been locked. | |||
16 | .Bl -tag -width Ds | 18 | .Bl -tag -width Ds |
17 | .It Fl v | 19 | .It Fl v |
18 | Print version information to stdout and exit. | 20 | Print version information to stdout and exit. |
21 | .It Fl f | ||
22 | List all valid X fonts and exit. | ||
23 | .It Fl m Ar message | ||
24 | Overrides default slock lock message. | ||
25 | .TP | ||
19 | .El | 26 | .El |
20 | .Sh SECURITY CONSIDERATIONS | 27 | .Sh SECURITY CONSIDERATIONS |
21 | To make sure a locked screen can not be bypassed by switching VTs | 28 | To make sure a locked screen can not be bypassed by switching VTs |
@@ -30,6 +30,9 @@ | |||
30 | 30 | ||
31 | char *argv0; | 31 | char *argv0; |
32 | 32 | ||
33 | /* global count to prevent repeated error messages */ | ||
34 | int count_error = 0; | ||
35 | |||
33 | enum { | 36 | enum { |
34 | INIT, | 37 | INIT, |
35 | INPUT, | 38 | INPUT, |
@@ -98,6 +101,98 @@ dontkillme(void) | |||
98 | } | 101 | } |
99 | #endif | 102 | #endif |
100 | 103 | ||
104 | static void | ||
105 | writemessage(Display *dpy, Window win, int screen) | ||
106 | { | ||
107 | int len, line_len, width, height, s_width, s_height, i, j, k, tab_replace, tab_size; | ||
108 | XGCValues gr_values; | ||
109 | XFontStruct *fontinfo; | ||
110 | XColor color, dummy; | ||
111 | XineramaScreenInfo *xsi; | ||
112 | GC gc; | ||
113 | fontinfo = XLoadQueryFont(dpy, font_name); | ||
114 | |||
115 | if (fontinfo == NULL) { | ||
116 | if (count_error == 0) { | ||
117 | fprintf(stderr, "slock: Unable to load font \"%s\"\n", font_name); | ||
118 | fprintf(stderr, "slock: Try listing fonts with 'slock -f'\n"); | ||
119 | count_error++; | ||
120 | } | ||
121 | return; | ||
122 | } | ||
123 | |||
124 | tab_size = 8 * XTextWidth(fontinfo, " ", 1); | ||
125 | |||
126 | XAllocNamedColor(dpy, DefaultColormap(dpy, screen), | ||
127 | text_color, &color, &dummy); | ||
128 | |||
129 | gr_values.font = fontinfo->fid; | ||
130 | gr_values.foreground = color.pixel; | ||
131 | gc=XCreateGC(dpy,win,GCFont+GCForeground, &gr_values); | ||
132 | |||
133 | /* To prevent "Uninitialized" warnings. */ | ||
134 | xsi = NULL; | ||
135 | |||
136 | /* | ||
137 | * Start formatting and drawing text | ||
138 | */ | ||
139 | |||
140 | len = strlen(message); | ||
141 | |||
142 | /* Max max line length (cut at '\n') */ | ||
143 | line_len = 0; | ||
144 | k = 0; | ||
145 | for (i = j = 0; i < len; i++) { | ||
146 | if (message[i] == '\n') { | ||
147 | if (i - j > line_len) | ||
148 | line_len = i - j; | ||
149 | k++; | ||
150 | i++; | ||
151 | j = i; | ||
152 | } | ||
153 | } | ||
154 | /* If there is only one line */ | ||
155 | if (line_len == 0) | ||
156 | line_len = len; | ||
157 | |||
158 | if (XineramaIsActive(dpy)) { | ||
159 | xsi = XineramaQueryScreens(dpy, &i); | ||
160 | s_width = xsi[0].width; | ||
161 | s_height = xsi[0].height; | ||
162 | } else { | ||
163 | s_width = DisplayWidth(dpy, screen); | ||
164 | s_height = DisplayHeight(dpy, screen); | ||
165 | } | ||
166 | |||
167 | height = s_height*3/7 - (k*20)/3; | ||
168 | width = (s_width - XTextWidth(fontinfo, message, line_len))/2; | ||
169 | |||
170 | /* Look for '\n' and print the text between them. */ | ||
171 | for (i = j = k = 0; i <= len; i++) { | ||
172 | /* i == len is the special case for the last line */ | ||
173 | if (i == len || message[i] == '\n') { | ||
174 | tab_replace = 0; | ||
175 | while (message[j] == '\t' && j < i) { | ||
176 | tab_replace++; | ||
177 | j++; | ||
178 | } | ||
179 | |||
180 | XDrawString(dpy, win, gc, width + tab_size*tab_replace, height + 20*k, message + j, i - j); | ||
181 | while (i < len && message[i] == '\n') { | ||
182 | i++; | ||
183 | j = i; | ||
184 | k++; | ||
185 | } | ||
186 | } | ||
187 | } | ||
188 | |||
189 | /* xsi should not be NULL anyway if Xinerama is active, but to be safe */ | ||
190 | if (XineramaIsActive(dpy) && xsi != NULL) | ||
191 | XFree(xsi); | ||
192 | } | ||
193 | |||
194 | |||
195 | |||
101 | static const char * | 196 | static const char * |
102 | gethash(void) | 197 | gethash(void) |
103 | { | 198 | { |
@@ -241,6 +336,7 @@ readpw(Display *dpy, struct xrandr *rr, struct lock **locks, int nscreens, | |||
241 | if (running && oldc != color) { | 336 | if (running && oldc != color) { |
242 | for (screen = 0; screen < nscreens; screen++) { | 337 | for (screen = 0; screen < nscreens; screen++) { |
243 | drawlogo(dpy, locks[screen], color); | 338 | drawlogo(dpy, locks[screen], color); |
339 | writemessage(dpy, locks[screen]->win, screen); | ||
244 | } | 340 | } |
245 | oldc = color; | 341 | oldc = color; |
246 | } | 342 | } |
@@ -385,7 +481,7 @@ lockscreen(Display *dpy, struct xrandr *rr, int screen) | |||
385 | static void | 481 | static void |
386 | usage(void) | 482 | usage(void) |
387 | { | 483 | { |
388 | die("usage: slock [-v] [cmd [arg ...]]\n"); | 484 | die("usage: slock [-v] [-f] [-m message] [cmd [arg ...]]\n"); |
389 | } | 485 | } |
390 | 486 | ||
391 | int | 487 | int |
@@ -398,12 +494,25 @@ main(int argc, char **argv) { | |||
398 | gid_t dgid; | 494 | gid_t dgid; |
399 | const char *hash; | 495 | const char *hash; |
400 | Display *dpy; | 496 | Display *dpy; |
401 | int s, nlocks, nscreens; | 497 | int i, s, nlocks, nscreens; |
498 | int count_fonts; | ||
499 | char **font_names; | ||
402 | 500 | ||
403 | ARGBEGIN { | 501 | ARGBEGIN { |
404 | case 'v': | 502 | case 'v': |
405 | fprintf(stderr, "slock-"VERSION"\n"); | 503 | fprintf(stderr, "slock-"VERSION"\n"); |
406 | return 0; | 504 | return 0; |
505 | case 'm': | ||
506 | message = EARGF(usage()); | ||
507 | break; | ||
508 | case 'f': | ||
509 | if (!(dpy = XOpenDisplay(NULL))) | ||
510 | die("slock: cannot open display\n"); | ||
511 | font_names = XListFonts(dpy, "*", 10000 /* list 10000 fonts*/, &count_fonts); | ||
512 | for (i=0; i<count_fonts; i++) { | ||
513 | fprintf(stderr, "%s\n", *(font_names+i)); | ||
514 | } | ||
515 | return 0; | ||
407 | default: | 516 | default: |
408 | usage(); | 517 | usage(); |
409 | } ARGEND | 518 | } ARGEND |
@@ -502,10 +611,12 @@ main(int argc, char **argv) { | |||
502 | if (!(locks = calloc(nscreens, sizeof(struct lock *)))) | 611 | if (!(locks = calloc(nscreens, sizeof(struct lock *)))) |
503 | die("slock: out of memory\n"); | 612 | die("slock: out of memory\n"); |
504 | for (nlocks = 0, s = 0; s < nscreens; s++) { | 613 | for (nlocks = 0, s = 0; s < nscreens; s++) { |
505 | if ((locks[s] = lockscreen(dpy, &rr, s)) != NULL) | 614 | if ((locks[s] = lockscreen(dpy, &rr, s)) != NULL) { |
615 | writemessage(dpy, locks[s]->win, s); | ||
506 | nlocks++; | 616 | nlocks++; |
507 | else | 617 | } else { |
508 | break; | 618 | break; |
619 | } | ||
509 | } | 620 | } |
510 | XSync(dpy, 0); | 621 | XSync(dpy, 0); |
511 | 622 | ||