Theme Editor: Factored out code to skip over enum/arg lists while scanning for childr...
[kugel-rb.git] / apps / menus / playback_menu.c
blobd5bfd3df0a62cb1910687476ee1490082b11c4cc
1 /***************************************************************************
2 * __________ __ ___.
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
7 * \/ \/ \/ \/ \/
8 * $Id$
10 * Copyright (C) 2007 Jonathan Gordon
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation; either version 2
15 * of the License, or (at your option) any later version.
17 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
18 * KIND, either express or implied.
20 ****************************************************************************/
22 #include <stdbool.h>
23 #include <stddef.h>
24 #include <limits.h>
25 #include "config.h"
26 #include "lang.h"
27 #include "action.h"
28 #include "splash.h"
29 #include "settings.h"
30 #include "menu.h"
31 #include "sound_menu.h"
32 #include "kernel.h"
33 #include "playlist.h"
34 #include "dsp.h"
35 #include "scrobbler.h"
36 #include "audio.h"
37 #include "cuesheet.h"
38 #if CONFIG_CODEC == SWCODEC
39 #include "playback.h"
40 #endif
43 #if (CONFIG_CODEC == SWCODEC) && defined(HAVE_CROSSFADE)
44 static int setcrossfadeonexit_callback(int action,const struct menu_item_ex *this_item)
46 (void)this_item;
47 switch (action)
49 case ACTION_EXIT_MENUITEM: /* on exit */
50 audio_set_crossfade(global_settings.crossfade);
51 break;
53 return action;
56 #endif /* CONFIG_CODEC == SWCODEC */
58 /***********************************/
59 /* PLAYBACK MENU */
60 static int playback_callback(int action,const struct menu_item_ex *this_item);
62 MENUITEM_SETTING(shuffle_item, &global_settings.playlist_shuffle, playback_callback);
63 MENUITEM_SETTING(repeat_mode, &global_settings.repeat_mode, playback_callback);
64 MENUITEM_SETTING(play_selected, &global_settings.play_selected, NULL);
66 MENUITEM_SETTING(ff_rewind_accel, &global_settings.ff_rewind_accel, NULL);
67 MENUITEM_SETTING(ff_rewind_min_step, &global_settings.ff_rewind_min_step, NULL);
68 MAKE_MENU(ff_rewind_settings_menu, ID2P(LANG_WIND_MENU), 0, Icon_NOICON,
69 &ff_rewind_min_step, &ff_rewind_accel);
70 #ifdef HAVE_DISK_STORAGE
71 #if CONFIG_CODEC == SWCODEC
72 static int buffermargin_callback(int action,const struct menu_item_ex *this_item)
74 (void)this_item;
75 switch (action)
77 case ACTION_EXIT_MENUITEM: /* on exit */
78 audio_set_buffer_margin(global_settings.buffer_margin);
79 break;
81 return action;
83 #else
84 # define buffermargin_callback NULL
85 #endif
86 MENUITEM_SETTING(buffer_margin, &global_settings.buffer_margin,
87 buffermargin_callback);
88 #endif /*HAVE_DISK_STORAGE */
89 MENUITEM_SETTING(fade_on_stop, &global_settings.fade_on_stop, NULL);
90 MENUITEM_SETTING(party_mode, &global_settings.party_mode, NULL);
92 #if CONFIG_CODEC == SWCODEC
93 #ifdef HAVE_CROSSFADE
94 /* crossfade submenu */
95 MENUITEM_SETTING(crossfade, &global_settings.crossfade, setcrossfadeonexit_callback);
96 MENUITEM_SETTING(crossfade_fade_in_delay,
97 &global_settings.crossfade_fade_in_delay, NULL);
98 MENUITEM_SETTING(crossfade_fade_in_duration,
99 &global_settings.crossfade_fade_in_duration, NULL);
100 MENUITEM_SETTING(crossfade_fade_out_delay,
101 &global_settings.crossfade_fade_out_delay, setcrossfadeonexit_callback);
102 MENUITEM_SETTING(crossfade_fade_out_duration,
103 &global_settings.crossfade_fade_out_duration, setcrossfadeonexit_callback);
104 MENUITEM_SETTING(crossfade_fade_out_mixmode,
105 &global_settings.crossfade_fade_out_mixmode,NULL);
106 MAKE_MENU(crossfade_settings_menu,ID2P(LANG_CROSSFADE),0, Icon_NOICON,
107 &crossfade, &crossfade_fade_in_delay, &crossfade_fade_in_duration,
108 &crossfade_fade_out_delay, &crossfade_fade_out_duration,
109 &crossfade_fade_out_mixmode);
110 #endif
112 /* replay gain submenu */
114 static int replaygain_callback(int action,const struct menu_item_ex *this_item)
116 (void)this_item;
117 switch (action)
119 case ACTION_EXIT_MENUITEM: /* on exit */
120 dsp_set_replaygain();
121 break;
123 return action;
125 MENUITEM_SETTING(replaygain_noclip, &global_settings.replaygain_noclip ,replaygain_callback);
126 MENUITEM_SETTING(replaygain_type, &global_settings.replaygain_type ,replaygain_callback);
127 MENUITEM_SETTING(replaygain_preamp, &global_settings.replaygain_preamp ,replaygain_callback);
128 MAKE_MENU(replaygain_settings_menu,ID2P(LANG_REPLAYGAIN),0, Icon_NOICON,
129 &replaygain_type, &replaygain_noclip, &replaygain_preamp);
131 MENUITEM_SETTING(beep, &global_settings.beep ,NULL);
132 #endif /* CONFIG_CODEC == SWCODEC */
134 #ifdef HAVE_SPDIF_POWER
135 MENUITEM_SETTING(spdif_enable, &global_settings.spdif_enable, NULL);
136 #endif
137 MENUITEM_SETTING(next_folder, &global_settings.next_folder, NULL);
138 static int audioscrobbler_callback(int action,const struct menu_item_ex *this_item)
140 (void)this_item;
141 switch (action)
143 case ACTION_EXIT_MENUITEM: /* on exit */
144 if (!scrobbler_is_enabled() && global_settings.audioscrobbler)
145 splash(HZ*2, ID2P(LANG_PLEASE_REBOOT));
147 if(scrobbler_is_enabled() && !global_settings.audioscrobbler)
148 scrobbler_shutdown();
149 break;
151 return action;
153 MENUITEM_SETTING(audioscrobbler, &global_settings.audioscrobbler, audioscrobbler_callback);
156 static int cuesheet_callback(int action,const struct menu_item_ex *this_item)
158 (void)this_item;
159 switch (action)
161 case ACTION_EXIT_MENUITEM: /* on exit */
162 if (global_settings.cuesheet)
163 splash(HZ*2, ID2P(LANG_PLEASE_REBOOT));
164 break;
166 return action;
168 MENUITEM_SETTING(cuesheet, &global_settings.cuesheet, cuesheet_callback);
170 #ifdef HAVE_HEADPHONE_DETECTION
171 MENUITEM_SETTING(unplug_mode, &global_settings.unplug_mode, NULL);
172 MENUITEM_SETTING(unplug_rw, &global_settings.unplug_rw, NULL);
173 MENUITEM_SETTING(unplug_autoresume, &global_settings.unplug_autoresume, NULL);
174 MAKE_MENU(unplug_menu, ID2P(LANG_HEADPHONE_UNPLUG), 0, Icon_NOICON,
175 &unplug_mode, &unplug_rw, &unplug_autoresume);
176 #endif
178 MENUITEM_SETTING(skip_length, &global_settings.skip_length, NULL);
179 MENUITEM_SETTING(prevent_skip, &global_settings.prevent_skip, NULL);
180 #if CONFIG_CODEC == SWCODEC
181 MENUITEM_SETTING(resume_rewind, &global_settings.resume_rewind, NULL);
182 #endif
184 MAKE_MENU(playback_settings,ID2P(LANG_PLAYBACK),0,
185 Icon_Playback_menu,
186 &shuffle_item, &repeat_mode, &play_selected,
187 &ff_rewind_settings_menu,
188 #ifdef HAVE_DISK_STORAGE
189 &buffer_margin,
190 #endif
191 &fade_on_stop, &party_mode,
193 #if (CONFIG_CODEC == SWCODEC) && defined(HAVE_CROSSFADE)
194 &crossfade_settings_menu,
195 #endif
197 #if CONFIG_CODEC == SWCODEC
198 &replaygain_settings_menu, &beep,
199 #endif
201 #ifdef HAVE_SPDIF_POWER
202 &spdif_enable,
203 #endif
204 &next_folder, &audioscrobbler, &cuesheet
205 #ifdef HAVE_HEADPHONE_DETECTION
206 ,&unplug_menu
207 #endif
208 ,&skip_length, &prevent_skip,
210 #if CONFIG_CODEC == SWCODEC
211 &resume_rewind,
212 #endif
215 static int playback_callback(int action,const struct menu_item_ex *this_item)
217 static bool old_shuffle = false;
218 static int old_repeat_mode = 0;
219 (void)this_item;
220 switch (action)
222 case ACTION_ENTER_MENUITEM:
223 if (this_item == &shuffle_item)
224 old_shuffle = global_settings.playlist_shuffle;
225 else if (this_item == &repeat_mode)
226 old_repeat_mode = global_settings.repeat_mode;
227 break;
228 case ACTION_EXIT_MENUITEM: /* on exit */
229 if ((this_item == &shuffle_item) &&
230 (old_shuffle != global_settings.playlist_shuffle)
231 && (audio_status() & AUDIO_STATUS_PLAY))
233 #if CONFIG_CODEC == SWCODEC
234 dsp_set_replaygain();
235 #endif
236 if (global_settings.playlist_shuffle)
238 playlist_randomise(NULL, current_tick, true);
240 else
242 playlist_sort(NULL, true);
245 break;
247 return action;
249 /* PLAYBACK MENU */
250 /***********************************/