]> icculus.org git repositories - dana/obconf.git/blob - src/tree.c
create new nodes with attributes correctly
[dana/obconf.git] / src / tree.c
1 /* -*- indent-tabs-mode: nil; tab-width: 4; c-basic-offset: 4; -*-
2
3    tree.c for ObConf, the configuration tool for Openbox
4    Copyright (c) 2003-2007   Dana Jansens
5
6    This program is free software; you can redistribute it and/or modify
7    it under the terms of the GNU General Public License as published by
8    the Free Software Foundation; either version 2 of the License, or
9    (at your option) any later version.
10
11    This program is distributed in the hope that it will be useful,
12    but WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14    GNU General Public License for more details.
15
16    See the COPYING file for a copy of the GNU General Public License.
17 */
18
19 #include "tree.h"
20 #include "main.h"
21 #include "openbox/parse.h"
22
23 #include <sys/types.h>
24 #include <signal.h>
25
26 xmlNodePtr tree_get_node(const gchar *path, const gchar *def)
27 {
28     xmlNodePtr n, c;
29     gchar **nodes;
30     gchar **it, **next;
31
32     n = root;
33
34     nodes = g_strsplit(path, "/", 0);
35     for (it = nodes; *it; it = next) {
36         gchar **attrs;
37         gboolean ok = FALSE;
38
39         attrs = g_strsplit(*it, ":", 0);
40         next = it + 1;
41
42         /* match attributes */
43         c = parse_find_node(attrs[0], n->children);
44         while (c && !ok) {
45             gint i;
46
47             ok = TRUE;
48             for (i = 1; attrs[i]; ++i) {
49                 gchar **eq = g_strsplit(attrs[i], "=", 2);
50                 if (eq[1] && !parse_attr_contains(eq[1], c, eq[0]))
51                     ok = FALSE;
52                 g_strfreev(eq);
53             }
54             if (!ok)
55                 c = parse_find_node(attrs[0], c->next);
56         }
57
58         if (!c) {
59             gint i;
60             gchar **attrs;
61
62             attrs = g_strsplit(*it, ":", 0);
63
64             c = xmlNewTextChild(n, NULL, *it, *next ? NULL : attrs[0]);
65
66             for (i = 1; attrs[i]; ++i) {
67                 gchar **eq = g_strsplit(attrs[i], "=", 2);
68                 if (eq[1])
69                     xmlNewProp(c, eq[0], eq[1]);
70                 g_strfreev(eq);
71             }
72         }
73         n = c;
74
75         g_strfreev(attrs);
76     }
77
78     g_strfreev(nodes);
79
80     return n;
81 }
82
83 void tree_apply()
84 {
85     gchar *p;
86     gboolean err;
87
88     p = g_build_filename(parse_xdg_config_home_path(), "openbox", NULL);
89     parse_mkdir_path(p, 0700);
90     g_free(p);
91
92     p = g_build_filename(parse_xdg_config_home_path(), "openbox",
93                          "rc.xml", NULL);
94     err = xmlSaveFormatFile(p, doc, 1) == -1;
95     if (err) {
96         gchar *s;
97         s = g_strdup_printf("An error occured while saving the "
98                             "config file '%s'", p);
99         obconf_error(s);
100         g_free(s);
101     }
102     g_free(p);
103
104     if (!err) {
105         GdkAtom type;
106         gint format;
107         gint length;
108         guint *pid;
109
110         if (gdk_property_get
111             (gdk_screen_get_root_window(gdk_screen_get_default()),
112              gdk_atom_intern("_OPENBOX_PID", FALSE),
113              gdk_atom_intern("CARDINAL", FALSE),
114              0, 4, FALSE, &type, &format, &length, (guchar**)&pid)) {
115             kill(*pid, SIGUSR2);
116             g_free(pid);
117         }
118     }
119 }
120
121 void tree_set_string(const gchar *node, const gchar *value)
122 {
123     xmlNodePtr n;
124
125     n = tree_get_node(node, NULL);
126     xmlNodeSetContent(n, (const xmlChar*) value);
127
128     tree_apply();
129 }
130
131 void tree_set_int(const gchar *node, const gint value)
132 {
133     xmlNodePtr n;
134     gchar *s;
135
136     n = tree_get_node(node, NULL);
137     s = g_strdup_printf("%d", value);
138     xmlNodeSetContent(n, (const xmlChar*) s);
139     g_free(s);
140
141     tree_apply();
142 }
143
144 void tree_set_bool(const gchar *node, const gboolean value)
145 {
146     xmlNodePtr n;
147
148     n = tree_get_node(node, NULL);
149     xmlNodeSetContent(n, (const xmlChar*) (value ? "yes" : "no"));
150
151     tree_apply();
152 }
153
154 gchar* tree_get_string(const gchar *node, const gchar *def)
155 {
156     xmlNodePtr n;
157
158     n = tree_get_node(node, def);
159     return parse_string(doc, n);
160 }
161
162 gint tree_get_int(const gchar *node, gint def)
163 {
164     xmlNodePtr n;
165     gchar *d;
166
167     d = g_strdup_printf("%d", def);
168     n = tree_get_node(node, d);
169     g_free(d);
170     return parse_int(doc, n);
171 }
172
173 gboolean tree_get_bool(const gchar *node, gboolean def)
174 {
175     xmlNodePtr n;
176
177     n = tree_get_node(node, (def ? "yes" : "no"));
178     return parse_bool(doc, n);
179 }