Use W_KeycodeToKeysym instead of XkbKeycodeToKeysym
[wmaker-crm.git] / util / setstyle.c
blob6308c0b384d1aadf961ac9242be873fd553cccac
1 /* setstyle.c - loads style related options to wmaker
3 * WindowMaker window manager
5 * Copyright (c) 1997-2003 Alfredo K. Kojima
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 #ifdef __GLIBC__
23 #define _GNU_SOURCE /* getopt_long */
24 #endif
26 #include "config.h"
28 #include <sys/stat.h>
30 #include <getopt.h>
31 #include <limits.h>
32 #include <stdio.h>
33 #include <stdlib.h>
34 #include <string.h>
35 #include <strings.h>
36 #include <unistd.h>
37 #include <X11/Xlib.h>
39 #ifdef HAVE_STDNORETURN
40 #include <stdnoreturn.h>
41 #endif
43 #include <WINGs/WUtil.h>
45 #include "../src/wconfig.h"
47 #include "common.h"
49 #define MAX_OPTIONS 128
51 char *FontOptions[] = {
52 "IconTitleFont",
53 "ClipTitleFont",
54 "LargeDisplayFont",
55 "MenuTextFont",
56 "MenuTitleFont",
57 "WindowTitleFont",
58 NULL
61 char *CursorOptions[] = {
62 "NormalCursor",
63 "ArrowCursor",
64 "MoveCursor",
65 "ResizeCursor",
66 "TopLeftResizeCursor",
67 "TopRightResizeCursor",
68 "BottomLeftResizeCursor",
69 "BottomRightResizeCursor",
70 "VerticalResizeCursor",
71 "HorizontalResizeCursor",
72 "WaitCursor",
73 "QuestionCursor",
74 "TextCursor",
75 "SelectCursor",
76 NULL
79 static const char *prog_name;
80 int ignoreFonts = 0;
81 int ignoreCursors = 0;
83 Display *dpy;
86 static Bool isCursorOption(const char *option)
88 int i;
90 for (i = 0; CursorOptions[i] != NULL; i++) {
91 if (strcasecmp(option, CursorOptions[i]) == 0) {
92 return True;
96 return False;
99 static Bool isFontOption(const char *option)
101 int i;
103 for (i = 0; FontOptions[i] != NULL; i++) {
104 if (strcasecmp(option, FontOptions[i]) == 0) {
105 return True;
109 return False;
113 * finds elements in `texture' that reference external files,
114 * prepends `prefix' to these files. `prefix' is a path component
115 * that qualifies the external references to be absolute, possibly
116 * pending further expansion
118 static void hackPathInTexture(WMPropList * texture, const char *prefix)
120 WMPropList *type;
121 char *t;
123 /* get texture type */
124 type = WMGetFromPLArray(texture, 0);
125 t = WMGetFromPLString(type);
126 if (t == NULL)
127 return;
129 if (strcasecmp(t, "tpixmap") == 0 ||
130 strcasecmp(t, "spixmap") == 0 ||
131 strcasecmp(t, "mpixmap") == 0 ||
132 strcasecmp(t, "cpixmap") == 0 ||
133 strcasecmp(t, "fpixmap") == 0 ||
134 strcasecmp(t, "tvgradient") == 0 ||
135 strcasecmp(t, "thgradient") == 0 ||
136 strcasecmp(t, "tdgradient") == 0) {
137 WMPropList *file;
138 char buffer[4018];
140 /* get pixmap file path */
141 file = WMGetFromPLArray(texture, 1);
142 sprintf(buffer, "%s/%s", prefix, WMGetFromPLString(file));
143 /* replace path with full path */
144 WMDeleteFromPLArray(texture, 1);
145 WMInsertInPLArray(texture, 1, WMCreatePLString(buffer));
147 } else if (strcasecmp(t, "bitmap") == 0) {
148 WMPropList *file;
149 char buffer[4018];
151 /* get bitmap file path */
152 file = WMGetFromPLArray(texture, 1);
153 sprintf(buffer, "%s/%s", prefix, WMGetFromPLString(file));
154 /* replace path with full path */
155 WMDeleteFromPLArray(texture, 1);
156 WMInsertInPLArray(texture, 1, WMCreatePLString(buffer));
158 /* get mask file path */
159 file = WMGetFromPLArray(texture, 2);
160 sprintf(buffer, "%s/%s", prefix, WMGetFromPLString(file));
161 /* replace path with full path */
162 WMDeleteFromPLArray(texture, 2);
163 WMInsertInPLArray(texture, 2, WMCreatePLString(buffer));
167 static void hackPaths(WMPropList * style, const char *prefix)
169 WMPropList *keys;
170 WMPropList *key;
171 WMPropList *value;
172 int i;
174 keys = WMGetPLDictionaryKeys(style);
176 for (i = 0; i < WMGetPropListItemCount(keys); i++) {
177 key = WMGetFromPLArray(keys, i);
179 value = WMGetFromPLDictionary(style, key);
180 if (!value)
181 continue;
183 if (strcasecmp(WMGetFromPLString(key), "WorkspaceSpecificBack") == 0) {
184 if (WMIsPLArray(value)) {
185 int j;
186 WMPropList *texture;
188 for (j = 0; j < WMGetPropListItemCount(value); j++) {
189 texture = WMGetFromPLArray(value, j);
191 if (texture && WMIsPLArray(texture)
192 && WMGetPropListItemCount(texture) > 2) {
194 hackPathInTexture(texture, prefix);
198 } else {
200 if (WMIsPLArray(value) && WMGetPropListItemCount(value) > 2) {
202 hackPathInTexture(value, prefix);
207 WMReleasePropList(keys);
210 static WMPropList *getColor(WMPropList * texture)
212 WMPropList *value, *type;
213 char *str;
215 type = WMGetFromPLArray(texture, 0);
216 if (!type)
217 return NULL;
219 value = NULL;
221 str = WMGetFromPLString(type);
222 if (strcasecmp(str, "solid") == 0) {
223 value = WMGetFromPLArray(texture, 1);
224 } else if (strcasecmp(str, "dgradient") == 0
225 || strcasecmp(str, "hgradient") == 0 || strcasecmp(str, "vgradient") == 0) {
226 WMPropList *c1, *c2;
227 int r1, g1, b1, r2, g2, b2;
228 char buffer[32];
230 c1 = WMGetFromPLArray(texture, 1);
231 c2 = WMGetFromPLArray(texture, 2);
232 if (!dpy) {
233 if (sscanf(WMGetFromPLString(c1), "#%2x%2x%2x", &r1, &g1, &b1) == 3
234 && sscanf(WMGetFromPLString(c2), "#%2x%2x%2x", &r2, &g2, &b2) == 3) {
235 sprintf(buffer, "#%02x%02x%02x", (r1 + r2) / 2, (g1 + g2) / 2, (b1 + b2) / 2);
236 value = WMCreatePLString(buffer);
237 } else {
238 value = c1;
240 } else {
241 XColor color1;
242 XColor color2;
244 XParseColor(dpy, DefaultColormap(dpy, DefaultScreen(dpy)), WMGetFromPLString(c1), &color1);
245 XParseColor(dpy, DefaultColormap(dpy, DefaultScreen(dpy)), WMGetFromPLString(c2), &color2);
247 sprintf(buffer, "#%02x%02x%02x",
248 (color1.red + color2.red) >> 9,
249 (color1.green + color2.green) >> 9, (color1.blue + color2.blue) >> 9);
250 value = WMCreatePLString(buffer);
252 } else if (strcasecmp(str, "mdgradient") == 0
253 || strcasecmp(str, "mhgradient") == 0 || strcasecmp(str, "mvgradient") == 0) {
255 value = WMGetFromPLArray(texture, 1);
257 } else if (strcasecmp(str, "tpixmap") == 0
258 || strcasecmp(str, "cpixmap") == 0 || strcasecmp(str, "spixmap") == 0) {
260 value = WMGetFromPLArray(texture, 2);
263 return value;
267 * since some of the options introduce incompatibilities, we will need
268 * to do a kluge here or the themes ppl will get real annoying.
269 * So, treat for the absence of the following options:
270 * IconTitleColor
271 * IconTitleBack
273 static void hackStyle(WMPropList * style)
275 WMPropList *keys, *tmp;
276 int foundIconTitle = 0, foundResizebarBack = 0;
277 int i;
279 keys = WMGetPLDictionaryKeys(style);
281 for (i = 0; i < WMGetPropListItemCount(keys); i++) {
282 char *str;
284 tmp = WMGetFromPLArray(keys, i);
285 str = WMGetFromPLString(tmp);
286 if (str) {
287 if (ignoreFonts && isFontOption(str)) {
288 WMRemoveFromPLDictionary(style, tmp);
289 continue;
291 if (ignoreCursors && isCursorOption(str)) {
292 WMRemoveFromPLDictionary(style, tmp);
293 continue;
295 if (isFontOption(str)) {
296 WMPropList *value;
297 char *newfont, *oldfont;
299 value = WMGetFromPLDictionary(style, tmp);
300 if (value) {
301 oldfont = WMGetFromPLString(value);
302 newfont = convertFont(oldfont, False);
303 if (newfont != oldfont) {
304 value = WMCreatePLString(newfont);
305 WMPutInPLDictionary(style, tmp, value);
306 WMReleasePropList(value);
307 wfree(newfont);
311 if (strcasecmp(str, "IconTitleColor") == 0 || strcasecmp(str, "IconTitleBack") == 0) {
312 foundIconTitle = 1;
313 } else if (strcasecmp(str, "ResizebarBack") == 0) {
314 foundResizebarBack = 1;
318 WMReleasePropList(keys);
320 if (!foundIconTitle) {
321 /* set the default values */
322 tmp = WMGetFromPLDictionary(style, WMCreatePLString("FTitleColor"));
323 if (tmp) {
324 WMPutInPLDictionary(style, WMCreatePLString("IconTitleColor"), tmp);
327 tmp = WMGetFromPLDictionary(style, WMCreatePLString("FTitleBack"));
328 if (tmp) {
329 WMPropList *value;
331 value = getColor(tmp);
333 if (value) {
334 WMPutInPLDictionary(style, WMCreatePLString("IconTitleBack"), value);
339 if (!foundResizebarBack) {
340 /* set the default values */
341 tmp = WMGetFromPLDictionary(style, WMCreatePLString("UTitleBack"));
342 if (tmp) {
343 WMPropList *value;
345 value = getColor(tmp);
347 if (value) {
348 WMPropList *t;
350 t = WMCreatePLArray(WMCreatePLString("solid"), value, NULL);
351 WMPutInPLDictionary(style, WMCreatePLString("ResizebarBack"), t);
356 if (!WMGetFromPLDictionary(style, WMCreatePLString("MenuStyle"))) {
357 WMPutInPLDictionary(style, WMCreatePLString("MenuStyle"), WMCreatePLString("normal"));
361 static noreturn void print_help(int print_usage, int exitval)
363 printf("Usage: %s [OPTIONS] FILE\n", prog_name);
364 if (print_usage) {
365 puts("Reads style/theme configuration from FILE and updates Window Maker.");
366 puts("");
367 puts(" --no-fonts ignore font related options");
368 puts(" --no-cursors ignore cursor related options");
369 puts(" --ignore <option> ignore changes in the specified option");
370 puts(" -h, --help display this help and exit");
371 puts(" -v, --version output version information and exit");
373 exit(exitval);
376 int main(int argc, char **argv)
378 WMPropList *prop, *style;
379 char *path;
380 char *file = NULL;
381 struct stat st;
382 int i, ch, ignflag = 0;
383 int ignoreCount = 0;
384 char *ignoreList[MAX_OPTIONS];
385 XEvent ev;
387 struct option longopts[] = {
388 { "version", no_argument, NULL, 'v' },
389 { "help", no_argument, NULL, 'h' },
390 { "no-fonts", no_argument, &ignoreFonts, 1 },
391 { "no-cursors", no_argument, &ignoreCursors, 1 },
392 { "ignore", required_argument, &ignflag, 1 },
393 { NULL, 0, NULL, 0 }
396 prog_name = argv[0];
397 while ((ch = getopt_long(argc, argv, "hv", longopts, NULL)) != -1)
398 switch(ch) {
399 case 'v':
400 printf("%s (Window Maker %s)\n", prog_name, VERSION);
401 return 0;
402 /* NOTREACHED */
403 case 'h':
404 print_help(1, 0);
405 /* NOTREACHED */
406 case 0:
407 if (ignflag) {
408 if (ignoreCount >= MAX_OPTIONS) {
409 printf("Maximum %d `ignore' arguments\n", MAX_OPTIONS);
410 return 1;
412 ignoreList[ignoreCount++] = optarg;
413 ignflag = 0;
415 break;
416 default:
417 print_help(0, 1);
418 /* NOTREACHED */
421 argc -= optind;
422 argv += optind;
424 if (argc != 1)
425 print_help(0, 1);
427 file = argv[0];
429 WMPLSetCaseSensitive(False);
431 path = wdefaultspathfordomain("WindowMaker");
433 prop = WMReadPropListFromFile(path);
434 if (!prop) {
435 perror(path);
436 printf("%s: could not load WindowMaker configuration file.\n", prog_name);
437 return 1;
440 if (stat(file, &st) < 0) {
441 perror(file);
442 return 1;
444 if (S_ISDIR(st.st_mode)) { /* theme pack */
445 char buf[PATH_MAX];
446 char *homedir;
448 if (realpath(file, buf) == NULL) {
449 perror(file);
450 return 1;
452 strncat(buf, "/style", sizeof(buf) - strlen(buf) - 1);
454 if (stat(buf, &st) != 0 || !S_ISREG(st.st_mode)) { /* maybe symlink too? */
455 printf("%s: %s: style file not found or not a file\n", prog_name, buf);
456 return 1;
459 style = WMReadPropListFromFile(buf);
460 if (!style) {
461 perror(buf);
462 printf("%s: could not load style file.\n", prog_name);
463 return 1;
466 buf[strlen(buf) - 6 /* strlen("/style") */] = '\0';
467 homedir = wstrdup(wgethomedir());
468 if (strlen(homedir) > 1 && /* this is insane, wgethomedir() returns `/' on error */
469 strncmp(homedir, buf, strlen(homedir)) == 0) {
470 /* theme pack is under ${HOME}; exchange ${HOME} part
471 * for `~' so it gets portable references to the user home dir */
472 *buf = '~';
473 memmove(buf + 1, buf + strlen(homedir), strlen(buf) - strlen(homedir) + 1);
475 wfree(homedir);
477 hackPaths(style, buf); /* this will prefix pixmaps in the style
478 * with absolute(ish) references */
480 } else { /* normal style file */
482 style = WMReadPropListFromFile(file);
483 if (!style) {
484 perror(file);
485 printf("%s:could not load style file.\n", prog_name);
486 return 1;
490 if (!WMIsPLDictionary(style)) {
491 printf("%s: '%s' is not a style file/theme\n", prog_name, file);
492 return 1;
495 hackStyle(style);
497 if (ignoreCount > 0) {
498 for (i = 0; i < ignoreCount; i++) {
499 WMRemoveFromPLDictionary(style, WMCreatePLString(ignoreList[i]));
503 WMMergePLDictionaries(prop, style, True);
505 WMWritePropListToFile(prop, path);
507 dpy = XOpenDisplay("");
508 if (dpy) {
509 memset(&ev, 0, sizeof(XEvent));
511 ev.xclient.type = ClientMessage;
512 ev.xclient.message_type = XInternAtom(dpy, "_WINDOWMAKER_COMMAND", False);
513 ev.xclient.window = DefaultRootWindow(dpy);
514 ev.xclient.format = 8;
515 strncpy(ev.xclient.data.b, "Reconfigure", sizeof(ev.xclient.data.b));
517 XSendEvent(dpy, DefaultRootWindow(dpy), False, SubstructureRedirectMask, &ev);
518 XFlush(dpy);
521 return 0;