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