| |
1 /* |
| |
2 * gaim |
| |
3 * |
| |
4 * Gaim is the legal property of its developers, whose names are too numerous |
| |
5 * to list here. Please refer to the COPYRIGHT file distributed with this |
| |
6 * source distribution. |
| |
7 * |
| |
8 * This program is free software; you can redistribute it and/or modify |
| |
9 * it under the terms of the GNU General Public License as published by |
| |
10 * the Free Software Foundation; either version 2 of the License, or |
| |
11 * (at your option) any later version. |
| |
12 * |
| |
13 * This program is distributed in the hope that it will be useful, |
| |
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| |
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| |
16 * GNU General Public License for more details. |
| |
17 * |
| |
18 * You should have received a copy of the GNU General Public License |
| |
19 * along with this program; if not, write to the Free Software |
| |
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| |
21 * |
| |
22 */ |
| |
23 |
| |
24 #ifdef HAVE_CONFIG_H |
| |
25 #include <config.h> |
| |
26 #endif |
| |
27 |
| |
28 #include <string.h> |
| |
29 #include <stdio.h> |
| |
30 #include <stdlib.h> |
| |
31 #include <sys/stat.h> |
| |
32 #include <sys/types.h> |
| |
33 #include <glib.h> |
| |
34 #include "internal.h" |
| |
35 #include "prefs.h" |
| |
36 #include "debug.h" |
| |
37 #include "util.h" |
| |
38 |
| |
39 #ifdef _WIN32 |
| |
40 #include "win32dep.h" |
| |
41 #endif |
| |
42 |
| |
43 struct pref_cb { |
| |
44 GaimPrefCallback func; |
| |
45 gpointer data; |
| |
46 guint id; |
| |
47 void *handle; |
| |
48 }; |
| |
49 |
| |
50 /* TODO: This should use GaimValues? */ |
| |
51 struct gaim_pref { |
| |
52 GaimPrefType type; |
| |
53 char *name; |
| |
54 union { |
| |
55 gpointer generic; |
| |
56 gboolean boolean; |
| |
57 int integer; |
| |
58 char *string; |
| |
59 GList *stringlist; |
| |
60 } value; |
| |
61 GSList *callbacks; |
| |
62 struct gaim_pref *parent; |
| |
63 struct gaim_pref *sibling; |
| |
64 struct gaim_pref *first_child; |
| |
65 }; |
| |
66 |
| |
67 |
| |
68 static struct gaim_pref prefs = { |
| |
69 GAIM_PREF_NONE, |
| |
70 NULL, |
| |
71 { NULL }, |
| |
72 NULL, |
| |
73 NULL, |
| |
74 NULL, |
| |
75 NULL |
| |
76 }; |
| |
77 |
| |
78 static GHashTable *prefs_hash = NULL; |
| |
79 static guint save_timer = 0; |
| |
80 static gboolean prefs_loaded = FALSE; |
| |
81 |
| |
82 |
| |
83 /********************************************************************* |
| |
84 * Private utility functions * |
| |
85 *********************************************************************/ |
| |
86 |
| |
87 static struct |
| |
88 gaim_pref *find_pref(const char *name) |
| |
89 { |
| |
90 if (!name || name[0] != '/') |
| |
91 return NULL; |
| |
92 else if (name[1] == '\0') |
| |
93 return &prefs; |
| |
94 else |
| |
95 return g_hash_table_lookup(prefs_hash, name); |
| |
96 } |
| |
97 |
| |
98 |
| |
99 /********************************************************************* |
| |
100 * Writing to disk * |
| |
101 *********************************************************************/ |
| |
102 |
| |
103 /* |
| |
104 * This function recursively creates the xmlnode tree from the prefs |
| |
105 * tree structure. Yay recursion! |
| |
106 */ |
| |
107 static void |
| |
108 pref_to_xmlnode(xmlnode *parent, struct gaim_pref *pref) |
| |
109 { |
| |
110 xmlnode *node, *childnode; |
| |
111 struct gaim_pref *child; |
| |
112 char buf[20]; |
| |
113 GList *cur; |
| |
114 |
| |
115 /* Create a new node */ |
| |
116 node = xmlnode_new_child(parent, "pref"); |
| |
117 xmlnode_set_attrib(node, "name", pref->name); |
| |
118 |
| |
119 /* Set the type of this node (if type == GAIM_PREF_NONE then do nothing) */ |
| |
120 if (pref->type == GAIM_PREF_INT) { |
| |
121 xmlnode_set_attrib(node, "type", "int"); |
| |
122 snprintf(buf, sizeof(buf), "%d", pref->value.integer); |
| |
123 xmlnode_set_attrib(node, "value", buf); |
| |
124 } |
| |
125 else if (pref->type == GAIM_PREF_STRING) { |
| |
126 xmlnode_set_attrib(node, "type", "string"); |
| |
127 xmlnode_set_attrib(node, "value", pref->value.string ? pref->value.string : ""); |
| |
128 } |
| |
129 else if (pref->type == GAIM_PREF_STRING_LIST) { |
| |
130 xmlnode_set_attrib(node, "type", "stringlist"); |
| |
131 for (cur = pref->value.stringlist; cur != NULL; cur = cur->next) |
| |
132 { |
| |
133 childnode = xmlnode_new_child(node, "item"); |
| |
134 xmlnode_set_attrib(childnode, "value", cur->data ? cur->data : ""); |
| |
135 } |
| |
136 } |
| |
137 else if (pref->type == GAIM_PREF_BOOLEAN) { |
| |
138 xmlnode_set_attrib(node, "type", "bool"); |
| |
139 snprintf(buf, sizeof(buf), "%d", pref->value.boolean); |
| |
140 xmlnode_set_attrib(node, "value", buf); |
| |
141 } |
| |
142 |
| |
143 /* All My Children */ |
| |
144 for (child = pref->first_child; child != NULL; child = child->sibling) |
| |
145 pref_to_xmlnode(node, child); |
| |
146 } |
| |
147 |
| |
148 static xmlnode * |
| |
149 prefs_to_xmlnode(void) |
| |
150 { |
| |
151 xmlnode *node; |
| |
152 struct gaim_pref *pref, *child; |
| |
153 |
| |
154 pref = &prefs; |
| |
155 |
| |
156 /* Create the root preference node */ |
| |
157 node = xmlnode_new("pref"); |
| |
158 xmlnode_set_attrib(node, "version", "1"); |
| |
159 xmlnode_set_attrib(node, "name", "/"); |
| |
160 |
| |
161 /* All My Children */ |
| |
162 for (child = pref->first_child; child != NULL; child = child->sibling) |
| |
163 pref_to_xmlnode(node, child); |
| |
164 |
| |
165 return node; |
| |
166 } |
| |
167 |
| |
168 static void |
| |
169 sync_prefs(void) |
| |
170 { |
| |
171 xmlnode *node; |
| |
172 char *data; |
| |
173 |
| |
174 if (!prefs_loaded) |
| |
175 { |
| |
176 /* |
| |
177 * TODO: Call schedule_prefs_save()? Ideally we wouldn't need to. |
| |
178 * (prefs.xml should be loaded when gaim_prefs_init is called) |
| |
179 */ |
| |
180 gaim_debug_error("prefs", "Attempted to save prefs before " |
| |
181 "they were read!\n"); |
| |
182 return; |
| |
183 } |
| |
184 |
| |
185 node = prefs_to_xmlnode(); |
| |
186 data = xmlnode_to_formatted_str(node, NULL); |
| |
187 gaim_util_write_data_to_file("prefs.xml", data, -1); |
| |
188 g_free(data); |
| |
189 xmlnode_free(node); |
| |
190 } |
| |
191 |
| |
192 static gboolean |
| |
193 save_cb(gpointer data) |
| |
194 { |
| |
195 sync_prefs(); |
| |
196 save_timer = 0; |
| |
197 return FALSE; |
| |
198 } |
| |
199 |
| |
200 static void |
| |
201 schedule_prefs_save(void) |
| |
202 { |
| |
203 if (save_timer == 0) |
| |
204 save_timer = gaim_timeout_add(5000, save_cb, NULL); |
| |
205 } |
| |
206 |
| |
207 |
| |
208 /********************************************************************* |
| |
209 * Reading from disk * |
| |
210 *********************************************************************/ |
| |
211 |
| |
212 static GList *prefs_stack = NULL; |
| |
213 |
| |
214 static void |
| |
215 prefs_start_element_handler (GMarkupParseContext *context, |
| |
216 const gchar *element_name, |
| |
217 const gchar **attribute_names, |
| |
218 const gchar **attribute_values, |
| |
219 gpointer user_data, |
| |
220 GError **error) |
| |
221 { |
| |
222 GaimPrefType pref_type = GAIM_PREF_NONE; |
| |
223 int i; |
| |
224 const char *pref_name = NULL, *pref_value = NULL; |
| |
225 GString *pref_name_full; |
| |
226 GList *tmp; |
| |
227 |
| |
228 if(strcmp(element_name, "pref") && strcmp(element_name, "item")) |
| |
229 return; |
| |
230 |
| |
231 for(i = 0; attribute_names[i]; i++) { |
| |
232 if(!strcmp(attribute_names[i], "name")) { |
| |
233 pref_name = attribute_values[i]; |
| |
234 } else if(!strcmp(attribute_names[i], "type")) { |
| |
235 if(!strcmp(attribute_values[i], "bool")) |
| |
236 pref_type = GAIM_PREF_BOOLEAN; |
| |
237 else if(!strcmp(attribute_values[i], "int")) |
| |
238 pref_type = GAIM_PREF_INT; |
| |
239 else if(!strcmp(attribute_values[i], "string")) |
| |
240 pref_type = GAIM_PREF_STRING; |
| |
241 else if(!strcmp(attribute_values[i], "stringlist")) |
| |
242 pref_type = GAIM_PREF_STRING_LIST; |
| |
243 else |
| |
244 return; |
| |
245 } else if(!strcmp(attribute_names[i], "value")) { |
| |
246 pref_value = attribute_values[i]; |
| |
247 } |
| |
248 } |
| |
249 |
| |
250 if(!strcmp(element_name, "item")) { |
| |
251 struct gaim_pref *pref; |
| |
252 |
| |
253 pref_name_full = g_string_new(""); |
| |
254 |
| |
255 for(tmp = prefs_stack; tmp; tmp = tmp->next) { |
| |
256 pref_name_full = g_string_prepend(pref_name_full, tmp->data); |
| |
257 pref_name_full = g_string_prepend_c(pref_name_full, '/'); |
| |
258 } |
| |
259 |
| |
260 pref = find_pref(pref_name_full->str); |
| |
261 |
| |
262 if(pref) { |
| |
263 pref->value.stringlist = g_list_append(pref->value.stringlist, |
| |
264 g_strdup(pref_value)); |
| |
265 } |
| |
266 } else { |
| |
267 if(!pref_name || !strcmp(pref_name, "/")) |
| |
268 return; |
| |
269 |
| |
270 pref_name_full = g_string_new(pref_name); |
| |
271 |
| |
272 for(tmp = prefs_stack; tmp; tmp = tmp->next) { |
| |
273 pref_name_full = g_string_prepend_c(pref_name_full, '/'); |
| |
274 pref_name_full = g_string_prepend(pref_name_full, tmp->data); |
| |
275 } |
| |
276 |
| |
277 pref_name_full = g_string_prepend_c(pref_name_full, '/'); |
| |
278 |
| |
279 switch(pref_type) { |
| |
280 case GAIM_PREF_NONE: |
| |
281 gaim_prefs_add_none(pref_name_full->str); |
| |
282 break; |
| |
283 case GAIM_PREF_BOOLEAN: |
| |
284 gaim_prefs_set_bool(pref_name_full->str, atoi(pref_value)); |
| |
285 break; |
| |
286 case GAIM_PREF_INT: |
| |
287 gaim_prefs_set_int(pref_name_full->str, atoi(pref_value)); |
| |
288 break; |
| |
289 case GAIM_PREF_STRING: |
| |
290 gaim_prefs_set_string(pref_name_full->str, pref_value); |
| |
291 break; |
| |
292 case GAIM_PREF_STRING_LIST: |
| |
293 gaim_prefs_set_string_list(pref_name_full->str, NULL); |
| |
294 break; |
| |
295 } |
| |
296 prefs_stack = g_list_prepend(prefs_stack, g_strdup(pref_name)); |
| |
297 g_string_free(pref_name_full, TRUE); |
| |
298 } |
| |
299 } |
| |
300 |
| |
301 static void |
| |
302 prefs_end_element_handler(GMarkupParseContext *context, |
| |
303 const gchar *element_name, |
| |
304 gpointer user_data, GError **error) |
| |
305 { |
| |
306 if(prefs_stack && !strcmp(element_name, "pref")) { |
| |
307 g_free(prefs_stack->data); |
| |
308 prefs_stack = g_list_delete_link(prefs_stack, prefs_stack); |
| |
309 } |
| |
310 } |
| |
311 |
| |
312 static GMarkupParser prefs_parser = { |
| |
313 prefs_start_element_handler, |
| |
314 prefs_end_element_handler, |
| |
315 NULL, |
| |
316 NULL, |
| |
317 NULL |
| |
318 }; |
| |
319 |
| |
320 gboolean |
| |
321 gaim_prefs_load() |
| |
322 { |
| |
323 gchar *filename = g_build_filename(gaim_user_dir(), "prefs.xml", NULL); |
| |
324 gchar *contents = NULL; |
| |
325 gsize length; |
| |
326 GMarkupParseContext *context; |
| |
327 GError *error = NULL; |
| |
328 |
| |
329 if (!filename) { |
| |
330 prefs_loaded = TRUE; |
| |
331 return FALSE; |
| |
332 } |
| |
333 |
| |
334 gaim_debug_info("prefs", "Reading %s\n", filename); |
| |
335 |
| |
336 if(!g_file_get_contents(filename, &contents, &length, &error)) { |
| |
337 #ifndef _WIN32 |
| |
338 g_free(filename); |
| |
339 g_error_free(error); |
| |
340 |
| |
341 error = NULL; |
| |
342 |
| |
343 filename = g_build_filename(SYSCONFDIR, "gaim", "prefs.xml", NULL); |
| |
344 |
| |
345 gaim_debug_info("prefs", "Reading %s\n", filename); |
| |
346 |
| |
347 if (!g_file_get_contents(filename, &contents, &length, &error)) { |
| |
348 gaim_debug_error("prefs", "Error reading prefs: %s\n", |
| |
349 error->message); |
| |
350 g_error_free(error); |
| |
351 g_free(filename); |
| |
352 prefs_loaded = TRUE; |
| |
353 |
| |
354 return FALSE; |
| |
355 } |
| |
356 #else /* _WIN32 */ |
| |
357 gaim_debug_error("prefs", "Error reading prefs: %s\n", |
| |
358 error->message); |
| |
359 g_error_free(error); |
| |
360 g_free(filename); |
| |
361 prefs_loaded = TRUE; |
| |
362 |
| |
363 return FALSE; |
| |
364 #endif /* _WIN32 */ |
| |
365 } |
| |
366 |
| |
367 context = g_markup_parse_context_new(&prefs_parser, 0, NULL, NULL); |
| |
368 |
| |
369 if(!g_markup_parse_context_parse(context, contents, length, NULL)) { |
| |
370 g_markup_parse_context_free(context); |
| |
371 g_free(contents); |
| |
372 g_free(filename); |
| |
373 prefs_loaded = TRUE; |
| |
374 |
| |
375 return FALSE; |
| |
376 } |
| |
377 |
| |
378 if(!g_markup_parse_context_end_parse(context, NULL)) { |
| |
379 gaim_debug_error("prefs", "Error parsing %s\n", filename); |
| |
380 g_markup_parse_context_free(context); |
| |
381 g_free(contents); |
| |
382 g_free(filename); |
| |
383 prefs_loaded = TRUE; |
| |
384 |
| |
385 return FALSE; |
| |
386 } |
| |
387 |
| |
388 gaim_debug_info("prefs", "Finished reading %s\n", filename); |
| |
389 g_markup_parse_context_free(context); |
| |
390 g_free(contents); |
| |
391 g_free(filename); |
| |
392 prefs_loaded = TRUE; |
| |
393 |
| |
394 /* I introduced a bug in 2.0.0beta2. This fixes the broken |
| |
395 * scores on upgrade. This can be removed sometime shortly |
| |
396 * after 2.0.0 final is released. -- rlaager */ |
| |
397 if (gaim_prefs_get_int("/core/status/scores/offline") == -500 && |
| |
398 gaim_prefs_get_int("/core/status/scores/available") == 100 && |
| |
399 gaim_prefs_get_int("/core/status/scores/invisible") == -50 && |
| |
400 gaim_prefs_get_int("/core/status/scores/away") == -100 && |
| |
401 gaim_prefs_get_int("/core/status/scores/extended_away") == -200 && |
| |
402 gaim_prefs_get_int("/core/status/scores/idle") == -400) |
| |
403 { |
| |
404 gaim_prefs_set_int("/core/status/scores/idle", -10); |
| |
405 } |
| |
406 |
| |
407 return TRUE; |
| |
408 } |
| |
409 |
| |
410 |
| |
411 |
| |
412 static void |
| |
413 prefs_save_cb(const char *name, GaimPrefType type, gconstpointer val, |
| |
414 gpointer user_data) |
| |
415 { |
| |
416 |
| |
417 if(!prefs_loaded) |
| |
418 return; |
| |
419 |
| |
420 gaim_debug_misc("prefs", "%s changed, scheduling save.\n", name); |
| |
421 |
| |
422 schedule_prefs_save(); |
| |
423 } |
| |
424 |
| |
425 static char * |
| |
426 get_path_dirname(const char *name) |
| |
427 { |
| |
428 char *c, *str; |
| |
429 |
| |
430 str = g_strdup(name); |
| |
431 |
| |
432 if ((c = strrchr(str, '/')) != NULL) { |
| |
433 *c = '\0'; |
| |
434 |
| |
435 if (*str == '\0') { |
| |
436 g_free(str); |
| |
437 |
| |
438 str = g_strdup("/"); |
| |
439 } |
| |
440 } |
| |
441 else { |
| |
442 g_free(str); |
| |
443 |
| |
444 str = g_strdup("."); |
| |
445 } |
| |
446 |
| |
447 return str; |
| |
448 } |
| |
449 |
| |
450 static char * |
| |
451 get_path_basename(const char *name) |
| |
452 { |
| |
453 const char *c; |
| |
454 |
| |
455 if ((c = strrchr(name, '/')) != NULL) |
| |
456 return g_strdup(c + 1); |
| |
457 |
| |
458 return g_strdup(name); |
| |
459 } |
| |
460 |
| |
461 static char * |
| |
462 pref_full_name(struct gaim_pref *pref) |
| |
463 { |
| |
464 GString *name; |
| |
465 struct gaim_pref *parent; |
| |
466 |
| |
467 if(!pref) |
| |
468 return NULL; |
| |
469 |
| |
470 if(pref == &prefs) |
| |
471 return g_strdup("/"); |
| |
472 |
| |
473 name = g_string_new(pref->name); |
| |
474 parent = pref->parent; |
| |
475 |
| |
476 for(parent = pref->parent; parent && parent->name; parent = parent->parent) { |
| |
477 name = g_string_prepend_c(name, '/'); |
| |
478 name = g_string_prepend(name, parent->name); |
| |
479 } |
| |
480 name = g_string_prepend_c(name, '/'); |
| |
481 return g_string_free(name, FALSE); |
| |
482 } |
| |
483 |
| |
484 static struct gaim_pref * |
| |
485 find_pref_parent(const char *name) |
| |
486 { |
| |
487 char *parent_name = get_path_dirname(name); |
| |
488 struct gaim_pref *ret = &prefs; |
| |
489 |
| |
490 if(strcmp(parent_name, "/")) { |
| |
491 ret = find_pref(parent_name); |
| |
492 } |
| |
493 |
| |
494 g_free(parent_name); |
| |
495 return ret; |
| |
496 } |
| |
497 |
| |
498 static void |
| |
499 free_pref_value(struct gaim_pref *pref) |
| |
500 { |
| |
501 switch(pref->type) { |
| |
502 case GAIM_PREF_BOOLEAN: |
| |
503 pref->value.boolean = FALSE; |
| |
504 break; |
| |
505 case GAIM_PREF_INT: |
| |
506 pref->value.integer = 0; |
| |
507 break; |
| |
508 case GAIM_PREF_STRING: |
| |
509 g_free(pref->value.string); |
| |
510 pref->value.string = NULL; |
| |
511 break; |
| |
512 case GAIM_PREF_STRING_LIST: |
| |
513 { |
| |
514 g_list_foreach(pref->value.stringlist, (GFunc)g_free, NULL); |
| |
515 g_list_free(pref->value.stringlist); |
| |
516 } break; |
| |
517 case GAIM_PREF_NONE: |
| |
518 break; |
| |
519 } |
| |
520 } |
| |
521 |
| |
522 static struct gaim_pref * |
| |
523 add_pref(GaimPrefType type, const char *name) |
| |
524 { |
| |
525 struct gaim_pref *parent; |
| |
526 struct gaim_pref *me; |
| |
527 struct gaim_pref *sibling; |
| |
528 char *my_name; |
| |
529 |
| |
530 parent = find_pref_parent(name); |
| |
531 |
| |
532 if(!parent) |
| |
533 return NULL; |
| |
534 |
| |
535 my_name = get_path_basename(name); |
| |
536 |
| |
537 for(sibling = parent->first_child; sibling; sibling = sibling->sibling) { |
| |
538 if(!strcmp(sibling->name, my_name)) { |
| |
539 g_free(my_name); |
| |
540 return NULL; |
| |
541 } |
| |
542 } |
| |
543 |
| |
544 me = g_new0(struct gaim_pref, 1); |
| |
545 me->type = type; |
| |
546 me->name = my_name; |
| |
547 |
| |
548 me->parent = parent; |
| |
549 if(parent->first_child) { |
| |
550 /* blatant abuse of a for loop */ |
| |
551 for(sibling = parent->first_child; sibling->sibling; |
| |
552 sibling = sibling->sibling); |
| |
553 sibling->sibling = me; |
| |
554 } else { |
| |
555 parent->first_child = me; |
| |
556 } |
| |
557 |
| |
558 g_hash_table_insert(prefs_hash, g_strdup(name), (gpointer)me); |
| |
559 |
| |
560 return me; |
| |
561 } |
| |
562 |
| |
563 void |
| |
564 gaim_prefs_add_none(const char *name) |
| |
565 { |
| |
566 add_pref(GAIM_PREF_NONE, name); |
| |
567 } |
| |
568 |
| |
569 void |
| |
570 gaim_prefs_add_bool(const char *name, gboolean value) |
| |
571 { |
| |
572 struct gaim_pref *pref = add_pref(GAIM_PREF_BOOLEAN, name); |
| |
573 |
| |
574 if(!pref) |
| |
575 return; |
| |
576 |
| |
577 pref->value.boolean = value; |
| |
578 } |
| |
579 |
| |
580 void |
| |
581 gaim_prefs_add_int(const char *name, int value) |
| |
582 { |
| |
583 struct gaim_pref *pref = add_pref(GAIM_PREF_INT, name); |
| |
584 |
| |
585 if(!pref) |
| |
586 return; |
| |
587 |
| |
588 pref->value.integer = value; |
| |
589 } |
| |
590 |
| |
591 void |
| |
592 gaim_prefs_add_string(const char *name, const char *value) |
| |
593 { |
| |
594 struct gaim_pref *pref = add_pref(GAIM_PREF_STRING, name); |
| |
595 |
| |
596 if(!pref) |
| |
597 return; |
| |
598 |
| |
599 pref->value.string = g_strdup(value); |
| |
600 } |
| |
601 |
| |
602 void |
| |
603 gaim_prefs_add_string_list(const char *name, GList *value) |
| |
604 { |
| |
605 struct gaim_pref *pref = add_pref(GAIM_PREF_STRING_LIST, name); |
| |
606 GList *tmp; |
| |
607 |
| |
608 if(!pref) |
| |
609 return; |
| |
610 |
| |
611 for(tmp = value; tmp; tmp = tmp->next) |
| |
612 pref->value.stringlist = g_list_append(pref->value.stringlist, |
| |
613 g_strdup(tmp->data)); |
| |
614 } |
| |
615 |
| |
616 static void |
| |
617 remove_pref(struct gaim_pref *pref) |
| |
618 { |
| |
619 char *name; |
| |
620 |
| |
621 if(!pref || pref == &prefs) |
| |
622 return; |
| |
623 |
| |
624 while(pref->first_child) |
| |
625 remove_pref(pref->first_child); |
| |
626 |
| |
627 if(pref->parent->first_child == pref) { |
| |
628 pref->parent->first_child = pref->sibling; |
| |
629 } else { |
| |
630 struct gaim_pref *sib = pref->parent->first_child; |
| |
631 while(sib && sib->sibling != pref) |
| |
632 sib = sib->sibling; |
| |
633 if(sib) |
| |
634 sib->sibling = pref->sibling; |
| |
635 } |
| |
636 |
| |
637 name = pref_full_name(pref); |
| |
638 |
| |
639 gaim_debug_info("prefs", "removing pref %s\n", name); |
| |
640 |
| |
641 g_hash_table_remove(prefs_hash, name); |
| |
642 g_free(name); |
| |
643 |
| |
644 free_pref_value(pref); |
| |
645 |
| |
646 g_slist_free(pref->callbacks); |
| |
647 g_free(pref->name); |
| |
648 g_free(pref); |
| |
649 } |
| |
650 |
| |
651 void |
| |
652 gaim_prefs_remove(const char *name) |
| |
653 { |
| |
654 struct gaim_pref *pref = find_pref(name); |
| |
655 |
| |
656 if(!pref) |
| |
657 return; |
| |
658 |
| |
659 remove_pref(pref); |
| |
660 } |
| |
661 |
| |
662 void |
| |
663 gaim_prefs_destroy() |
| |
664 { |
| |
665 gaim_prefs_remove("/"); |
| |
666 } |
| |
667 |
| |
668 static void |
| |
669 do_callbacks(const char* name, struct gaim_pref *pref) |
| |
670 { |
| |
671 GSList *cbs; |
| |
672 struct gaim_pref *cb_pref; |
| |
673 for(cb_pref = pref; cb_pref; cb_pref = cb_pref->parent) { |
| |
674 for(cbs = cb_pref->callbacks; cbs; cbs = cbs->next) { |
| |
675 struct pref_cb *cb = cbs->data; |
| |
676 cb->func(name, pref->type, pref->value.generic, cb->data); |
| |
677 } |
| |
678 } |
| |
679 } |
| |
680 |
| |
681 void |
| |
682 gaim_prefs_trigger_callback(const char *name) |
| |
683 { |
| |
684 struct gaim_pref *pref = find_pref(name); |
| |
685 |
| |
686 if(!pref) { |
| |
687 gaim_debug_error("prefs", |
| |
688 "gaim_prefs_trigger_callback: Unknown pref %s\n", name); |
| |
689 return; |
| |
690 } |
| |
691 |
| |
692 do_callbacks(name, pref); |
| |
693 } |
| |
694 |
| |
695 void |
| |
696 gaim_prefs_set_generic(const char *name, gpointer value) |
| |
697 { |
| |
698 struct gaim_pref *pref = find_pref(name); |
| |
699 |
| |
700 if(!pref) { |
| |
701 gaim_debug_error("prefs", |
| |
702 "gaim_prefs_set_generic: Unknown pref %s\n", name); |
| |
703 return; |
| |
704 } |
| |
705 |
| |
706 pref->value.generic = value; |
| |
707 do_callbacks(name, pref); |
| |
708 } |
| |
709 |
| |
710 void |
| |
711 gaim_prefs_set_bool(const char *name, gboolean value) |
| |
712 { |
| |
713 struct gaim_pref *pref = find_pref(name); |
| |
714 |
| |
715 if(pref) { |
| |
716 if(pref->type != GAIM_PREF_BOOLEAN) { |
| |
717 gaim_debug_error("prefs", |
| |
718 "gaim_prefs_set_bool: %s not a boolean pref\n", name); |
| |
719 return; |
| |
720 } |
| |
721 |
| |
722 if(pref->value.boolean != value) { |
| |
723 pref->value.boolean = value; |
| |
724 do_callbacks(name, pref); |
| |
725 } |
| |
726 } else { |
| |
727 gaim_prefs_add_bool(name, value); |
| |
728 } |
| |
729 } |
| |
730 |
| |
731 void |
| |
732 gaim_prefs_set_int(const char *name, int value) |
| |
733 { |
| |
734 struct gaim_pref *pref = find_pref(name); |
| |
735 |
| |
736 if(pref) { |
| |
737 if(pref->type != GAIM_PREF_INT) { |
| |
738 gaim_debug_error("prefs", |
| |
739 "gaim_prefs_set_int: %s not an integer pref\n", name); |
| |
740 return; |
| |
741 } |
| |
742 |
| |
743 if(pref->value.integer != value) { |
| |
744 pref->value.integer = value; |
| |
745 do_callbacks(name, pref); |
| |
746 } |
| |
747 } else { |
| |
748 gaim_prefs_add_int(name, value); |
| |
749 } |
| |
750 } |
| |
751 |
| |
752 void |
| |
753 gaim_prefs_set_string(const char *name, const char *value) |
| |
754 { |
| |
755 struct gaim_pref *pref = find_pref(name); |
| |
756 |
| |
757 if(pref) { |
| |
758 if(pref->type != GAIM_PREF_STRING) { |
| |
759 gaim_debug_error("prefs", |
| |
760 "gaim_prefs_set_string: %s not a string pref\n", name); |
| |
761 return; |
| |
762 } |
| |
763 |
| |
764 if((value && !pref->value.string) || |
| |
765 (!value && pref->value.string) || |
| |
766 (value && pref->value.string && |
| |
767 strcmp(pref->value.string, value))) { |
| |
768 g_free(pref->value.string); |
| |
769 pref->value.string = g_strdup(value); |
| |
770 do_callbacks(name, pref); |
| |
771 } |
| |
772 } else { |
| |
773 gaim_prefs_add_string(name, value); |
| |
774 } |
| |
775 } |
| |
776 |
| |
777 void |
| |
778 gaim_prefs_set_string_list(const char *name, GList *value) |
| |
779 { |
| |
780 struct gaim_pref *pref = find_pref(name); |
| |
781 if(pref) { |
| |
782 GList *tmp; |
| |
783 |
| |
784 if(pref->type != GAIM_PREF_STRING_LIST) { |
| |
785 gaim_debug_error("prefs", |
| |
786 "gaim_prefs_set_string_list: %s not a string list pref\n", |
| |
787 name); |
| |
788 return; |
| |
789 } |
| |
790 |
| |
791 g_list_foreach(pref->value.stringlist, (GFunc)g_free, NULL); |
| |
792 g_list_free(pref->value.stringlist); |
| |
793 pref->value.stringlist = NULL; |
| |
794 |
| |
795 for(tmp = value; tmp; tmp = tmp->next) |
| |
796 pref->value.stringlist = g_list_prepend(pref->value.stringlist, |
| |
797 g_strdup(tmp->data)); |
| |
798 pref->value.stringlist = g_list_reverse(pref->value.stringlist); |
| |
799 |
| |
800 do_callbacks(name, pref); |
| |
801 |
| |
802 } else { |
| |
803 gaim_prefs_add_string_list(name, value); |
| |
804 } |
| |
805 } |
| |
806 |
| |
807 gboolean |
| |
808 gaim_prefs_exists(const char *name) |
| |
809 { |
| |
810 struct gaim_pref *pref = find_pref(name); |
| |
811 |
| |
812 if (pref != NULL) |
| |
813 return TRUE; |
| |
814 |
| |
815 return FALSE; |
| |
816 } |
| |
817 |
| |
818 GaimPrefType |
| |
819 gaim_prefs_get_type(const char *name) |
| |
820 { |
| |
821 struct gaim_pref *pref = find_pref(name); |
| |
822 |
| |
823 if (pref == NULL) |
| |
824 return GAIM_PREF_NONE; |
| |
825 |
| |
826 return (pref->type); |
| |
827 } |
| |
828 |
| |
829 gboolean |
| |
830 gaim_prefs_get_bool(const char *name) |
| |
831 { |
| |
832 struct gaim_pref *pref = find_pref(name); |
| |
833 |
| |
834 if(!pref) { |
| |
835 gaim_debug_error("prefs", |
| |
836 "gaim_prefs_get_bool: Unknown pref %s\n", name); |
| |
837 return FALSE; |
| |
838 } else if(pref->type != GAIM_PREF_BOOLEAN) { |
| |
839 gaim_debug_error("prefs", |
| |
840 "gaim_prefs_get_bool: %s not a boolean pref\n", name); |
| |
841 return FALSE; |
| |
842 } |
| |
843 |
| |
844 return pref->value.boolean; |
| |
845 } |
| |
846 |
| |
847 int |
| |
848 gaim_prefs_get_int(const char *name) |
| |
849 { |
| |
850 struct gaim_pref *pref = find_pref(name); |
| |
851 |
| |
852 if(!pref) { |
| |
853 gaim_debug_error("prefs", |
| |
854 "gaim_prefs_get_int: Unknown pref %s\n", name); |
| |
855 return 0; |
| |
856 } else if(pref->type != GAIM_PREF_INT) { |
| |
857 gaim_debug_error("prefs", |
| |
858 "gaim_prefs_get_int: %s not an integer pref\n", name); |
| |
859 return 0; |
| |
860 } |
| |
861 |
| |
862 return pref->value.integer; |
| |
863 } |
| |
864 |
| |
865 const char * |
| |
866 gaim_prefs_get_string(const char *name) |
| |
867 { |
| |
868 struct gaim_pref *pref = find_pref(name); |
| |
869 |
| |
870 if(!pref) { |
| |
871 gaim_debug_error("prefs", |
| |
872 "gaim_prefs_get_string: Unknown pref %s\n", name); |
| |
873 return NULL; |
| |
874 } else if(pref->type != GAIM_PREF_STRING) { |
| |
875 gaim_debug_error("prefs", |
| |
876 "gaim_prefs_get_string: %s not a string pref\n", name); |
| |
877 return NULL; |
| |
878 } |
| |
879 |
| |
880 return pref->value.string; |
| |
881 } |
| |
882 |
| |
883 GList * |
| |
884 gaim_prefs_get_string_list(const char *name) |
| |
885 { |
| |
886 struct gaim_pref *pref = find_pref(name); |
| |
887 GList *ret = NULL, *tmp; |
| |
888 |
| |
889 if(!pref) { |
| |
890 gaim_debug_error("prefs", |
| |
891 "gaim_prefs_get_string_list: Unknown pref %s\n", name); |
| |
892 return NULL; |
| |
893 } else if(pref->type != GAIM_PREF_STRING_LIST) { |
| |
894 gaim_debug_error("prefs", |
| |
895 "gaim_prefs_get_string_list: %s not a string list pref\n", name); |
| |
896 return NULL; |
| |
897 } |
| |
898 |
| |
899 for(tmp = pref->value.stringlist; tmp; tmp = tmp->next) |
| |
900 ret = g_list_prepend(ret, g_strdup(tmp->data)); |
| |
901 ret = g_list_reverse(ret); |
| |
902 |
| |
903 return ret; |
| |
904 } |
| |
905 |
| |
906 void |
| |
907 gaim_prefs_rename(const char *oldname, const char *newname) |
| |
908 { |
| |
909 struct gaim_pref *oldpref, *newpref; |
| |
910 |
| |
911 oldpref = find_pref(oldname); |
| |
912 |
| |
913 /* it's already been renamed, call off the dogs */ |
| |
914 if(!oldpref) |
| |
915 return; |
| |
916 |
| |
917 if (oldpref->first_child != NULL) /* can't rename parents */ |
| |
918 { |
| |
919 gaim_debug_error("prefs", "Unable to rename %s to %s: can't rename parents\n", oldname, newname); |
| |
920 return; |
| |
921 } |
| |
922 |
| |
923 |
| |
924 newpref = find_pref(newname); |
| |
925 |
| |
926 if (newpref == NULL) |
| |
927 { |
| |
928 gaim_debug_error("prefs", "Unable to rename %s to %s: new pref not created\n", oldname, newname); |
| |
929 return; |
| |
930 } |
| |
931 |
| |
932 if (oldpref->type != newpref->type) |
| |
933 { |
| |
934 gaim_debug_error("prefs", "Unable to rename %s to %s: differing types\n", oldname, newname); |
| |
935 return; |
| |
936 } |
| |
937 |
| |
938 gaim_debug_info("prefs", "Renaming %s to %s\n", oldname, newname); |
| |
939 |
| |
940 switch(oldpref->type) { |
| |
941 case GAIM_PREF_NONE: |
| |
942 break; |
| |
943 case GAIM_PREF_BOOLEAN: |
| |
944 gaim_prefs_set_bool(newname, oldpref->value.boolean); |
| |
945 break; |
| |
946 case GAIM_PREF_INT: |
| |
947 gaim_prefs_set_int(newname, oldpref->value.integer); |
| |
948 break; |
| |
949 case GAIM_PREF_STRING: |
| |
950 gaim_prefs_set_string(newname, oldpref->value.string); |
| |
951 break; |
| |
952 case GAIM_PREF_STRING_LIST: |
| |
953 gaim_prefs_set_string_list(newname, oldpref->value.stringlist); |
| |
954 break; |
| |
955 } |
| |
956 |
| |
957 remove_pref(oldpref); |
| |
958 } |
| |
959 |
| |
960 void |
| |
961 gaim_prefs_rename_boolean_toggle(const char *oldname, const char *newname) |
| |
962 { |
| |
963 struct gaim_pref *oldpref, *newpref; |
| |
964 |
| |
965 oldpref = find_pref(oldname); |
| |
966 |
| |
967 /* it's already been renamed, call off the cats */ |
| |
968 if(!oldpref) |
| |
969 return; |
| |
970 |
| |
971 if (oldpref->type != GAIM_PREF_BOOLEAN) |
| |
972 { |
| |
973 gaim_debug_error("prefs", "Unable to rename %s to %s: old pref not a boolean\n", oldname, newname); |
| |
974 return; |
| |
975 } |
| |
976 |
| |
977 if (oldpref->first_child != NULL) /* can't rename parents */ |
| |
978 { |
| |
979 gaim_debug_error("prefs", "Unable to rename %s to %s: can't rename parents\n", oldname, newname); |
| |
980 return; |
| |
981 } |
| |
982 |
| |
983 |
| |
984 newpref = find_pref(newname); |
| |
985 |
| |
986 if (newpref == NULL) |
| |
987 { |
| |
988 gaim_debug_error("prefs", "Unable to rename %s to %s: new pref not created\n", oldname, newname); |
| |
989 return; |
| |
990 } |
| |
991 |
| |
992 if (oldpref->type != newpref->type) |
| |
993 { |
| |
994 gaim_debug_error("prefs", "Unable to rename %s to %s: differing types\n", oldname, newname); |
| |
995 return; |
| |
996 } |
| |
997 |
| |
998 gaim_debug_info("prefs", "Renaming and toggling %s to %s\n", oldname, newname); |
| |
999 gaim_prefs_set_bool(newname, !(oldpref->value.boolean)); |
| |
1000 |
| |
1001 remove_pref(oldpref); |
| |
1002 } |
| |
1003 |
| |
1004 guint |
| |
1005 gaim_prefs_connect_callback(void *handle, const char *name, GaimPrefCallback func, gpointer data) |
| |
1006 { |
| |
1007 struct gaim_pref *pref; |
| |
1008 struct pref_cb *cb; |
| |
1009 static guint cb_id = 0; |
| |
1010 |
| |
1011 pref = find_pref(name); |
| |
1012 if (pref == NULL) |
| |
1013 return 0; |
| |
1014 |
| |
1015 cb = g_new0(struct pref_cb, 1); |
| |
1016 |
| |
1017 cb->func = func; |
| |
1018 cb->data = data; |
| |
1019 cb->id = ++cb_id; |
| |
1020 cb->handle = handle; |
| |
1021 |
| |
1022 pref->callbacks = g_slist_append(pref->callbacks, cb); |
| |
1023 |
| |
1024 return cb->id; |
| |
1025 } |
| |
1026 |
| |
1027 static gboolean |
| |
1028 disco_callback_helper(struct gaim_pref *pref, guint callback_id) |
| |
1029 { |
| |
1030 GSList *cbs; |
| |
1031 struct gaim_pref *child; |
| |
1032 |
| |
1033 if(!pref) |
| |
1034 return FALSE; |
| |
1035 |
| |
1036 for(cbs = pref->callbacks; cbs; cbs = cbs->next) { |
| |
1037 struct pref_cb *cb = cbs->data; |
| |
1038 if(cb->id == callback_id) { |
| |
1039 pref->callbacks = g_slist_delete_link(pref->callbacks, cbs); |
| |
1040 g_free(cb); |
| |
1041 return TRUE; |
| |
1042 } |
| |
1043 } |
| |
1044 |
| |
1045 for(child = pref->first_child; child; child = child->sibling) { |
| |
1046 if(disco_callback_helper(child, callback_id)) |
| |
1047 return TRUE; |
| |
1048 } |
| |
1049 |
| |
1050 return FALSE; |
| |
1051 } |
| |
1052 |
| |
1053 void |
| |
1054 gaim_prefs_disconnect_callback(guint callback_id) |
| |
1055 { |
| |
1056 disco_callback_helper(&prefs, callback_id); |
| |
1057 } |
| |
1058 |
| |
1059 static void |
| |
1060 disco_callback_helper_handle(struct gaim_pref *pref, void *handle) |
| |
1061 { |
| |
1062 GSList *cbs; |
| |
1063 struct gaim_pref *child; |
| |
1064 |
| |
1065 if(!pref) |
| |
1066 return; |
| |
1067 |
| |
1068 cbs = pref->callbacks; |
| |
1069 while (cbs != NULL) { |
| |
1070 struct pref_cb *cb = cbs->data; |
| |
1071 if(cb->handle == handle) { |
| |
1072 pref->callbacks = g_slist_delete_link(pref->callbacks, cbs); |
| |
1073 g_free(cb); |
| |
1074 cbs = pref->callbacks; |
| |
1075 } else |
| |
1076 cbs = cbs->next; |
| |
1077 } |
| |
1078 |
| |
1079 for(child = pref->first_child; child; child = child->sibling) |
| |
1080 disco_callback_helper_handle(child, handle); |
| |
1081 } |
| |
1082 |
| |
1083 void |
| |
1084 gaim_prefs_disconnect_by_handle(void *handle) |
| |
1085 { |
| |
1086 g_return_if_fail(handle != NULL); |
| |
1087 |
| |
1088 disco_callback_helper_handle(&prefs, handle); |
| |
1089 } |
| |
1090 |
| |
1091 void |
| |
1092 gaim_prefs_update_old() |
| |
1093 { |
| |
1094 /* Remove some no-longer-used prefs */ |
| |
1095 gaim_prefs_remove("/core/away/auto_response/enabled"); |
| |
1096 gaim_prefs_remove("/core/away/auto_response/idle_only"); |
| |
1097 gaim_prefs_remove("/core/away/auto_response/in_active_conv"); |
| |
1098 gaim_prefs_remove("/core/away/auto_response/sec_before_resend"); |
| |
1099 gaim_prefs_remove("/core/away/auto_response"); |
| |
1100 gaim_prefs_remove("/core/away/default_message"); |
| |
1101 gaim_prefs_remove("/core/buddies/use_server_alias"); |
| |
1102 gaim_prefs_remove("/core/conversations/away_back_on_send"); |
| |
1103 gaim_prefs_remove("/core/conversations/send_urls_as_links"); |
| |
1104 gaim_prefs_remove("/core/conversations/im/show_login"); |
| |
1105 gaim_prefs_remove("/core/conversations/chat/show_join"); |
| |
1106 gaim_prefs_remove("/core/conversations/chat/show_leave"); |
| |
1107 gaim_prefs_remove("/core/conversations/combine_chat_im"); |
| |
1108 gaim_prefs_remove("/core/conversations/use_alias_for_title"); |
| |
1109 gaim_prefs_remove("/core/logging/log_signon_signoff"); |
| |
1110 gaim_prefs_remove("/core/logging/log_idle_state"); |
| |
1111 gaim_prefs_remove("/core/logging/log_away_state"); |
| |
1112 gaim_prefs_remove("/core/logging/log_own_states"); |
| |
1113 gaim_prefs_remove("/core/status/scores/hidden"); |
| |
1114 gaim_prefs_remove("/plugins/core/autorecon/hide_connected_error"); |
| |
1115 gaim_prefs_remove("/plugins/core/autorecon/hide_connecting_error"); |
| |
1116 gaim_prefs_remove("/plugins/core/autorecon/hide_reconnecting_dialog"); |
| |
1117 gaim_prefs_remove("/plugins/core/autorecon/restore_state"); |
| |
1118 gaim_prefs_remove("/plugins/core/autorecon"); |
| |
1119 } |
| |
1120 |
| |
1121 void * |
| |
1122 gaim_prefs_get_handle(void) |
| |
1123 { |
| |
1124 static int handle; |
| |
1125 |
| |
1126 return &handle; |
| |
1127 } |
| |
1128 |
| |
1129 void |
| |
1130 gaim_prefs_init(void) |
| |
1131 { |
| |
1132 void *handle = gaim_prefs_get_handle(); |
| |
1133 |
| |
1134 prefs_hash = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, NULL); |
| |
1135 |
| |
1136 gaim_prefs_connect_callback(handle, "/", prefs_save_cb, NULL); |
| |
1137 |
| |
1138 gaim_prefs_add_none("/core"); |
| |
1139 gaim_prefs_add_none("/plugins"); |
| |
1140 gaim_prefs_add_none("/plugins/core"); |
| |
1141 gaim_prefs_add_none("/plugins/lopl"); |
| |
1142 gaim_prefs_add_none("/plugins/prpl"); |
| |
1143 |
| |
1144 /* Away */ |
| |
1145 gaim_prefs_add_none("/core/away"); |
| |
1146 gaim_prefs_add_string("/core/away/idle_reporting", "system"); |
| |
1147 gaim_prefs_add_bool("/core/away/away_when_idle", TRUE); |
| |
1148 gaim_prefs_add_int("/core/away/mins_before_away", 5); |
| |
1149 |
| |
1150 /* Away -> Auto-Reply */ |
| |
1151 if (!gaim_prefs_exists("/core/away/auto_response/enabled") || |
| |
1152 !gaim_prefs_exists("/core/away/auto_response/idle_only")) |
| |
1153 { |
| |
1154 gaim_prefs_add_string("/core/away/auto_reply", "awayidle"); |
| |
1155 } |
| |
1156 else |
| |
1157 { |
| |
1158 if (!gaim_prefs_get_bool("/core/away/auto_response/enabled")) |
| |
1159 { |
| |
1160 gaim_prefs_add_string("/core/away/auto_reply", "never"); |
| |
1161 } |
| |
1162 else |
| |
1163 { |
| |
1164 if (gaim_prefs_get_bool("/core/away/auto_response/idle_only")) |
| |
1165 { |
| |
1166 gaim_prefs_add_string("/core/away/auto_reply", "awayidle"); |
| |
1167 } |
| |
1168 else |
| |
1169 { |
| |
1170 gaim_prefs_add_string("/core/away/auto_reply", "away"); |
| |
1171 } |
| |
1172 } |
| |
1173 } |
| |
1174 |
| |
1175 /* Buddies */ |
| |
1176 gaim_prefs_add_none("/core/buddies"); |
| |
1177 |
| |
1178 /* Contact Priority Settings */ |
| |
1179 gaim_prefs_add_none("/core/contact"); |
| |
1180 gaim_prefs_add_bool("/core/contact/last_match", FALSE); |
| |
1181 gaim_prefs_remove("/core/contact/offline_score"); |
| |
1182 gaim_prefs_remove("/core/contact/away_score"); |
| |
1183 gaim_prefs_remove("/core/contact/idle_score"); |
| |
1184 } |
| |
1185 |
| |
1186 void |
| |
1187 gaim_prefs_uninit() |
| |
1188 { |
| |
1189 if (save_timer != 0) |
| |
1190 { |
| |
1191 gaim_timeout_remove(save_timer); |
| |
1192 save_timer = 0; |
| |
1193 sync_prefs(); |
| |
1194 } |
| |
1195 |
| |
1196 gaim_prefs_disconnect_by_handle(gaim_prefs_get_handle()); |
| |
1197 } |