]> git.sesse.net Git - vlc/blob - src/input/var.c
var: undef Windef.h CALLBACK as we use it here.
[vlc] / src / input / var.c
1 /*****************************************************************************
2  * var.c: object variables for input thread
3  *****************************************************************************
4  * Copyright (C) 2004-2007 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Laurent Aimar <fenrir@via.ecp.fr>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27 #ifdef HAVE_CONFIG_H
28 # include "config.h"
29 #endif
30
31 #include <vlc_common.h>
32 #include <stdio.h>
33 #include <stdlib.h>
34
35 #include "input_internal.h"
36
37 /*****************************************************************************
38  * Callbacks
39  *****************************************************************************/
40 static int StateCallback   ( vlc_object_t *p_this, char const *psz_cmd,
41                              vlc_value_t oldval, vlc_value_t newval, void * );
42 static int RateCallback    ( vlc_object_t *p_this, char const *psz_cmd,
43                              vlc_value_t oldval, vlc_value_t newval, void * );
44 static int PositionCallback( vlc_object_t *p_this, char const *psz_cmd,
45                              vlc_value_t oldval, vlc_value_t newval, void * );
46 static int TimeCallback    ( vlc_object_t *p_this, char const *psz_cmd,
47                              vlc_value_t oldval, vlc_value_t newval, void * );
48 static int ProgramCallback ( vlc_object_t *p_this, char const *psz_cmd,
49                              vlc_value_t oldval, vlc_value_t newval, void * );
50 static int TitleCallback   ( vlc_object_t *p_this, char const *psz_cmd,
51                              vlc_value_t oldval, vlc_value_t newval, void * );
52 static int SeekpointCallback( vlc_object_t *p_this, char const *psz_cmd,
53                              vlc_value_t oldval, vlc_value_t newval, void * );
54 static int NavigationCallback( vlc_object_t *p_this, char const *psz_cmd,
55                              vlc_value_t oldval, vlc_value_t newval, void * );
56 static int ESCallback      ( vlc_object_t *p_this, char const *psz_cmd,
57                              vlc_value_t oldval, vlc_value_t newval, void * );
58 static int EsDelayCallback ( vlc_object_t *p_this, char const *psz_cmd,
59                              vlc_value_t oldval, vlc_value_t newval, void * );
60
61 static int BookmarkCallback( vlc_object_t *p_this, char const *psz_cmd,
62                              vlc_value_t oldval, vlc_value_t newval, void * );
63
64 typedef struct
65 {
66     const char *psz_name;
67     vlc_callback_t callback;
68 } vlc_input_callback_t;
69 static void InputAddCallbacks( input_thread_t *, const vlc_input_callback_t * );
70 static void InputDelCallbacks( input_thread_t *, const vlc_input_callback_t * );
71
72 #ifdef CALLBACK /* For windows */
73 # undef CALLBACK /* We don't care of this one here */
74 #endif
75 /* List all callbacks added by input */
76 #define CALLBACK(name,cb) { name, cb }
77 static const vlc_input_callback_t p_input_callbacks[] =
78 {
79     CALLBACK( "state", StateCallback ),
80     CALLBACK( "rate", RateCallback ),
81     CALLBACK( "rate-slower", RateCallback ),
82     CALLBACK( "rate-faster", RateCallback ),
83     CALLBACK( "position", PositionCallback ),
84     CALLBACK( "position-offset", PositionCallback ),
85     CALLBACK( "time", TimeCallback ),
86     CALLBACK( "time-offset", TimeCallback ),
87     CALLBACK( "bookmark", BookmarkCallback ),
88     CALLBACK( "program", ProgramCallback ),
89     CALLBACK( "title", TitleCallback ),
90     CALLBACK( "chapter", SeekpointCallback ),
91     CALLBACK( "audio-delay", EsDelayCallback ),
92     CALLBACK( "spu-delay", EsDelayCallback ),
93     CALLBACK( "video-es", ESCallback ),
94     CALLBACK( "audio-es", ESCallback ),
95     CALLBACK( "spu-es", ESCallback ),
96
97     CALLBACK( NULL, NULL )
98 };
99 static const vlc_input_callback_t p_input_navigation_callbacks[] =
100 {
101     CALLBACK( "next-title", TitleCallback ),
102     CALLBACK( "prev-title", TitleCallback ),
103
104     CALLBACK( NULL, NULL )
105 };
106 static const vlc_input_callback_t p_input_title_callbacks[] =
107 {
108     CALLBACK( "next-chapter", SeekpointCallback ),
109     CALLBACK( "prev-chapter", SeekpointCallback ),
110
111     CALLBACK( NULL, NULL )
112 };
113 #undef CALLBACK
114
115 /*****************************************************************************
116  * input_ControlVarInit:
117  *  Create all control object variables with their callbacks
118  *****************************************************************************/
119 void input_ControlVarInit ( input_thread_t *p_input )
120 {
121     vlc_value_t val, text;
122
123     /* State */
124     var_Create( p_input, "state", VLC_VAR_INTEGER );
125     val.i_int = p_input->i_state;
126     var_Change( p_input, "state", VLC_VAR_SETVALUE, &val, NULL );
127
128     /* Rate */
129     var_Create( p_input, "rate", VLC_VAR_INTEGER );
130     val.i_int = p_input->p->i_rate;
131     var_Change( p_input, "rate", VLC_VAR_SETVALUE, &val, NULL );
132
133     var_Create( p_input, "rate-slower", VLC_VAR_VOID );
134
135     var_Create( p_input, "rate-faster", VLC_VAR_VOID );
136
137     /* Position */
138     var_Create( p_input, "position",  VLC_VAR_FLOAT );
139     var_Create( p_input, "position-offset",  VLC_VAR_FLOAT );
140     val.f_float = 0.0;
141     var_Change( p_input, "position", VLC_VAR_SETVALUE, &val, NULL );
142
143     /* Time */
144     var_Create( p_input, "time",  VLC_VAR_TIME );
145     var_Create( p_input, "time-offset",  VLC_VAR_TIME );    /* relative */
146     val.i_time = 0;
147     var_Change( p_input, "time", VLC_VAR_SETVALUE, &val, NULL );
148
149     /* Bookmark */
150     var_Create( p_input, "bookmark", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE |
151                 VLC_VAR_ISCOMMAND );
152     val.psz_string = _("Bookmark");
153     var_Change( p_input, "bookmark", VLC_VAR_SETTEXT, &val, NULL );
154
155     /* Program */
156     var_Create( p_input, "program", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE |
157                 VLC_VAR_DOINHERIT );
158     var_Get( p_input, "program", &val );
159     if( val.i_int <= 0 )
160         var_Change( p_input, "program", VLC_VAR_DELCHOICE, &val, NULL );
161     text.psz_string = _("Program");
162     var_Change( p_input, "program", VLC_VAR_SETTEXT, &text, NULL );
163
164     /* Programs */
165     var_Create( p_input, "programs", VLC_VAR_LIST | VLC_VAR_DOINHERIT );
166     text.psz_string = _("Programs");
167     var_Change( p_input, "programs", VLC_VAR_SETTEXT, &text, NULL );
168
169     /* Title */
170     var_Create( p_input, "title", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE );
171     text.psz_string = _("Title");
172     var_Change( p_input, "title", VLC_VAR_SETTEXT, &text, NULL );
173
174     /* Chapter */
175     var_Create( p_input, "chapter", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE );
176     text.psz_string = _("Chapter");
177     var_Change( p_input, "chapter", VLC_VAR_SETTEXT, &text, NULL );
178
179     /* Navigation The callback is added after */
180     var_Create( p_input, "navigation", VLC_VAR_VARIABLE | VLC_VAR_HASCHOICE );
181     text.psz_string = _("Navigation");
182     var_Change( p_input, "navigation", VLC_VAR_SETTEXT, &text, NULL );
183
184     /* Delay */
185     var_Create( p_input, "audio-delay", VLC_VAR_TIME );
186     val.i_time = 0;
187     var_Change( p_input, "audio-delay", VLC_VAR_SETVALUE, &val, NULL );
188     var_Create( p_input, "spu-delay", VLC_VAR_TIME );
189     val.i_time = 0;
190     var_Change( p_input, "spu-delay", VLC_VAR_SETVALUE, &val, NULL );
191
192     p_input->p->pts_adjust.auto_adjust = var_CreateGetBool(
193             p_input, "auto-adjust-pts-delay" );
194
195     /* Video ES */
196     var_Create( p_input, "video-es", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE );
197     text.psz_string = _("Video Track");
198     var_Change( p_input, "video-es", VLC_VAR_SETTEXT, &text, NULL );
199
200     /* Audio ES */
201     var_Create( p_input, "audio-es", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE );
202     text.psz_string = _("Audio Track");
203     var_Change( p_input, "audio-es", VLC_VAR_SETTEXT, &text, NULL );
204
205     /* Spu ES */
206     var_Create( p_input, "spu-es", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE );
207     text.psz_string = _("Subtitles Track");
208     var_Change( p_input, "spu-es", VLC_VAR_SETTEXT, &text, NULL );
209
210     /* Special read only objects variables for intf */
211     var_Create( p_input, "bookmarks", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
212
213     var_Create( p_input, "length",  VLC_VAR_TIME );
214     val.i_time = 0;
215     var_Change( p_input, "length", VLC_VAR_SETVALUE, &val, NULL );
216
217     if( !p_input->b_preparsing )
218     {
219         /* Special "intf-change" variable, it allows intf to set up a callback
220          * to be notified of some changes.
221          *
222          * Add rate-change to inform about rate changin
223          *
224          * TODO list all changes warn by this callbacks */
225         var_Create( p_input, "intf-change", VLC_VAR_BOOL );
226         var_SetBool( p_input, "intf-change", true );
227         var_Create( p_input, "rate-change", VLC_VAR_BOOL );
228         var_SetBool( p_input, "rate-change", true );
229     }
230
231     /* Add all callbacks
232      * XXX we put callback only in non preparsing mode. We need to create the variable
233      * unless someone want to check all var_Get/var_Change return value ... */
234     if( !p_input->b_preparsing )
235         InputAddCallbacks( p_input, p_input_callbacks );
236 }
237
238 /*****************************************************************************
239  * input_ControlVarStop:
240  *****************************************************************************/
241 void input_ControlVarStop( input_thread_t *p_input )
242 {
243     InputDelCallbacks( p_input, p_input_callbacks );
244
245     if( p_input->p->i_title > 0 )
246     {
247         char name[sizeof("title ") + 5 ];
248         int i;
249
250         InputDelCallbacks( p_input, p_input_navigation_callbacks );
251         InputDelCallbacks( p_input, p_input_title_callbacks );
252
253         for( i = 0; i < p_input->p->i_title; i++ )
254         {
255             snprintf( name, sizeof(name), "title %2i", i );
256             var_DelCallback( p_input, name, NavigationCallback, (void *)(intptr_t)i );
257         }
258     }
259 }
260
261 /*****************************************************************************
262  * input_ControlVarNavigation:
263  *  Create all remaining control object variables
264  *****************************************************************************/
265 void input_ControlVarNavigation( input_thread_t *p_input )
266 {
267     vlc_value_t val, text;
268     int  i;
269
270     /* Create more command variables */
271     if( p_input->p->i_title > 1 )
272     {
273         var_Create( p_input, "next-title", VLC_VAR_VOID );
274         text.psz_string = _("Next title");
275         var_Change( p_input, "next-title", VLC_VAR_SETTEXT, &text, NULL );
276         var_AddCallback( p_input, "next-title", TitleCallback, NULL );
277
278         var_Create( p_input, "prev-title", VLC_VAR_VOID );
279         text.psz_string = _("Previous title");
280         var_Change( p_input, "prev-title", VLC_VAR_SETTEXT, &text, NULL );
281         var_AddCallback( p_input, "prev-title", TitleCallback, NULL );
282     }
283
284     /* Create title and navigation */
285     val.psz_string = malloc( sizeof("title ") + 5 );
286     for( i = 0; i < p_input->p->i_title; i++ )
287     {
288         vlc_value_t val2, text, text2;
289         int j;
290
291         /* Add Navigation entries */
292         sprintf( val.psz_string,  "title %2i", i );
293         var_Destroy( p_input, val.psz_string );
294         var_Create( p_input, val.psz_string,
295                     VLC_VAR_INTEGER|VLC_VAR_HASCHOICE|VLC_VAR_ISCOMMAND );
296         var_AddCallback( p_input, val.psz_string,
297                          NavigationCallback, (void *)(intptr_t)i );
298
299         if( p_input->p->title[i]->psz_name == NULL ||
300             *p_input->p->title[i]->psz_name == '\0' )
301         {
302             asprintf( &text.psz_string, _("Title %i"),
303                       i + p_input->p->i_title_offset );
304         }
305         else
306         {
307             text.psz_string = strdup( p_input->p->title[i]->psz_name );
308         }
309         var_Change( p_input, "navigation", VLC_VAR_ADDCHOICE, &val, &text );
310
311         /* Add title choice */
312         val2.i_int = i;
313         var_Change( p_input, "title", VLC_VAR_ADDCHOICE, &val2, &text );
314
315         free( text.psz_string );
316
317         for( j = 0; j < p_input->p->title[i]->i_seekpoint; j++ )
318         {
319             val2.i_int = j;
320
321             if( p_input->p->title[i]->seekpoint[j]->psz_name == NULL ||
322                 *p_input->p->title[i]->seekpoint[j]->psz_name == '\0' )
323             {
324                 /* Default value */
325                 asprintf( &text2.psz_string, _("Chapter %i"),
326                           j + p_input->p->i_seekpoint_offset );
327             }
328             else
329             {
330                 text2.psz_string =
331                     strdup( p_input->p->title[i]->seekpoint[j]->psz_name );
332             }
333
334             var_Change( p_input, val.psz_string, VLC_VAR_ADDCHOICE,
335                         &val2, &text2 );
336             free( text2.psz_string );
337         }
338
339     }
340     free( val.psz_string );
341 }
342
343 /*****************************************************************************
344  * input_ControlVarTitle:
345  *  Create all variables for a title
346  *****************************************************************************/
347 void input_ControlVarTitle( input_thread_t *p_input, int i_title )
348 {
349     input_title_t *t = p_input->p->title[i_title];
350     vlc_value_t val;
351     int  i;
352
353     /* Create/Destroy command variables */
354     if( t->i_seekpoint <= 1 )
355     {
356         var_Destroy( p_input, "next-chapter" );
357         var_Destroy( p_input, "prev-chapter" );
358     }
359     else if( var_Get( p_input, "next-chapter", &val ) != VLC_SUCCESS )
360     {
361         vlc_value_t text;
362
363         var_Create( p_input, "next-chapter", VLC_VAR_VOID );
364         text.psz_string = _("Next chapter");
365         var_Change( p_input, "next-chapter", VLC_VAR_SETTEXT, &text, NULL );
366         var_AddCallback( p_input, "next-chapter", SeekpointCallback, NULL );
367
368         var_Create( p_input, "prev-chapter", VLC_VAR_VOID );
369         text.psz_string = _("Previous chapter");
370         var_Change( p_input, "prev-chapter", VLC_VAR_SETTEXT, &text, NULL );
371         var_AddCallback( p_input, "prev-chapter", SeekpointCallback, NULL );
372     }
373
374     /* Build chapter list */
375     var_Change( p_input, "chapter", VLC_VAR_CLEARCHOICES, NULL, NULL );
376     for( i = 0; i <  t->i_seekpoint; i++ )
377     {
378         vlc_value_t text;
379         val.i_int = i;
380
381         if( t->seekpoint[i]->psz_name == NULL ||
382             *t->seekpoint[i]->psz_name == '\0' )
383         {
384             /* Default value */
385             asprintf( &text.psz_string, _("Chapter %i"),
386                       i + p_input->p->i_seekpoint_offset );
387         }
388         else
389         {
390             text.psz_string = strdup( t->seekpoint[i]->psz_name );
391         }
392
393         var_Change( p_input, "chapter", VLC_VAR_ADDCHOICE, &val, &text );
394         free( text.psz_string );
395     }
396 }
397
398 /*****************************************************************************
399  * input_ConfigVarInit:
400  *  Create all config object variables
401  *****************************************************************************/
402 void input_ConfigVarInit ( input_thread_t *p_input )
403 {
404     vlc_value_t val;
405
406     /* Create Object Variables for private use only */
407
408     if( !p_input->b_preparsing )
409     {
410         var_Create( p_input, "video", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
411         var_Create( p_input, "audio", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
412         var_Create( p_input, "spu", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
413
414         var_Create( p_input, "audio-track", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
415         var_Create( p_input, "sub-track", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
416
417         var_Create( p_input, "audio-language",
418                     VLC_VAR_STRING|VLC_VAR_DOINHERIT );
419         var_Create( p_input, "sub-language",
420                     VLC_VAR_STRING|VLC_VAR_DOINHERIT );
421
422         var_Create( p_input, "audio-track-id",
423                     VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
424         var_Create( p_input, "sub-track-id",
425                     VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
426
427         var_Create( p_input, "sub-file", VLC_VAR_FILE | VLC_VAR_DOINHERIT );
428         var_Create( p_input, "sub-autodetect-file", VLC_VAR_BOOL |
429                     VLC_VAR_DOINHERIT );
430         var_Create( p_input, "sub-autodetect-path", VLC_VAR_STRING |
431                     VLC_VAR_DOINHERIT );
432         var_Create( p_input, "sub-autodetect-fuzzy", VLC_VAR_INTEGER |
433                     VLC_VAR_DOINHERIT );
434
435         var_Create( p_input, "sout", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
436         var_Create( p_input, "sout-all",   VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
437         var_Create( p_input, "sout-audio", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
438         var_Create( p_input, "sout-video", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
439         var_Create( p_input, "sout-spu", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
440         var_Create( p_input, "sout-keep",  VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
441
442         var_Create( p_input, "input-repeat",
443                     VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
444         var_Create( p_input, "start-time", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
445         var_Create( p_input, "stop-time", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
446         var_Create( p_input, "run-time", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
447
448         var_Create( p_input, "input-slave",
449                     VLC_VAR_STRING | VLC_VAR_DOINHERIT );
450
451         var_Create( p_input, "minimize-threads",
452                     VLC_VAR_BOOL|VLC_VAR_DOINHERIT );
453
454         var_Create( p_input, "audio-desync",
455                     VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
456         var_Create( p_input, "cr-average",
457                     VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
458         var_Create( p_input, "clock-synchro",
459                     VLC_VAR_INTEGER | VLC_VAR_DOINHERIT);
460     }
461
462     var_Create( p_input, "seekable", VLC_VAR_BOOL );
463     val.b_bool = true; /* Fixed later*/
464     var_Change( p_input, "seekable", VLC_VAR_SETVALUE, &val, NULL );
465     var_Create( p_input, "can-pause", VLC_VAR_BOOL );
466     val.b_bool = true; /* Fixed later*/
467     var_Change( p_input, "can-pause", VLC_VAR_SETVALUE, &val, NULL );
468     var_Create( p_input, "teletext-es", VLC_VAR_INTEGER );
469     var_SetInteger( p_input, "teletext-es", -1 );
470
471     /* */
472     var_Create( p_input, "access-filter", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
473     var_Create( p_input, "access", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
474     var_Create( p_input, "demux", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
475
476     /* Meta */
477     var_Create( p_input, "meta-title", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
478     var_Create( p_input, "meta-author", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
479     var_Create( p_input, "meta-artist", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
480     var_Create( p_input, "meta-genre", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
481     var_Create( p_input, "meta-copyright", VLC_VAR_STRING | VLC_VAR_DOINHERIT);
482     var_Create( p_input, "meta-description", VLC_VAR_STRING|VLC_VAR_DOINHERIT);
483     var_Create( p_input, "meta-date", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
484     var_Create( p_input, "meta-url", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
485 }
486
487 /*****************************************************************************
488  * Callbacks managements:
489  *****************************************************************************/
490 static void InputAddCallbacks( input_thread_t *p_input,
491                                const vlc_input_callback_t *p_callbacks )
492 {
493     int i;
494     for( i = 0; p_callbacks[i].psz_name != NULL; i++ )
495         var_AddCallback( p_input,
496                          p_callbacks[i].psz_name,
497                          p_callbacks[i].callback, NULL );
498 }
499 static void InputDelCallbacks( input_thread_t *p_input,
500                                const vlc_input_callback_t *p_callbacks )
501 {
502     int i;
503     for( i = 0; p_callbacks[i].psz_name != NULL; i++ )
504         var_DelCallback( p_input,
505                          p_callbacks[i].psz_name,
506                          p_callbacks[i].callback, NULL );
507 }
508 /*****************************************************************************
509  * All Callbacks:
510  *****************************************************************************/
511 static int StateCallback( vlc_object_t *p_this, char const *psz_cmd,
512                           vlc_value_t oldval, vlc_value_t newval,
513                           void *p_data )
514 {
515     input_thread_t *p_input = (input_thread_t*)p_this;
516     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval); VLC_UNUSED(p_data);
517
518     if( newval.i_int == PLAYING_S || newval.i_int == PAUSE_S )
519     {
520         input_ControlPush( p_input, INPUT_CONTROL_SET_STATE, &newval );
521         return VLC_SUCCESS;
522     }
523
524     return VLC_EGENERIC;
525 }
526
527 static int RateCallback( vlc_object_t *p_this, char const *psz_cmd,
528                          vlc_value_t oldval, vlc_value_t newval, void *p_data )
529 {
530     input_thread_t *p_input = (input_thread_t*)p_this;
531     VLC_UNUSED(oldval); VLC_UNUSED(p_data);
532
533     /* Problem with this way: the "rate" variable is update after the input thread do the change */
534     if( !strcmp( psz_cmd, "rate-slower" ) )
535     {
536         input_ControlPush( p_input, INPUT_CONTROL_SET_RATE_SLOWER, NULL );
537     }
538     else if( !strcmp( psz_cmd, "rate-faster" ) )
539     {
540         input_ControlPush( p_input, INPUT_CONTROL_SET_RATE_FASTER, NULL );
541     }
542     else
543     {
544         input_ControlPush( p_input, INPUT_CONTROL_SET_RATE, &newval );
545     }
546
547     return VLC_SUCCESS;
548 }
549
550 static int PositionCallback( vlc_object_t *p_this, char const *psz_cmd,
551                              vlc_value_t oldval, vlc_value_t newval,
552                              void *p_data )
553 {
554     input_thread_t *p_input = (input_thread_t*)p_this;
555     vlc_value_t val, length;
556     VLC_UNUSED(oldval); VLC_UNUSED(p_data);
557
558     if( !strcmp( psz_cmd, "position-offset" ) )
559     {
560         input_ControlPush( p_input, INPUT_CONTROL_SET_POSITION_OFFSET, &newval );
561
562         val.f_float = var_GetFloat( p_input, "position" ) + newval.f_float;
563         if( val.f_float < 0.0 ) val.f_float = 0.0;
564         if( val.f_float > 1.0 ) val.f_float = 1.0;
565         var_Change( p_input, "position", VLC_VAR_SETVALUE, &val, NULL );
566     }
567     else
568     {
569         val.f_float = newval.f_float;
570         input_ControlPush( p_input, INPUT_CONTROL_SET_POSITION, &newval );
571     }
572
573     /* Update "position" for better intf behavour */
574     var_Get( p_input, "length", &length );
575     if( length.i_time > 0 && val.f_float >= 0.0 && val.f_float <= 1.0 )
576     {
577         val.i_time = length.i_time * val.f_float;
578         var_Change( p_input, "time", VLC_VAR_SETVALUE, &val, NULL );
579     }
580
581     return VLC_SUCCESS;
582 }
583
584 static int TimeCallback( vlc_object_t *p_this, char const *psz_cmd,
585                          vlc_value_t oldval, vlc_value_t newval, void *p_data )
586 {
587     input_thread_t *p_input = (input_thread_t*)p_this;
588     vlc_value_t val, length;
589     VLC_UNUSED(oldval); VLC_UNUSED(p_data);
590
591     if( !strcmp( psz_cmd, "time-offset" ) )
592     {
593         input_ControlPush( p_input, INPUT_CONTROL_SET_TIME_OFFSET, &newval );
594         val.i_time = var_GetTime( p_input, "time" ) + newval.i_time;
595         if( val.i_time < 0 ) val.i_time = 0;
596         /* TODO maybe test against i_length ? */
597         var_Change( p_input, "time", VLC_VAR_SETVALUE, &val, NULL );
598     }
599     else
600     {
601         val.i_time = newval.i_time;
602         input_ControlPush( p_input, INPUT_CONTROL_SET_TIME, &newval );
603     }
604
605     /* Update "position" for better intf behavour */
606     var_Get( p_input, "length", &length );
607     if( length.i_time > 0 && val.i_time >= 0 && val.i_time <= length.i_time )
608     {
609         val.f_float = (double)val.i_time/(double)length.i_time;
610         var_Change( p_input, "position", VLC_VAR_SETVALUE, &val, NULL );
611     }
612
613     return VLC_SUCCESS;
614 }
615
616 static int ProgramCallback( vlc_object_t *p_this, char const *psz_cmd,
617                             vlc_value_t oldval, vlc_value_t newval,
618                             void *p_data )
619 {
620     input_thread_t *p_input = (input_thread_t*)p_this;
621     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval); VLC_UNUSED(p_data);
622
623     input_ControlPush( p_input, INPUT_CONTROL_SET_PROGRAM, &newval );
624
625     return VLC_SUCCESS;
626 }
627
628 static int TitleCallback( vlc_object_t *p_this, char const *psz_cmd,
629                           vlc_value_t oldval, vlc_value_t newval,
630                           void *p_data )
631 {
632     input_thread_t *p_input = (input_thread_t*)p_this;
633     vlc_value_t val, count;
634     VLC_UNUSED(oldval); VLC_UNUSED(p_data);
635
636     if( !strcmp( psz_cmd, "next-title" ) )
637     {
638         input_ControlPush( p_input, INPUT_CONTROL_SET_TITLE_NEXT, NULL );
639
640         val.i_int = var_GetInteger( p_input, "title" ) + 1;
641         var_Change( p_input, "title", VLC_VAR_CHOICESCOUNT, &count, NULL );
642         if( val.i_int < count.i_int )
643             var_Change( p_input, "title", VLC_VAR_SETVALUE, &val, NULL );
644     }
645     else if( !strcmp( psz_cmd, "prev-title" ) )
646     {
647         input_ControlPush( p_input, INPUT_CONTROL_SET_TITLE_PREV, NULL );
648
649         val.i_int = var_GetInteger( p_input, "title" ) - 1;
650         if( val.i_int >= 0 )
651             var_Change( p_input, "title", VLC_VAR_SETVALUE, &val, NULL );
652     }
653     else
654     {
655         input_ControlPush( p_input, INPUT_CONTROL_SET_TITLE, &newval );
656     }
657
658     return VLC_SUCCESS;
659 }
660
661 static int SeekpointCallback( vlc_object_t *p_this, char const *psz_cmd,
662                               vlc_value_t oldval, vlc_value_t newval,
663                               void *p_data )
664 {
665     input_thread_t *p_input = (input_thread_t*)p_this;
666     vlc_value_t val, count;
667     VLC_UNUSED(oldval); VLC_UNUSED(p_data);
668
669     if( !strcmp( psz_cmd, "next-chapter" ) )
670     {
671         input_ControlPush( p_input, INPUT_CONTROL_SET_SEEKPOINT_NEXT, NULL );
672
673         val.i_int = var_GetInteger( p_input, "chapter" ) + 1;
674         var_Change( p_input, "chapter", VLC_VAR_CHOICESCOUNT, &count, NULL );
675         if( val.i_int < count.i_int )
676             var_Change( p_input, "chapter", VLC_VAR_SETVALUE, &val, NULL );
677     }
678     else if( !strcmp( psz_cmd, "prev-chapter" ) )
679     {
680         input_ControlPush( p_input, INPUT_CONTROL_SET_SEEKPOINT_PREV, NULL );
681
682         val.i_int = var_GetInteger( p_input, "chapter" ) - 1;
683         if( val.i_int >= 0 )
684             var_Change( p_input, "chapter", VLC_VAR_SETVALUE, &val, NULL );
685     }
686     else
687     {
688         input_ControlPush( p_input, INPUT_CONTROL_SET_SEEKPOINT, &newval );
689     }
690
691     return VLC_SUCCESS;
692 }
693
694 static int NavigationCallback( vlc_object_t *p_this, char const *psz_cmd,
695                                vlc_value_t oldval, vlc_value_t newval,
696                                void *p_data )
697 {
698     input_thread_t *p_input = (input_thread_t*)p_this;
699     vlc_value_t     val;
700     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval);
701
702     /* Issue a title change */
703     val.i_int = (intptr_t)p_data;
704     input_ControlPush( p_input, INPUT_CONTROL_SET_TITLE, &val );
705
706     var_Change( p_input, "title", VLC_VAR_SETVALUE, &val, NULL );
707
708     /* And a chapter change */
709     input_ControlPush( p_input, INPUT_CONTROL_SET_SEEKPOINT, &newval );
710
711     var_Change( p_input, "chapter", VLC_VAR_SETVALUE, &newval, NULL );
712
713     return VLC_SUCCESS;
714 }
715
716 static int ESCallback( vlc_object_t *p_this, char const *psz_cmd,
717                        vlc_value_t oldval, vlc_value_t newval, void *p_data )
718 {
719     input_thread_t *p_input = (input_thread_t*)p_this;
720     VLC_UNUSED(oldval); VLC_UNUSED(p_data);
721
722     if( newval.i_int < 0 )
723     {
724         vlc_value_t v;
725         /* Hack */
726         if( !strcmp( psz_cmd, "audio-es" ) )
727             v.i_int = -AUDIO_ES;
728         else if( !strcmp( psz_cmd, "video-es" ) )
729             v.i_int = -VIDEO_ES;
730         else if( !strcmp( psz_cmd, "spu-es" ) )
731             v.i_int = -SPU_ES;
732         else
733             v.i_int = 0;
734         if( v.i_int != 0 )
735             input_ControlPush( p_input, INPUT_CONTROL_SET_ES, &v );
736     }
737     else
738     {
739         input_ControlPush( p_input, INPUT_CONTROL_SET_ES, &newval );
740     }
741
742     return VLC_SUCCESS;
743 }
744
745 static int EsDelayCallback ( vlc_object_t *p_this, char const *psz_cmd,
746                              vlc_value_t oldval, vlc_value_t newval, void *p_data )
747 {
748     input_thread_t *p_input = (input_thread_t*)p_this;
749     VLC_UNUSED(oldval); VLC_UNUSED(p_data);
750
751     if( !strcmp( psz_cmd, "audio-delay" ) )
752     {
753         /*Change i_pts_delay to make sure es are decoded in time*/
754         if (newval.i_int < 0 || oldval.i_int < 0 )
755         {
756             p_input->i_pts_delay -= newval.i_int - oldval.i_int;
757         }
758         input_ControlPush( p_input, INPUT_CONTROL_SET_AUDIO_DELAY, &newval );
759     }
760     else if( !strcmp( psz_cmd, "spu-delay" ) )
761         input_ControlPush( p_input, INPUT_CONTROL_SET_SPU_DELAY, &newval );
762     return VLC_SUCCESS;
763 }
764
765 static int BookmarkCallback( vlc_object_t *p_this, char const *psz_cmd,
766                              vlc_value_t oldval, vlc_value_t newval,
767                              void *p_data )
768 {
769     input_thread_t *p_input = (input_thread_t*)p_this;
770     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval); VLC_UNUSED(p_data);
771
772     input_ControlPush( p_input, INPUT_CONTROL_SET_BOOKMARK, &newval );
773
774     return VLC_SUCCESS;
775 }