14192
|
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 }
|