OSDN Git Service

LinGui: Presets and preferences are now stored as plists
[handbrake-jp/handbrake-jp-git.git] / gtk / src / presets.c
1 /* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 4; tab-width: 4 -*- */
2 /*
3  * presets.c
4  * Copyright (C) John Stebbins 2008 <stebbins@stebbins>
5  * 
6  * presets.c is free software.
7  * 
8  * You may redistribute it and/or modify it under the terms of the
9  * GNU General Public License, as published by the Free Software
10  * Foundation; either version 2 of the License, or (at your option)
11  * any later version.
12  * 
13  */
14 #include <glib.h>
15 #include <glib-object.h>
16 #include <glib/gstdio.h>
17 #include <string.h>
18 #include <gtk/gtk.h>
19 #include "settings.h"
20 #include "plist.h"
21 #include "presets.h"
22 #include "values.h"
23
24 //
25 // Internal defaults stored in character arrays and parsed
26 // the same as external settings files.
27 const gchar defaultSettings[] =
28 #include "internal_defaults.h"
29 ;
30 const gchar standardPresets[] =
31 #include "standard_presets.h"
32 ;
33
34 static GValue *presetsPlist = NULL;
35 static GValue *internalPlist = NULL;
36 static GValue *prefsPlist = NULL;
37
38 static GValue*
39 plist_get_dict(GValue *presets, const gchar *name)
40 {
41         if (presets == NULL || name == NULL) return NULL;
42         return ghb_dict_lookup(presets, name);
43 }
44
45 void
46 ghb_set_preset_default(GValue *settings)
47 {
48         gchar *preset;
49         
50         preset = ghb_settings_get_string (settings, "preset");
51         ghb_settings_set_string(settings, "default_preset", preset);
52         ghb_prefs_save(settings);
53         g_free(preset);
54 }
55
56 // Used for sorting dictionaries.
57 gint
58 key_cmp(gconstpointer a, gconstpointer b)
59 {
60         gchar *stra = (gchar*)a;
61         gchar *strb = (gchar*)b;
62
63         return strcmp(stra, strb);
64 }
65
66 gchar*
67 ghb_presets_get_description(const gchar *name)
68 {
69         GValue *pdict;
70         pdict = plist_get_dict(presetsPlist, name);
71         if (pdict == NULL) return g_strdup("");
72         return ghb_value_string(ghb_dict_lookup(pdict, "preset_description"));
73 }
74
75 static const GValue*
76 preset_dict_get_value(
77         GValue *dict,
78         const gchar *key)
79 {
80         const GValue *gval = NULL;
81
82         if (dict)
83         {
84                 gval = ghb_dict_lookup(dict, key);
85         }
86         if (internalPlist == NULL) return NULL;
87         if (gval == NULL)
88         {
89                 dict = plist_get_dict(internalPlist, "Presets");
90                 if (dict == NULL) return NULL;
91                 gval = ghb_dict_lookup(dict, key);
92         }
93         return gval;
94 }
95
96 static const GValue*
97 preset_get_value(
98         const gchar *name,
99         const gchar *key)
100 {
101         GValue *dict;
102
103         dict = plist_get_dict(presetsPlist, name);
104         return preset_dict_get_value(dict, key);
105 }
106
107 GList*
108 ghb_presets_get_names()
109 {
110         GHashTable *dict;
111         GList *names, *link;
112         GList *standard = NULL;
113         GList *custom = NULL;
114
115         if (presetsPlist == NULL) return NULL;
116         dict = g_value_get_boxed(presetsPlist);
117         link = names = g_hash_table_get_keys(dict);
118         while (link)
119         {
120                 gchar *name;
121                 gint ptype;
122
123                 name = (gchar*)link->data;
124                 ptype = ghb_value_int(preset_get_value(name, "preset_type"));
125                 if (ptype)
126                         custom = g_list_append(custom, name);
127                 else
128                         standard = g_list_append(standard, name);
129                 link = link->next;
130         }
131         custom = g_list_sort(custom, key_cmp);
132         standard = g_list_sort(standard, key_cmp);
133         g_list_free(names);
134         names = g_list_concat(standard, custom);
135         return names;
136 }
137
138 gint
139 ghb_preset_flags(const gchar *name)
140 {
141         GValue *dict;
142         const GValue *gval;
143         gint ptype;
144         gint ret = 0;
145
146         dict = plist_get_dict(presetsPlist, name);
147         gval = preset_dict_get_value(dict, "preset_type");
148         if (gval)
149         {
150                 ptype = ghb_value_int(gval);
151                 ret = (ptype != 0 ? PRESET_CUSTOM : 0);
152         }
153         return ret;
154 }
155
156 static void init_settings_from_dict(
157         GValue *dest, GValue *internal, GValue *dict);
158
159 static void
160 init_settings_from_array(
161         GValue *dest, 
162         GValue *internal,
163         GValue *array)
164 {
165         GValue *gval, *val;
166         gint count, ii;
167         
168         count = ghb_array_len(array);
169         // The first element of the internal version is always the 
170         // template for the allowed values
171         gval = ghb_array_get_nth(internal, 0);
172         for (ii = 0; ii < count; ii++)
173         {
174                 val = NULL;
175                 val = ghb_array_get_nth(array, ii);
176                 if (val == NULL)
177                         val = gval;
178                 if (G_VALUE_TYPE(gval) == ghb_dict_get_type())
179                 {
180                         GValue *new_dict;
181                         new_dict = ghb_dict_value_new();
182                         ghb_array_append(dest, new_dict);
183                         if (G_VALUE_TYPE(val) == ghb_dict_get_type())
184                                 init_settings_from_dict(new_dict, gval, val);
185                         else
186                                 init_settings_from_dict(new_dict, gval, gval);
187                 }
188                 else if (G_VALUE_TYPE(gval) == ghb_array_get_type())
189                 {
190                         GValue *new_array;
191                         new_array = ghb_array_value_new(8);
192                         ghb_array_append(dest, new_array);
193                         if (G_VALUE_TYPE(val) == ghb_array_get_type())
194                                 init_settings_from_array(new_array, gval, val);
195                         else
196                                 init_settings_from_array(new_array, gval, gval);
197                 }
198                 else
199                 {
200                         ghb_array_append(dest, val);
201                 }
202         }
203 }
204
205 static void
206 init_settings_from_dict(
207         GValue *dest, 
208         GValue *internal,
209         GValue *dict)
210 {
211         GHashTableIter iter;
212         gchar *key;
213         GValue *gval, *val;
214         
215         ghb_dict_iter_init(&iter, internal);
216         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
217         {
218                 val = NULL;
219                 if (dict)
220                         val = ghb_dict_lookup(dict, key);
221                 if (val == NULL)
222                         val = gval;
223                 if (G_VALUE_TYPE(gval) == ghb_dict_get_type())
224                 {
225                         GValue *new_dict;
226                         new_dict = ghb_dict_value_new();
227                         ghb_settings_take_value(dest, key, new_dict);
228                         if (G_VALUE_TYPE(val) == ghb_dict_get_type())
229                                 init_settings_from_dict(new_dict, gval, val);
230                         else
231                                 init_settings_from_dict(new_dict, gval, gval);
232                 }
233                 else if (G_VALUE_TYPE(gval) == ghb_array_get_type())
234                 {
235                         GValue *new_array;
236                         new_array = ghb_array_value_new(8);
237                         ghb_settings_take_value(dest, key, new_array);
238                         if (G_VALUE_TYPE(val) == ghb_array_get_type())
239                                 init_settings_from_array(new_array, gval, val);
240                         else
241                                 init_settings_from_array(new_array, gval, gval);
242         
243                 }
244                 else
245                 {
246                         ghb_settings_set_value(dest, key, val);
247                 }
248         }
249 }
250
251 void
252 init_ui_from_dict(
253         signal_user_data_t *ud, 
254         GValue *internal,
255         GValue *dict)
256 {
257         GHashTableIter iter;
258         gchar *key;
259         GValue *gval, *val;
260         
261         ghb_dict_iter_init(&iter, internal);
262         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
263         {
264                 val = NULL;
265                 if (dict)
266                         val = ghb_dict_lookup(dict, key);
267                 if (val == NULL)
268                         val = gval;
269                 ghb_ui_update(ud, key, val);
270         }
271 }
272
273 static void
274 preset_to_ui(signal_user_data_t *ud, GValue *dict)
275 {
276         g_debug("preset_to_ui()\n");
277         // Initialize the ui from presets file.
278         GValue *internal;
279
280         // Get key list from internal default presets.  This way we do not
281         // load any unknown keys.
282         if (internalPlist == NULL) return;
283         internal = plist_get_dict(internalPlist, "Presets");
284         // Setting a ui widget will cause the corresponding setting
285         // to be set, but it also triggers a callback that can 
286         // have the side effect of using other settings values
287         // that have not yet been set.  So set *all* settings first
288         // then update the ui.
289         init_settings_from_dict(ud->settings, internal, dict);
290         init_ui_from_dict(ud, internal, dict);
291
292         if (ghb_settings_get_boolean(ud->settings, "allow_tweaks"))
293         {
294                 const GValue *gval;
295                 gval = preset_dict_get_value(dict, "deinterlace");
296                 if (gval)
297                 {
298                         ghb_ui_update(ud, "tweak_deinterlace", gval);
299                 }
300                 gval = preset_dict_get_value(dict, "denoise");
301                 if (gval)
302                 {
303                         ghb_ui_update(ud, "tweak_denoise", gval);
304                 }
305         }
306 }
307
308 void
309 ghb_set_preset(signal_user_data_t *ud, const gchar *name)
310 {
311         GValue *dict;
312         
313         g_debug("ghb_set_preset() %s\n", name);
314         if (name == NULL)
315         {
316                 GList *presets;
317                 // Try to get the first preset
318                 presets = ghb_presets_get_names();
319                 if (presets)
320                 {
321                         name = (const gchar*)presets->data;
322                         g_list_free(presets);
323                 }
324         }
325         dict = plist_get_dict(presetsPlist, name);
326         if (dict == NULL || name == NULL)
327         {
328                 preset_to_ui(ud, NULL);
329         }
330         else
331         {
332                 preset_to_ui(ud, dict);
333                 ghb_settings_set_string(ud->settings, "preset", name);
334         }
335 }
336
337 void
338 ghb_update_from_preset(
339         signal_user_data_t *ud, 
340         const gchar *name, 
341         const gchar *key)
342 {
343         const GValue *gval;
344         
345         g_debug("ghb_update_from_preset() %s %s", name, key);
346         if (name == NULL) return;
347         gval = preset_get_value(name, key);
348         if (gval != NULL)
349         {
350                 ghb_ui_update(ud, key, gval);
351         }
352 }
353
354 static void
355 store_plist(GValue *plist, const gchar *name)
356 {
357         const gchar *dir;
358         gchar *config;
359         FILE *file;
360
361         dir = g_get_user_config_dir();
362         config = g_strdup_printf ("%s/ghb", dir);
363         if (!g_file_test(config, G_FILE_TEST_IS_DIR))
364         {
365                 g_mkdir (config, 0755);
366         }
367         g_free(config);
368         config = g_strdup_printf ("%s/ghb/%s", dir, name);
369         file = g_fopen(config, "w");
370         g_free(config);
371         ghb_plist_write(file, plist);
372         fclose(file);
373 }
374
375 static gboolean prefs_initializing = FALSE;
376
377 void
378 ghb_prefs_to_ui(signal_user_data_t *ud)
379 {
380         const GValue *gval;
381         gchar *key;
382         gchar *str;
383         GValue *internal, *dict;
384         GHashTableIter iter;
385         
386
387         prefs_initializing = TRUE;
388
389         // Setting a ui widget will cause the corresponding setting
390         // to be set, but it also triggers a callback that can 
391         // have the side effect of using other settings values
392         // that have not yet been set.  So set *all* settings first
393         // then update the ui.
394         internal = plist_get_dict(internalPlist, "Initialization");
395         ghb_dict_iter_init(&iter, internal);
396         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
397         {
398                 ghb_ui_update(ud, key, gval);
399         }
400
401         dict = plist_get_dict(prefsPlist, "Preferences");
402         internal = plist_get_dict(internalPlist, "Preferences");
403         ghb_dict_iter_init(&iter, internal);
404         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
405     {
406                 const GValue *value = NULL;
407                 if (dict)
408                         value = ghb_dict_lookup(dict, key);
409                 if (value == NULL)
410                         value = gval;
411                 ghb_settings_set_value(ud->settings, key, value);
412     }
413         internal = plist_get_dict(internalPlist, "Preferences");
414         ghb_dict_iter_init(&iter, internal);
415         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
416         {
417                 const GValue *value = NULL;
418                 if (dict)
419                         value = ghb_dict_lookup(dict, key);
420                 if (value == NULL)
421                         value = gval;
422                 ghb_ui_update(ud, key, value);
423         }
424         const GValue *val;
425         val = ghb_settings_get_value(ud->settings, "show_presets");
426         ghb_ui_update(ud, "show_presets", val);
427         if (ghb_settings_get_boolean(ud->settings, "hbfd_feature"))
428         {
429                 GtkAction *action;
430                 val = ghb_settings_get_value(ud->settings, "hbfd");
431                 ghb_ui_update(ud, "hbfd", val);
432                 action = GHB_ACTION (ud->builder, "hbfd");
433                 gtk_action_set_visible(action, TRUE);
434         }
435         else
436         {
437                 ghb_ui_update(ud, "hbfd", ghb_int64_value(0));
438         }
439         gval = ghb_settings_get_value(ud->settings, "default_source");
440         ghb_settings_set_value (ud->settings, "source", gval);
441         str = ghb_settings_get_string(ud->settings, "destination_dir");
442
443         gchar *path = g_strdup_printf ("%s/new_video.mp4", str);
444         ghb_ui_update(ud, "destination", ghb_string_value(path));
445         g_free(str);
446         g_free(path);
447
448         prefs_initializing = FALSE;
449 }
450
451 void
452 ghb_prefs_save(GValue *settings)
453 {
454         GValue *dict;
455         GValue *pref_dict;
456         GHashTableIter iter;
457         gchar *key;
458         const GValue *value;
459         
460         if (prefs_initializing) return;
461         dict = plist_get_dict(internalPlist, "Preferences");
462         if (dict == NULL) return;
463         pref_dict = plist_get_dict(prefsPlist, "Preferences");
464         if (pref_dict == NULL) return;
465         ghb_dict_iter_init(&iter, dict);
466     while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&value))
467     {
468             value = ghb_settings_get_value(settings, key);
469             if (value != NULL)
470             {
471                         ghb_dict_insert(pref_dict, g_strdup(key), ghb_value_dup(value));
472             }
473         }
474     store_plist(prefsPlist, "preferences");
475 }
476
477 void
478 ghb_pref_save(GValue *settings, const gchar *key)
479 {
480         const GValue *value;
481         
482         if (prefs_initializing) return;
483         value = ghb_settings_get_value(settings, key);
484         if (value != NULL)
485         {
486                 GValue *dict;
487                 dict = plist_get_dict(prefsPlist, "Preferences");
488                 if (dict == NULL) return;
489                 ghb_dict_insert(dict, g_strdup(key), ghb_value_dup(value));
490                 store_plist(prefsPlist, "preferences");
491         }
492 }
493
494 void
495 ghb_settings_init(signal_user_data_t *ud)
496 {
497         GValue *internal;
498         GHashTableIter iter;
499         gchar *key;
500         GValue *gval;
501
502
503         g_debug("ghb_settings_init");
504         prefs_initializing = TRUE;
505
506         internalPlist = ghb_plist_parse(defaultSettings, sizeof(defaultSettings)-1);
507         // Setting a ui widget will cause the corresponding setting
508         // to be set, but it also triggers a callback that can 
509         // have the side effect of using other settings values
510         // that have not yet been set.  So set *all* settings first
511         // then update the ui.
512         internal = plist_get_dict(internalPlist, "Initialization");
513         ghb_dict_iter_init(&iter, internal);
514         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
515         {
516                 ghb_settings_set_value(ud->settings, key, gval);
517         }
518
519         internal = plist_get_dict(internalPlist, "Presets");
520         ghb_dict_iter_init(&iter, internal);
521         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
522         {
523                 ghb_settings_set_value(ud->settings, key, gval);
524         }
525
526         internal = plist_get_dict(internalPlist, "Preferences");
527         ghb_dict_iter_init(&iter, internal);
528         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
529         {
530                 ghb_settings_set_value(ud->settings, key, gval);
531         }
532         prefs_initializing = FALSE;
533 }
534
535 void
536 ghb_prefs_load(signal_user_data_t *ud)
537 {
538         const gchar *dir;
539         gchar *config;
540         GValue *dict, *internal;
541         GHashTableIter iter;
542         gchar *key;
543         GValue *gval;
544         
545         g_debug("ghb_prefs_load");
546         dir = g_get_user_config_dir();
547         config = g_strdup_printf ("%s/ghb/preferences", dir);
548         if (g_file_test(config, G_FILE_TEST_IS_REGULAR))
549         {
550                 prefsPlist = ghb_plist_parse_file(config);
551                 if (prefsPlist == NULL)
552                         prefsPlist = ghb_dict_value_new();
553         }
554         else
555         {
556                 // Make an empty plist
557                 prefsPlist = ghb_dict_value_new();
558         }
559         g_free(config);
560         dict = plist_get_dict(prefsPlist, "Preferences");
561         internal = plist_get_dict(internalPlist, "Preferences");
562     if (dict == NULL && internal)
563     {
564                 dict = ghb_dict_value_new();
565                 ghb_dict_insert(prefsPlist, g_strdup("Preferences"), dict);
566
567         // Get defaults from internal defaults 
568                 ghb_dict_iter_init(&iter, internal);
569                 while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&gval))
570         {
571                         ghb_dict_insert(dict, g_strdup(key), ghb_value_dup(gval));
572         }
573                 const gchar *dir = g_get_user_special_dir (G_USER_DIRECTORY_VIDEOS);
574                 ghb_dict_insert(dict, 
575                         g_strdup("destination_dir"), ghb_value_dup(ghb_string_value(dir)));
576                 store_plist(prefsPlist, "preferences");
577     }
578
579 }
580
581 void
582 ghb_presets_reload(signal_user_data_t *ud)
583 {
584         GValue *std_dict, *dict;
585         GHashTableIter std_iter;
586
587         g_debug("ghb_presets_reload()\n");
588         std_dict = ghb_plist_parse(standardPresets, sizeof(standardPresets)-1);
589         if (std_dict == NULL) return;
590
591         // Merge the keyfile contents into our presets
592         gchar *name;
593         GValue *orig_dict;
594
595         ghb_dict_iter_init(&std_iter, std_dict);
596         while (g_hash_table_iter_next(&std_iter, (gpointer*)&name, (gpointer*)&orig_dict))
597         {
598                 GHashTableIter iter;
599                 gchar *key;
600                 GValue *value;
601
602                 dict = ghb_dict_value_new();
603                 ghb_dict_insert(presetsPlist, g_strdup(name), dict);
604                 ghb_dict_iter_init(&iter, orig_dict);
605                 while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&value))
606                 {
607                         ghb_dict_insert(dict, g_strdup(key), ghb_value_dup(value));
608                 }
609         }
610         ghb_value_free(std_dict);
611         store_plist(presetsPlist, "presets");
612 }
613
614 static void
615 presets_store()
616 {
617         g_debug("presets_store ()\n");
618         store_plist(presetsPlist, "presets");
619 }
620
621 void
622 ghb_presets_load()
623 {
624         const gchar *dir;
625         gchar *config;
626
627         dir = g_get_user_config_dir();
628         config = g_strdup_printf ("%s/ghb/presets", dir);
629         if (g_file_test(config, G_FILE_TEST_IS_REGULAR))
630         {
631                 presetsPlist = ghb_plist_parse_file(config);
632                 if (presetsPlist == NULL)
633                 {
634                         presetsPlist = ghb_plist_parse(
635                                 standardPresets, sizeof(standardPresets)-1);
636                         presets_store();
637                 }
638         }
639         else
640         {
641                 presetsPlist = ghb_plist_parse(
642                         standardPresets, sizeof(standardPresets)-1);
643         }
644         g_free(config);
645 }
646
647 void
648 ghb_settings_save(signal_user_data_t *ud, const gchar *name)
649 {
650         GValue *dict, *internal;
651         GHashTableIter iter;
652         gchar *key;
653         GValue *value;
654         gboolean autoscale;
655
656         if (internalPlist == NULL) return;
657         if (ghb_settings_get_boolean(ud->settings, "allow_tweaks"))
658         {
659                 gchar *str;
660                 str = ghb_settings_get_string(ud->settings, "tweak_deinterlace");
661                 if (str)
662                 {
663                         ghb_settings_set_string(ud->settings, "deinterlace", str);
664                         g_free(str);
665                 }
666                 str = ghb_settings_get_string(ud->settings, "tweak_denoise");
667                 if (str)
668                 {
669                         ghb_settings_set_string(ud->settings, "denoise", str);
670                         g_free(str);
671                 }
672         }
673         autoscale = ghb_settings_get_boolean(ud->settings, "autoscale");
674         ghb_settings_set_int64(ud->settings, "preset_type", 1);
675
676         dict = ghb_dict_value_new();
677         ghb_dict_insert(presetsPlist, g_strdup(name), dict);
678         internal = plist_get_dict(internalPlist, "Presets");
679
680         ghb_dict_iter_init(&iter, internal);
681         while (g_hash_table_iter_next(&iter, (gpointer*)&key, (gpointer*)&value))
682         {
683                 if (!autoscale)
684                 {
685                         if (strcmp(key, "scale_width"))
686                         {
687                                 key = "max_width";
688                         }
689                         if (strcmp(key, "scale_height"))
690                         {
691                                 key = "max_height";
692                         }
693                 }
694                 const GValue *gval;
695                 gval = ghb_settings_get_value(ud->settings, key);
696                 if (gval == NULL)
697                 {
698                         g_debug("Setting (%s) is not in defaults\n", (gchar*)key);
699                         continue;
700                 }
701                 if (ghb_value_cmp(gval, value) != 0)
702                 {
703                         // Differs from default value.  Store it.
704                         ghb_dict_insert(dict, g_strdup(key), ghb_value_dup(gval));
705                 }
706         }
707         presets_store();
708         ud->dont_clear_presets = TRUE;
709         ghb_set_preset (ud, name);
710         ud->dont_clear_presets = FALSE;
711 }
712
713 void
714 ghb_presets_remove(const gchar *name)
715 {
716         if (ghb_dict_lookup(presetsPlist, name))
717         {
718                 ghb_dict_remove(presetsPlist, name);
719                 presets_store();
720         }
721 }
722