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.
23 #define _GNU_SOURCE /* getopt_long */
39 #ifdef HAVE_STDNORETURN
40 #include <stdnoreturn.h>
43 #include <WINGs/WUtil.h>
45 #include "../src/wconfig.h"
49 #define MAX_OPTIONS 128
51 char *FontOptions
[] = {
61 char *CursorOptions
[] = {
66 "TopLeftResizeCursor",
67 "TopRightResizeCursor",
68 "BottomLeftResizeCursor",
69 "BottomRightResizeCursor",
70 "VerticalResizeCursor",
71 "HorizontalResizeCursor",
79 static const char *prog_name
;
81 int ignoreCursors
= 0;
86 static Bool
isCursorOption(const char *option
)
90 for (i
= 0; CursorOptions
[i
] != NULL
; i
++) {
91 if (strcasecmp(option
, CursorOptions
[i
]) == 0) {
99 static Bool
isFontOption(const char *option
)
103 for (i
= 0; FontOptions
[i
] != NULL
; i
++) {
104 if (strcasecmp(option
, FontOptions
[i
]) == 0) {
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
)
123 /* get texture type */
124 type
= WMGetFromPLArray(texture
, 0);
125 t
= WMGetFromPLString(type
);
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) {
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) {
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
)
174 keys
= WMGetPLDictionaryKeys(style
);
176 for (i
= 0; i
< WMGetPropListItemCount(keys
); i
++) {
177 key
= WMGetFromPLArray(keys
, i
);
179 value
= WMGetFromPLDictionary(style
, key
);
183 if (strcasecmp(WMGetFromPLString(key
), "WorkspaceSpecificBack") == 0) {
184 if (WMIsPLArray(value
)) {
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
);
200 if (WMIsPLArray(value
) && WMGetPropListItemCount(value
) > 2) {
202 hackPathInTexture(value
, prefix
);
207 WMReleasePropList(keys
);
210 static WMPropList
*getColor(WMPropList
* texture
)
212 WMPropList
*value
, *type
;
215 type
= WMGetFromPLArray(texture
, 0);
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) {
227 int r1
, g1
, b1
, r2
, g2
, b2
;
230 c1
= WMGetFromPLArray(texture
, 1);
231 c2
= WMGetFromPLArray(texture
, 2);
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
);
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);
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:
273 static void hackStyle(WMPropList
* style
)
275 WMPropList
*keys
, *tmp
;
276 int foundIconTitle
= 0, foundResizebarBack
= 0;
279 keys
= WMGetPLDictionaryKeys(style
);
281 for (i
= 0; i
< WMGetPropListItemCount(keys
); i
++) {
284 tmp
= WMGetFromPLArray(keys
, i
);
285 str
= WMGetFromPLString(tmp
);
287 if (ignoreFonts
&& isFontOption(str
)) {
288 WMRemoveFromPLDictionary(style
, tmp
);
291 if (ignoreCursors
&& isCursorOption(str
)) {
292 WMRemoveFromPLDictionary(style
, tmp
);
295 if (isFontOption(str
)) {
297 char *newfont
, *oldfont
;
299 value
= WMGetFromPLDictionary(style
, tmp
);
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
);
311 if (strcasecmp(str
, "IconTitleColor") == 0 || strcasecmp(str
, "IconTitleBack") == 0) {
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"));
324 WMPutInPLDictionary(style
, WMCreatePLString("IconTitleColor"), tmp
);
327 tmp
= WMGetFromPLDictionary(style
, WMCreatePLString("FTitleBack"));
331 value
= getColor(tmp
);
334 WMPutInPLDictionary(style
, WMCreatePLString("IconTitleBack"), value
);
339 if (!foundResizebarBack
) {
340 /* set the default values */
341 tmp
= WMGetFromPLDictionary(style
, WMCreatePLString("UTitleBack"));
345 value
= getColor(tmp
);
350 t
= WMCreatePLArray(WMCreatePLString("solid"), value
, NULL
);
351 WMPutInPLDictionary(style
, WMCreatePLString("ResizebarBack"), t
);
352 WMReleasePropList(value
);
357 if (!WMGetFromPLDictionary(style
, WMCreatePLString("MenuStyle"))) {
358 WMPutInPLDictionary(style
, WMCreatePLString("MenuStyle"), WMCreatePLString("normal"));
362 static noreturn
void print_help(int print_usage
, int exitval
)
364 printf("Usage: %s [OPTIONS] FILE\n", prog_name
);
366 puts("Reads style/theme configuration from FILE and updates Window Maker.");
368 puts(" --no-fonts ignore font related options");
369 puts(" --no-cursors ignore cursor related options");
370 puts(" --ignore <option> ignore changes in the specified option");
371 puts(" -h, --help display this help and exit");
372 puts(" -v, --version output version information and exit");
377 int main(int argc
, char **argv
)
379 WMPropList
*prop
, *style
;
383 int i
, ch
, ignflag
= 0;
385 char *ignoreList
[MAX_OPTIONS
];
388 struct option longopts
[] = {
389 { "version", no_argument
, NULL
, 'v' },
390 { "help", no_argument
, NULL
, 'h' },
391 { "no-fonts", no_argument
, &ignoreFonts
, 1 },
392 { "no-cursors", no_argument
, &ignoreCursors
, 1 },
393 { "ignore", required_argument
, &ignflag
, 1 },
398 while ((ch
= getopt_long(argc
, argv
, "hv", longopts
, NULL
)) != -1)
401 printf("%s (Window Maker %s)\n", prog_name
, VERSION
);
409 if (ignoreCount
>= MAX_OPTIONS
) {
410 printf("Maximum %d `ignore' arguments\n", MAX_OPTIONS
);
413 ignoreList
[ignoreCount
++] = optarg
;
430 WMPLSetCaseSensitive(False
);
432 path
= wdefaultspathfordomain("WindowMaker");
434 prop
= WMReadPropListFromFile(path
);
437 printf("%s: could not load WindowMaker configuration file.\n", prog_name
);
441 if (stat(file
, &st
) < 0) {
445 if (S_ISDIR(st
.st_mode
)) { /* theme pack */
449 if (realpath(file
, buf
) == NULL
) {
453 strncat(buf
, "/style", sizeof(buf
) - strlen(buf
) - 1);
455 if (stat(buf
, &st
) != 0 || !S_ISREG(st
.st_mode
)) { /* maybe symlink too? */
456 printf("%s: %s: style file not found or not a file\n", prog_name
, buf
);
460 style
= WMReadPropListFromFile(buf
);
463 printf("%s: could not load style file.\n", prog_name
);
467 buf
[strlen(buf
) - 6 /* strlen("/style") */] = '\0';
468 homedir
= wstrdup(wgethomedir());
469 if (strlen(homedir
) > 1 && /* this is insane, wgethomedir() returns `/' on error */
470 strncmp(homedir
, buf
, strlen(homedir
)) == 0) {
471 /* theme pack is under ${HOME}; exchange ${HOME} part
472 * for `~' so it gets portable references to the user home dir */
474 memmove(buf
+ 1, buf
+ strlen(homedir
), strlen(buf
) - strlen(homedir
) + 1);
478 hackPaths(style
, buf
); /* this will prefix pixmaps in the style
479 * with absolute(ish) references */
481 } else { /* normal style file */
483 style
= WMReadPropListFromFile(file
);
486 printf("%s:could not load style file.\n", prog_name
);
491 if (!WMIsPLDictionary(style
)) {
492 printf("%s: '%s' is not a style file/theme\n", prog_name
, file
);
498 if (ignoreCount
> 0) {
499 for (i
= 0; i
< ignoreCount
; i
++) {
500 WMRemoveFromPLDictionary(style
, WMCreatePLString(ignoreList
[i
]));
504 WMMergePLDictionaries(prop
, style
, True
);
506 WMWritePropListToFile(prop
, path
);
508 dpy
= XOpenDisplay("");
510 memset(&ev
, 0, sizeof(XEvent
));
512 ev
.xclient
.type
= ClientMessage
;
513 ev
.xclient
.message_type
= XInternAtom(dpy
, "_WINDOWMAKER_COMMAND", False
);
514 ev
.xclient
.window
= DefaultRootWindow(dpy
);
515 ev
.xclient
.format
= 8;
516 strncpy(ev
.xclient
.data
.b
, "Reconfigure", sizeof(ev
.xclient
.data
.b
));
518 XSendEvent(dpy
, DefaultRootWindow(dpy
), False
, SubstructureRedirectMask
, &ev
);