]> git.sesse.net Git - vlc/blob - src/video_output/vout_intf.c
Lua: remove write-only set_intf()
[vlc] / src / video_output / vout_intf.c
1 /*****************************************************************************
2  * vout_intf.c : video output interface
3  *****************************************************************************
4  * Copyright (C) 2000-2007 VLC authors and VideoLAN
5  *
6  * Authors: Gildas Bazin <gbazin@videolan.org>
7  *
8  * This program is free software; you can redistribute it and/or modify it
9  * under the terms of the GNU Lesser General Public License as published by
10  * the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details.
17  *
18  * You should have received a copy of the GNU Lesser General Public License
19  * along with this program; if not, write to the Free Software Foundation,
20  * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
21  *****************************************************************************/
22
23 /*****************************************************************************
24  * Preamble
25  *****************************************************************************/
26
27 #ifdef HAVE_CONFIG_H
28 # include "config.h"
29 #endif
30
31 #include <vlc_common.h>
32
33 #include <stdio.h>
34 #include <stdlib.h>                                                /* free() */
35 #include <assert.h>
36
37 #include <vlc_block.h>
38 #include <vlc_modules.h>
39
40 #include <vlc_vout.h>
41 #include <vlc_vout_osd.h>
42 #include <vlc_strings.h>
43 #include <vlc_charset.h>
44 #include "vout_internal.h"
45
46 /*****************************************************************************
47  * Local prototypes
48  *****************************************************************************/
49 /* Object variables callbacks */
50 static int CropCallback( vlc_object_t *, char const *,
51                          vlc_value_t, vlc_value_t, void * );
52 static int CropBorderCallback( vlc_object_t *, char const *,
53                                vlc_value_t, vlc_value_t, void * );
54 static int AspectCallback( vlc_object_t *, char const *,
55                            vlc_value_t, vlc_value_t, void * );
56 static int AutoScaleCallback( vlc_object_t *, char const *,
57                               vlc_value_t, vlc_value_t, void * );
58 static int ScaleCallback( vlc_object_t *, char const *,
59                           vlc_value_t, vlc_value_t, void * );
60 static int ZoomCallback( vlc_object_t *, char const *,
61                          vlc_value_t, vlc_value_t, void * );
62 static int OnTopCallback( vlc_object_t *, char const *,
63                           vlc_value_t, vlc_value_t, void * );
64 static int FullscreenCallback( vlc_object_t *, char const *,
65                                vlc_value_t, vlc_value_t, void * );
66 static int SnapshotCallback( vlc_object_t *, char const *,
67                              vlc_value_t, vlc_value_t, void * );
68 static int VideoFilterCallback( vlc_object_t *, char const *,
69                                 vlc_value_t, vlc_value_t, void * );
70 static int SubSourceCallback( vlc_object_t *, char const *,
71                               vlc_value_t, vlc_value_t, void * );
72 static int SubFilterCallback( vlc_object_t *, char const *,
73                               vlc_value_t, vlc_value_t, void * );
74 static int SubMarginCallback( vlc_object_t *, char const *,
75                               vlc_value_t, vlc_value_t, void * );
76
77 /*****************************************************************************
78  * vout_IntfInit: called during the vout creation to initialise misc things.
79  *****************************************************************************/
80 static const struct
81 {
82     double f_value;
83     char psz_label[13];
84 } p_zoom_values[] = {
85     { 0.25, N_("1:4 Quarter") },
86     { 0.5, N_("1:2 Half") },
87     { 1, N_("1:1 Original") },
88     { 2, N_("2:1 Double") },
89 };
90
91 static const struct
92 {
93     char psz_value[8];
94     char psz_label[8];
95 } p_crop_values[] = {
96     { "", N_("Default") },
97     { "16:10", "16:10" },
98     { "16:9", "16:9" },
99     { "185:100", "1.85:1" },
100     { "221:100", "2.21:1" },
101     { "235:100", "2.35:1" },
102     { "239:100", "2.39:1" },
103     { "5:3", "5:3" },
104     { "4:3", "4:3" },
105     { "5:4", "5:4" },
106     { "1:1", "1:1" },
107 };
108
109 static const struct
110 {
111     char psz_value[8];
112     char psz_label[8];
113 } p_aspect_ratio_values[] = {
114     { "", N_("Default") },
115     { "1:1", "1:1" },
116     { "4:3", "4:3" },
117     { "16:9", "16:9" },
118     { "16:10", "16:10" },
119     { "221:100", "2.21:1" },
120     { "235:100", "2.35:1" },
121     { "239:100", "2.39:1" },
122     { "5:4", "5:4" },
123 };
124
125 static void AddCustomRatios( vout_thread_t *p_vout, const char *psz_var,
126                              char *psz_list )
127 {
128     assert( psz_list );
129
130     char *psz_cur = psz_list;
131     char *psz_next;
132     while( psz_cur && *psz_cur )
133     {
134         vlc_value_t val, text;
135         psz_next = strchr( psz_cur, ',' );
136         if( psz_next )
137         {
138             *psz_next = '\0';
139             psz_next++;
140         }
141         val.psz_string = psz_cur;
142         text.psz_string = psz_cur;
143         var_Change( p_vout, psz_var, VLC_VAR_ADDCHOICE, &val, &text);
144         psz_cur = psz_next;
145     }
146 }
147
148 void vout_IntfInit( vout_thread_t *p_vout )
149 {
150     vlc_value_t val, text, old_val;
151     char *psz_buf;
152
153     /* Create a few object variables we'll need later on */
154     var_Create( p_vout, "snapshot-num", VLC_VAR_INTEGER );
155     var_SetInteger( p_vout, "snapshot-num", 1 );
156
157     var_Create( p_vout, "width", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
158     var_Create( p_vout, "height", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
159     var_Create( p_vout, "align", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
160
161     var_Create( p_vout, "video-x", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
162     var_Create( p_vout, "video-y", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
163
164     var_Create( p_vout, "mouse-hide-timeout",
165                 VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
166
167     /* Add variables to manage scaling video */
168     var_Create( p_vout, "autoscale", VLC_VAR_BOOL | VLC_VAR_DOINHERIT
169                 | VLC_VAR_ISCOMMAND );
170     text.psz_string = _("Autoscale video");
171     var_Change( p_vout, "autoscale", VLC_VAR_SETTEXT, &text, NULL );
172     var_AddCallback( p_vout, "autoscale", AutoScaleCallback, NULL );
173
174     var_Create( p_vout, "scale", VLC_VAR_FLOAT | VLC_VAR_DOINHERIT
175                 | VLC_VAR_ISCOMMAND );
176     text.psz_string = _("Scale factor");
177     var_Change( p_vout, "scale", VLC_VAR_SETTEXT, &text, NULL );
178     var_AddCallback( p_vout, "scale", ScaleCallback, NULL );
179
180     /* Zoom object var */
181     var_Create( p_vout, "zoom", VLC_VAR_FLOAT | VLC_VAR_ISCOMMAND |
182                 VLC_VAR_HASCHOICE | VLC_VAR_DOINHERIT );
183
184     text.psz_string = _("Zoom");
185     var_Change( p_vout, "zoom", VLC_VAR_SETTEXT, &text, NULL );
186     old_val.f_float = var_GetFloat( p_vout, "zoom" );
187
188     for( size_t i = 0; i < ARRAY_SIZE(p_zoom_values); i++ )
189     {
190         val.f_float = p_zoom_values[i].f_value;
191         text.psz_string = vlc_gettext( p_zoom_values[i].psz_label );
192         /* FIXME: This DELCHOICE hack corrupts the the "zoom" variable value
193          * for a short time window. Same for "crop" and "aspect-ratio". */
194         if( old_val.f_float == val.f_float )
195             var_Change( p_vout, "zoom", VLC_VAR_DELCHOICE, &old_val, NULL );
196         var_Change( p_vout, "zoom", VLC_VAR_ADDCHOICE, &val, &text );
197         if( old_val.f_float == val.f_float )
198             var_Change( p_vout, "zoom", VLC_VAR_SETVALUE, &old_val, NULL );
199     }
200
201     var_AddCallback( p_vout, "zoom", ZoomCallback, NULL );
202
203     /* Crop offset vars */
204     var_Create( p_vout, "crop-left", VLC_VAR_INTEGER | VLC_VAR_ISCOMMAND );
205     var_Create( p_vout, "crop-top", VLC_VAR_INTEGER | VLC_VAR_ISCOMMAND );
206     var_Create( p_vout, "crop-right", VLC_VAR_INTEGER | VLC_VAR_ISCOMMAND );
207     var_Create( p_vout, "crop-bottom", VLC_VAR_INTEGER | VLC_VAR_ISCOMMAND );
208
209     var_AddCallback( p_vout, "crop-left", CropBorderCallback, NULL );
210     var_AddCallback( p_vout, "crop-top", CropBorderCallback, NULL );
211     var_AddCallback( p_vout, "crop-right", CropBorderCallback, NULL );
212     var_AddCallback( p_vout, "crop-bottom", CropBorderCallback, NULL );
213
214     /* Crop object var */
215     var_Create( p_vout, "crop", VLC_VAR_STRING | VLC_VAR_ISCOMMAND |
216                 VLC_VAR_HASCHOICE | VLC_VAR_DOINHERIT );
217
218     text.psz_string = _("Crop");
219     var_Change( p_vout, "crop", VLC_VAR_SETTEXT, &text, NULL );
220
221     val.psz_string = (char*)"";
222     var_Change( p_vout, "crop", VLC_VAR_DELCHOICE, &val, 0 );
223
224     for( size_t i = 0; i < ARRAY_SIZE(p_crop_values); i++ )
225     {
226         val.psz_string = (char*)p_crop_values[i].psz_value;
227         text.psz_string = _( p_crop_values[i].psz_label );
228         var_Change( p_vout, "crop", VLC_VAR_ADDCHOICE, &val, &text );
229     }
230
231     /* Add custom crop ratios */
232     psz_buf = var_CreateGetNonEmptyString( p_vout, "custom-crop-ratios" );
233     if( psz_buf )
234     {
235         AddCustomRatios( p_vout, "crop", psz_buf );
236         free( psz_buf );
237     }
238
239     var_AddCallback( p_vout, "crop", CropCallback, NULL );
240
241     /* Monitor pixel aspect-ratio */
242     var_Create( p_vout, "monitor-par", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
243
244     /* Aspect-ratio object var */
245     var_Create( p_vout, "aspect-ratio", VLC_VAR_STRING | VLC_VAR_ISCOMMAND |
246                 VLC_VAR_HASCHOICE | VLC_VAR_DOINHERIT );
247
248     text.psz_string = _("Aspect ratio");
249     var_Change( p_vout, "aspect-ratio", VLC_VAR_SETTEXT, &text, NULL );
250
251     val.psz_string = (char*)"";
252     var_Change( p_vout, "aspect-ratio", VLC_VAR_DELCHOICE, &val, 0 );
253
254     for( size_t i = 0; i < ARRAY_SIZE(p_aspect_ratio_values); i++ )
255     {
256         val.psz_string = (char*)p_aspect_ratio_values[i].psz_value;
257         text.psz_string = _( p_aspect_ratio_values[i].psz_label );
258         var_Change( p_vout, "aspect-ratio", VLC_VAR_ADDCHOICE, &val, &text );
259     }
260
261     /* Add custom aspect ratios */
262     psz_buf = var_CreateGetNonEmptyString( p_vout, "custom-aspect-ratios" );
263     if( psz_buf )
264     {
265         AddCustomRatios( p_vout, "aspect-ratio", psz_buf );
266         free( psz_buf );
267     }
268
269     var_AddCallback( p_vout, "aspect-ratio", AspectCallback, NULL );
270
271     /* Add a variable to indicate if the window should be on top of others */
272     var_Create( p_vout, "video-on-top", VLC_VAR_BOOL | VLC_VAR_DOINHERIT
273                 | VLC_VAR_ISCOMMAND );
274     text.psz_string = _("Always on top");
275     var_Change( p_vout, "video-on-top", VLC_VAR_SETTEXT, &text, NULL );
276     var_AddCallback( p_vout, "video-on-top", OnTopCallback, NULL );
277
278     /* Add a variable to indicate whether we want window decoration or not */
279     var_Create( p_vout, "video-deco", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
280
281     /* Add a fullscreen variable */
282     var_Create( p_vout, "fullscreen",
283                 VLC_VAR_BOOL | VLC_VAR_DOINHERIT | VLC_VAR_ISCOMMAND );
284     text.psz_string = _("Fullscreen");
285     var_Change( p_vout, "fullscreen", VLC_VAR_SETTEXT, &text, NULL );
286     var_AddCallback( p_vout, "fullscreen", FullscreenCallback, NULL );
287
288     /* Add a snapshot variable */
289     var_Create( p_vout, "video-snapshot", VLC_VAR_VOID | VLC_VAR_ISCOMMAND );
290     text.psz_string = _("Snapshot");
291     var_Change( p_vout, "video-snapshot", VLC_VAR_SETTEXT, &text, NULL );
292     var_AddCallback( p_vout, "video-snapshot", SnapshotCallback, NULL );
293
294     /* Add a video-filter variable */
295     var_Create( p_vout, "video-filter",
296                 VLC_VAR_STRING | VLC_VAR_DOINHERIT | VLC_VAR_ISCOMMAND );
297     var_AddCallback( p_vout, "video-filter", VideoFilterCallback, NULL );
298
299     /* Add a sub-source variable */
300     var_Create( p_vout, "sub-source",
301                 VLC_VAR_STRING | VLC_VAR_DOINHERIT | VLC_VAR_ISCOMMAND );
302     var_AddCallback( p_vout, "sub-source", SubSourceCallback, NULL );
303
304     /* Add a sub-filter variable */
305     var_Create( p_vout, "sub-filter",
306                 VLC_VAR_STRING | VLC_VAR_DOINHERIT | VLC_VAR_ISCOMMAND );
307     var_AddCallback( p_vout, "sub-filter", SubFilterCallback, NULL );
308
309     /* Add sub-margin variable */
310     var_Create( p_vout, "sub-margin",
311                 VLC_VAR_INTEGER | VLC_VAR_DOINHERIT | VLC_VAR_ISCOMMAND );
312     var_AddCallback( p_vout, "sub-margin", SubMarginCallback, NULL );
313
314     /* Mouse coordinates */
315     var_Create( p_vout, "mouse-button-down", VLC_VAR_INTEGER );
316     var_Create( p_vout, "mouse-moved", VLC_VAR_COORDS );
317     var_Create( p_vout, "mouse-clicked", VLC_VAR_COORDS );
318     var_Create( p_vout, "mouse-object", VLC_VAR_BOOL );
319
320     vout_IntfReinit( p_vout );
321 }
322
323 void vout_IntfReinit( vout_thread_t *p_vout )
324 {
325     var_TriggerCallback( p_vout, "zoom" );
326     var_TriggerCallback( p_vout, "crop" );
327     var_TriggerCallback( p_vout, "aspect-ratio" );
328
329     var_TriggerCallback( p_vout, "video-on-top" );
330
331     var_TriggerCallback( p_vout, "video-filter" );
332     var_TriggerCallback( p_vout, "sub-source" );
333     var_TriggerCallback( p_vout, "sub-filter" );
334     var_TriggerCallback( p_vout, "sub-margin" );
335 }
336
337 /*****************************************************************************
338  * vout_Snapshot: generates a snapshot.
339  *****************************************************************************/
340 /**
341  * This function will inject a subpicture into the vout with the provided
342  * picture
343  */
344 static int VoutSnapshotPip( vout_thread_t *p_vout, picture_t *p_pic )
345 {
346     subpicture_t *p_subpic = subpicture_NewFromPicture( VLC_OBJECT(p_vout),
347                                                         p_pic, VLC_CODEC_YUVA );
348     if( !p_subpic )
349         return VLC_EGENERIC;
350
351     /* FIXME SPU_DEFAULT_CHANNEL is not good (used by the text) but
352      * hardcoded 0 doesn't seem right */
353     p_subpic->i_channel = 0;
354     p_subpic->i_start = mdate();
355     p_subpic->i_stop  = p_subpic->i_start + 4000000;
356     p_subpic->b_ephemer = true;
357     p_subpic->b_fade = true;
358
359     /* Reduce the picture to 1/4^2 of the screen */
360     p_subpic->i_original_picture_width  *= 4;
361     p_subpic->i_original_picture_height *= 4;
362
363     vout_PutSubpicture( p_vout, p_subpic );
364     return VLC_SUCCESS;
365 }
366
367 /**
368  * This function will display the name and a PIP of the provided snapshot
369  */
370 static void VoutOsdSnapshot( vout_thread_t *p_vout, picture_t *p_pic, const char *psz_filename )
371 {
372     msg_Dbg( p_vout, "snapshot taken (%s)", psz_filename );
373     vout_OSDMessage( p_vout, SPU_DEFAULT_CHANNEL, "%s", psz_filename );
374
375     if( var_InheritBool( p_vout, "snapshot-preview" ) )
376     {
377         if( VoutSnapshotPip( p_vout, p_pic ) )
378             msg_Warn( p_vout, "Failed to display snapshot" );
379     }
380 }
381
382 /**
383  * This function will handle a snapshot request
384  */
385 static void VoutSaveSnapshot( vout_thread_t *p_vout )
386 {
387     char *psz_path = var_InheritString( p_vout, "snapshot-path" );
388     char *psz_format = var_InheritString( p_vout, "snapshot-format" );
389     char *psz_prefix = var_InheritString( p_vout, "snapshot-prefix" );
390
391     /* */
392     picture_t *p_picture;
393     block_t *p_image;
394     video_format_t fmt;
395
396     /* 500ms timeout
397      * XXX it will cause trouble with low fps video (< 2fps) */
398     if( vout_GetSnapshot( p_vout, &p_image, &p_picture, &fmt, psz_format, 500*1000 ) )
399     {
400         p_picture = NULL;
401         p_image = NULL;
402         goto exit;
403     }
404
405     if( !psz_path )
406     {
407         psz_path = vout_snapshot_GetDirectory();
408         if( !psz_path )
409         {
410             msg_Err( p_vout, "no path specified for snapshots" );
411             goto exit;
412         }
413     }
414
415     vout_snapshot_save_cfg_t cfg;
416     memset( &cfg, 0, sizeof(cfg) );
417     cfg.is_sequential = var_InheritBool( p_vout, "snapshot-sequential" );
418     cfg.sequence = var_GetInteger( p_vout, "snapshot-num" );
419     cfg.path = psz_path;
420     cfg.format = psz_format;
421     cfg.prefix_fmt = psz_prefix;
422
423     char *psz_filename;
424     int  i_sequence;
425     if (vout_snapshot_SaveImage( &psz_filename, &i_sequence,
426                                  p_image, VLC_OBJECT(p_vout), &cfg ) )
427         goto exit;
428     if( cfg.is_sequential )
429         var_SetInteger( p_vout, "snapshot-num", i_sequence + 1 );
430
431     VoutOsdSnapshot( p_vout, p_picture, psz_filename );
432
433     /* signal creation of a new snapshot file */
434     var_SetString( p_vout->p_libvlc, "snapshot-file", psz_filename );
435
436     free( psz_filename );
437
438 exit:
439     if( p_image )
440         block_Release( p_image );
441     if( p_picture )
442         picture_Release( p_picture );
443     free( psz_prefix );
444     free( psz_format );
445     free( psz_path );
446 }
447
448 /*****************************************************************************
449  * Handle filters
450  *****************************************************************************/
451
452 void vout_EnableFilter( vout_thread_t *p_vout, const char *psz_name,
453                         bool b_add, bool b_setconfig )
454 {
455     char *psz_parser;
456     char *psz_string;
457     const char *psz_filter_type;
458
459     module_t *p_obj = module_find( psz_name );
460     if( !p_obj )
461     {
462         msg_Err( p_vout, "Unable to find filter module \"%s\".", psz_name );
463         return;
464     }
465
466     if( module_provides( p_obj, "video filter2" ) )
467     {
468         psz_filter_type = "video-filter";
469     }
470     else if( module_provides( p_obj, "sub source" ) )
471     {
472         psz_filter_type = "sub-source";
473     }
474     else if( module_provides( p_obj, "sub filter" ) )
475     {
476         psz_filter_type = "sub-filter";
477     }
478     else
479     {
480         msg_Err( p_vout, "Unknown video filter type." );
481         return;
482     }
483
484     psz_string = var_GetString( p_vout, psz_filter_type );
485
486     /* Todo : Use some generic chain manipulation functions */
487     if( !psz_string ) psz_string = strdup("");
488
489     psz_parser = strstr( psz_string, psz_name );
490     if( b_add )
491     {
492         if( !psz_parser )
493         {
494             psz_parser = psz_string;
495             if( asprintf( &psz_string, (*psz_string) ? "%s:%s" : "%s%s",
496                           psz_string, psz_name ) == -1 )
497             {
498                 free( psz_parser );
499                 return;
500             }
501             free( psz_parser );
502         }
503         else
504             return;
505     }
506     else
507     {
508         if( psz_parser )
509         {
510             memmove( psz_parser, psz_parser + strlen(psz_name) +
511                             (*(psz_parser + strlen(psz_name)) == ':' ? 1 : 0 ),
512                             strlen(psz_parser + strlen(psz_name)) + 1 );
513
514             /* Remove trailing : : */
515             if( *(psz_string+strlen(psz_string ) -1 ) == ':' )
516             {
517                 *(psz_string+strlen(psz_string ) -1 ) = '\0';
518             }
519          }
520          else
521          {
522              free( psz_string );
523              return;
524          }
525     }
526
527     if( b_setconfig )
528     {
529         config_PutPsz( p_vout, psz_filter_type, psz_string );
530     }
531
532     var_SetString( p_vout, psz_filter_type, psz_string );
533
534     free( psz_string );
535 }
536
537 /*****************************************************************************
538  * Object variables callbacks
539  *****************************************************************************/
540 static int CropCallback( vlc_object_t *object, char const *cmd,
541                          vlc_value_t oldval, vlc_value_t newval, void *data )
542 {
543     vout_thread_t *vout = (vout_thread_t *)object;
544     VLC_UNUSED(cmd); VLC_UNUSED(oldval); VLC_UNUSED(data);
545     unsigned num, den;
546     unsigned y, x;
547     unsigned width, height;
548     unsigned left, top, right, bottom;
549
550     if (sscanf(newval.psz_string, "%u:%u", &num, &den) == 2) {
551         vout_ControlChangeCropRatio(vout, num, den);
552     } else if (sscanf(newval.psz_string, "%ux%u+%u+%u",
553                       &width, &height, &x, &y) == 4) {
554         vout_ControlChangeCropWindow(vout, x, y, width, height);
555     } else if (sscanf(newval.psz_string, "%u+%u+%u+%u",
556                     &left, &top, &right, &bottom) == 4) {
557         vout_ControlChangeCropBorder(vout, left, top, right, bottom);
558     } else if (*newval.psz_string == '\0') {
559         vout_ControlChangeCropRatio(vout, 0, 0);
560     } else {
561         msg_Err(object, "Unknown crop format (%s)", newval.psz_string);
562     }
563     return VLC_SUCCESS;
564 }
565
566 static int CropBorderCallback(vlc_object_t *object, char const *cmd,
567                               vlc_value_t oldval, vlc_value_t newval, void *data)
568 {
569     vout_thread_t *vout = (vout_thread_t *)object;
570     VLC_UNUSED(cmd); VLC_UNUSED(oldval); VLC_UNUSED(data); VLC_UNUSED(newval);
571
572     vout_ControlChangeCropBorder(vout,
573                                  var_GetInteger(object, "crop-left"),
574                                  var_GetInteger(object, "crop-top"),
575                                  var_GetInteger(object, "crop-right"),
576                                  var_GetInteger(object, "crop-bottom"));
577     return VLC_SUCCESS;
578 }
579
580 static int AspectCallback( vlc_object_t *object, char const *cmd,
581                          vlc_value_t oldval, vlc_value_t newval, void *data )
582 {
583     vout_thread_t *vout = (vout_thread_t *)object;
584     VLC_UNUSED(cmd); VLC_UNUSED(oldval); VLC_UNUSED(data);
585     unsigned num, den;
586
587     if (sscanf(newval.psz_string, "%u:%u", &num, &den) == 2 &&
588         (num > 0) == (den > 0))
589         vout_ControlChangeSampleAspectRatio(vout, num, den);
590     else if (*newval.psz_string == '\0')
591         vout_ControlChangeSampleAspectRatio(vout, 0, 0);
592     return VLC_SUCCESS;
593 }
594
595 static int AutoScaleCallback( vlc_object_t *obj, char const *name,
596                               vlc_value_t prev, vlc_value_t cur, void *data )
597 {
598     vout_thread_t *p_vout = (vout_thread_t *)obj;
599
600     (void) name; (void) prev; (void) data;
601     vout_ControlChangeDisplayFilled( p_vout, cur.b_bool );
602     return VLC_SUCCESS;
603 }
604
605 static int ScaleCallback( vlc_object_t *obj, char const *name,
606                           vlc_value_t prev, vlc_value_t cur, void *data )
607 {
608     vout_thread_t *p_vout = (vout_thread_t *)obj;
609
610     (void) name; (void) prev; (void) data;
611     vout_ControlChangeZoom( p_vout, 1000 * cur.f_float, 1000 );
612     return VLC_SUCCESS;
613 }
614
615 static int ZoomCallback( vlc_object_t *obj, char const *name,
616                          vlc_value_t prev, vlc_value_t cur, void *data )
617 {
618     (void) name; (void) prev; (void) data;
619     return var_SetFloat( obj, "scale", cur.f_float );
620 }
621
622 static int OnTopCallback( vlc_object_t *p_this, char const *psz_cmd,
623                          vlc_value_t oldval, vlc_value_t newval, void *p_data )
624 {
625     vout_thread_t *p_vout = (vout_thread_t *)p_this;
626     (void)psz_cmd; (void)oldval; (void)p_data;
627
628     vout_ControlChangeOnTop( p_vout, newval.b_bool );
629     return VLC_SUCCESS;
630 }
631
632 static int FullscreenCallback( vlc_object_t *p_this, char const *psz_cmd,
633                        vlc_value_t oldval, vlc_value_t newval, void *p_data )
634 {
635     vout_thread_t *p_vout = (vout_thread_t *)p_this;
636     (void)psz_cmd; (void)p_data;
637
638     if( oldval.b_bool != newval.b_bool )
639         vout_ControlChangeFullscreen( p_vout, newval.b_bool );
640     return VLC_SUCCESS;
641 }
642
643 static int SnapshotCallback( vlc_object_t *p_this, char const *psz_cmd,
644                        vlc_value_t oldval, vlc_value_t newval, void *p_data )
645 {
646     vout_thread_t *p_vout = (vout_thread_t *)p_this;
647     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval);
648     VLC_UNUSED(newval); VLC_UNUSED(p_data);
649
650     VoutSaveSnapshot( p_vout );
651     return VLC_SUCCESS;
652 }
653
654 static int VideoFilterCallback( vlc_object_t *p_this, char const *psz_cmd,
655                                 vlc_value_t oldval, vlc_value_t newval, void *p_data)
656 {
657     vout_thread_t *p_vout = (vout_thread_t *)p_this;
658     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval); VLC_UNUSED(p_data);
659
660     vout_ControlChangeFilters( p_vout, newval.psz_string );
661     return VLC_SUCCESS;
662 }
663
664 static int SubSourceCallback( vlc_object_t *p_this, char const *psz_cmd,
665                               vlc_value_t oldval, vlc_value_t newval, void *p_data)
666 {
667     vout_thread_t *p_vout = (vout_thread_t *)p_this;
668     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval); VLC_UNUSED(p_data);
669
670     vout_ControlChangeSubSources( p_vout, newval.psz_string );
671     return VLC_SUCCESS;
672 }
673
674 static int SubFilterCallback( vlc_object_t *p_this, char const *psz_cmd,
675                               vlc_value_t oldval, vlc_value_t newval, void *p_data)
676 {
677     vout_thread_t *p_vout = (vout_thread_t *)p_this;
678     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval); VLC_UNUSED(p_data);
679
680     vout_ControlChangeSubFilters( p_vout, newval.psz_string );
681     return VLC_SUCCESS;
682 }
683
684 static int SubMarginCallback( vlc_object_t *p_this, char const *psz_cmd,
685                               vlc_value_t oldval, vlc_value_t newval, void *p_data)
686 {
687     vout_thread_t *p_vout = (vout_thread_t *)p_this;
688     VLC_UNUSED(psz_cmd); VLC_UNUSED(oldval); VLC_UNUSED(p_data);
689
690     vout_ControlChangeSubMargin( p_vout, newval.i_int );
691     return VLC_SUCCESS;
692 }
693