]> git.sesse.net Git - vlc/blob - modules/gui/macosx/sfilters.m
For consistency, remove references to vlc from libvlc
[vlc] / modules / gui / macosx / sfilters.m
1 /*****************************************************************************
2  * sfilter.m: MacOS X Subpicture filters dialogue
3  *****************************************************************************
4  * Copyright (C) 2005-2006 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Felix Kühne <fkuehne@users.sf.net>
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 /*****************************************************************************
26  * Note: 
27  * the code used to bind with VLC's core is partially based upon the 
28  * RC-interface, written by Antoine Cellerier and Mark F. Moriarty  
29  * (members of the VideoLAN team) 
30  *****************************************************************************/
31
32 #import "sfilters.h"
33 #import "intf.h"
34 #import <vlc/vout.h>
35
36 /* TODO:
37     - check for memory leaks
38     - save the preferences, if requested
39 */
40
41 @implementation VLCsFilters
42
43 static VLCsFilters *_o_sharedInstance = nil;
44
45 + (VLCsFilters *)sharedInstance
46 {
47     return _o_sharedInstance ? _o_sharedInstance : [[self alloc] init];
48 }
49
50 - (id)init
51 {
52     if (_o_sharedInstance) {
53         [self dealloc];
54     } else {
55         _o_sharedInstance = [super init];
56     }
57
58     return _o_sharedInstance;
59 }
60
61 - (void)dealloc
62 {
63     [o_colors release];
64
65     [super dealloc];
66 }
67
68 - (void)initStrings
69 {
70     [o_sfilter_win setTitle: _NS("Subpicture Filters")];
71     [[o_sfilter_tabView tabViewItemAtIndex: 0] setLabel: _NS("Logo")];
72     [[o_sfilter_tabView tabViewItemAtIndex: 1] setLabel: _NS("Time")];
73     [[o_sfilter_tabView tabViewItemAtIndex: 2] setLabel: _NS("Marquee")];
74     [o_sfilter_saveSettings_ckb setTitle: _NS("Save settings")];
75     [o_logo_image_btn setTitle: _NS("Browse...")];
76     [o_logo_enabled_ckb setTitle: _NS("Enabled")];
77     [o_logo_image_lbl setStringValue: _NS("Image:")];
78     [o_logo_pos_lbl setStringValue: _NS("Position:")];
79     [o_logo_opaque_lbl setStringValue: _NS("Opaqueness")];
80     [o_time_enabled_ckb setTitle: _NS("Enabled")];
81     [o_time_stamp_lbl setStringValue: _NS("Timestamp:")];
82     [o_time_size_lbl setStringValue: _NS("Size:")];
83     [o_time_color_lbl setStringValue: _NS("Color:")];
84     [o_time_opaque_lbl setStringValue: _NS("Opaqueness:")];
85     [o_time_pos_lbl setStringValue: _NS("Position:")];
86     [o_time_size_inPx_lbl setStringValue: _NS("(in pixels)")];
87     [o_marq_enabled_ckb setTitle: _NS("Enabled")];
88     [o_marq_color_lbl setStringValue: _NS("Color:")];
89     [o_marq_marq_lbl setStringValue: _NS("Marquee:")];
90     [o_marq_opaque_lbl setStringValue: _NS("Opaqueness")];
91     [o_marq_tmOut_lbl setStringValue: _NS("Timeout:")];
92     [o_marq_tmOut_ms_lbl setStringValue: _NS("ms")];
93     [o_marq_pos_lbl setStringValue: _NS("Position:")];
94     [o_marq_size_lbl setStringValue: _NS("Size:")];
95     [o_time_color_lbl setStringValue: _NS("(in pixels)")];
96 }
97
98 - (void)awakeFromNib
99 {
100     /* colors as implemented at the beginning of marq.c and time.c
101      * feel free to add more colors, but remember to add them to these files 
102      * as well to keep a certain level of consistency across the interfaces */
103     NSArray * o_default;
104     NSArray * o_black;
105     NSArray * o_gray;
106     NSArray * o_silver;
107     NSArray * o_white;
108     NSArray * o_maroon;
109     NSArray * o_red;
110     NSArray * o_fuchsia;
111     NSArray * o_yellow;
112     NSArray * o_olive;
113     NSArray * o_green;
114     NSArray * o_teal;
115     NSArray * o_lime;
116     NSArray * o_purple;
117     NSArray * o_navy;
118     NSArray * o_blue;
119     NSArray * o_aqua;
120     o_default = [NSArray arrayWithObjects: _NS("Default"), @"-1", nil];
121     o_black = [NSArray arrayWithObjects: _NS("Black"), @"0x000000", nil];
122     o_gray = [NSArray arrayWithObjects: _NS("Gray"), @"0x808080", nil];
123     o_silver = [NSArray arrayWithObjects: _NS("Silver"), @"0xC0C0C0", nil];
124     o_white = [NSArray arrayWithObjects: _NS("White"), @"0xFFFFFF", nil];
125     o_maroon = [NSArray arrayWithObjects: _NS("Maroon"), @"0x800000", nil];
126     o_red = [NSArray arrayWithObjects: _NS("Red"), @"0xFF0000", nil];
127     o_fuchsia = [NSArray arrayWithObjects: _NS("Fuchsia"), @"0xFF00FF", nil];
128     o_yellow = [NSArray arrayWithObjects: _NS("Yellow"), @"0xFFFF00", nil];
129     o_olive = [NSArray arrayWithObjects: _NS("Olive"), @"0x808000", nil];
130     o_green = [NSArray arrayWithObjects: _NS("Green"), @"0x008000", nil];
131     o_teal = [NSArray arrayWithObjects: _NS("Teal"), @"0x008080", nil];
132     o_lime = [NSArray arrayWithObjects: _NS("Lime"), @"0x00FF00", nil];
133     o_purple = [NSArray arrayWithObjects: _NS("Purple"), @"0x800080", nil];
134     o_navy = [NSArray arrayWithObjects: _NS("Navy"), @"0x000080", nil];
135     o_blue = [NSArray arrayWithObjects: _NS("Blue"), @"0x0000FF", nil];
136     o_aqua = [NSArray arrayWithObjects: _NS("Aqua"), @"0x00FFFF", nil];
137     o_colors = [[NSArray alloc] initWithObjects: o_default, o_black, o_gray,
138         o_silver, o_white, o_maroon, o_red, o_fuchsia, o_yellow, o_olive,
139         o_green, o_teal, o_lime, o_purple, o_navy, o_blue, o_aqua, nil];
140
141     unsigned int x = 0;
142     [o_marq_color_pop removeAllItems];
143     [o_time_color_pop removeAllItems];
144     
145     /* we are adding tags to the items, so we can easily identify them even if 
146      * the menu was sorted */
147     while (x != [o_colors count])
148     {
149         [o_marq_color_pop addItemWithTitle: [[o_colors objectAtIndex:x]
150             objectAtIndex:0]];
151         [[o_marq_color_pop lastItem] setTag: x];
152         
153         [o_time_color_pop addItemWithTitle: [[o_colors objectAtIndex:x]
154             objectAtIndex:0]];
155         [[o_time_color_pop lastItem] setTag: x];
156         
157         x = (x + 1);
158     }
159
160     [o_marq_color_pop selectItemAtIndex:0];
161     [o_time_color_pop selectItemAtIndex:0];
162
163     /* define the relative positions and copy them to the menues
164      * we can destroy the array afterwards, because we are saving the ints 
165      * as tags to the menu-items */
166     /*NSArray * o_cnt_cnt;
167     NSArray * o_lft_cnt;
168     NSArray * o_rht_cnt;
169     NSArray * o_cnt_top;
170     NSArray * o_lft_top;
171     NSArray * o_rht_top;
172     NSArray * o_cnt_btm;
173     NSArray * o_lft_btm;
174     NSArray * o_rht_btm;
175     NSArray * o_positions;
176     o_cnt_cnt = [NSArray arrayWithObjects: _NS("Center-Center"), @"0", nil];
177     o_lft_cnt = [NSArray arrayWithObjects: _NS("Left-Center"), @"1", nil];
178     o_rht_cnt = [NSArray arrayWithObjects: _NS("Right-Center"), @"2", nil];
179     o_cnt_top = [NSArray arrayWithObjects: _NS("Center-Top"), @"4", nil];
180     o_lft_top = [NSArray arrayWithObjects: _NS("Left-Top"), @"5", nil];
181     o_rht_top = [NSArray arrayWithObjects: _NS("Right-Top"), @"6", nil];
182     o_cnt_btm = [NSArray arrayWithObjects: _NS("Center-Bottom"), @"8", nil];
183     o_lft_btm = [NSArray arrayWithObjects: _NS("Left-Bottom"), @"9", nil];
184     o_rht_btm = [NSArray arrayWithObjects: _NS("Right-Bottom"), @"10", nil];
185     o_positions = [[NSArray alloc] initWithObjects: o_cnt_cnt, o_lft_cnt,
186         o_rht_cnt, o_cnt_top, o_lft_top, o_rht_top, o_cnt_btm, o_lft_btm,
187         o_rht_btm, nil];
188         
189     x = 0;
190     [o_time_pos_rel_pop removeAllItems];
191     [o_marq_pos_rel_pop removeAllItems];
192     [o_logo_pos_rel_pop removeAllItems];
193     
194     * we are adding a tag here, so we can easily select an item later on *
195     while ( x != [o_positions count] )
196     {
197         [o_time_pos_rel_pop addItemWithTitle: [[o_positions objectAtIndex:x]
198             objectAtIndex:0]];
199         [[o_time_pos_rel_pop lastItem] setTag: [[[o_positions objectAtIndex:x]
200             objectAtIndex:1] intValue]];
201         [o_marq_pos_rel_pop addItemWithTitle: [[o_positions objectAtIndex:x]
202             objectAtIndex:0]];
203         [[o_marq_pos_rel_pop lastItem] setTag: [[[o_positions objectAtIndex:x]
204             objectAtIndex:1] intValue]];
205         [o_logo_pos_rel_pop addItemWithTitle: [[o_positions objectAtIndex:x]
206             objectAtIndex:0]];
207         [[o_logo_pos_rel_pop lastItem] setTag: [[[o_positions objectAtIndex:x]
208             objectAtIndex:1] intValue]];
209
210         x = (x + 1);
211     }
212     [o_positions release];*/
213
214     NSArray * o_sizes;
215     o_sizes = [[NSArray alloc] initWithObjects: @"6", @"8", @"10", @"11", @"12",\
216         @"14", @"13", @"16", @"18", @"24", @"36", @"48", @"64", @"72", @"96",
217         @"144", @"288", nil];
218     [o_marq_size_pop removeAllItems];
219     [o_marq_size_pop addItemsWithTitles: o_sizes];
220     [o_time_size_pop removeAllItems];
221     [o_time_size_pop addItemsWithTitles: o_sizes];
222     [o_sizes release];
223 }
224
225 - (void)showAsPanel
226 {
227     /* called from intf.m */
228     [o_sfilter_win displayIfNeeded];
229     [o_sfilter_win makeKeyAndOrderFront:nil];
230
231     intf_thread_t * p_intf = VLCIntf;
232
233     /* retrieve the marquee settings */
234     int x = 0;
235     int tempInt = config_GetInt( p_intf, "marq-color" );
236     while( strtol([[[o_colors objectAtIndex:x] objectAtIndex:1] UTF8String],
237         NULL, 0) != tempInt )
238     {
239         x = (x + 1);
240         
241         if( x >= [o_marq_color_pop numberOfItems] )
242         {
243             x = 0;
244             return;
245         }
246     }
247     [o_marq_color_pop selectItemAtIndex: x];
248     [o_marq_marq_fld setStringValue: [NSString stringWithUTF8String:
249         config_GetPsz( p_intf, "marq-marquee" )]];
250     [o_marq_opaque_sld setIntValue: config_GetInt( p_intf, "marq-opacity")];
251     [o_marq_pos_radio selectCellWithTag: config_GetInt( p_intf, "marq-position" )];
252     /* FIXME: the following line doesn't work with "-1", which is the default
253      * value */
254     [o_marq_size_pop selectItemWithTitle: 
255         [[NSNumber numberWithInt: config_GetInt( p_intf, "marq-size" )]
256             stringValue]];
257     [o_marq_size_pop selectItemAtIndex: x];
258     [o_marq_tmOut_fld setStringValue: [[NSNumber numberWithInt:
259         config_GetInt( p_intf, "marq-timeout" )] stringValue]];
260     
261     /* retrieve the time settings */
262     x = 0;
263     tempInt = config_GetInt( p_intf, "time-color" );
264     while( strtol([[[o_colors objectAtIndex:x] objectAtIndex:1] UTF8String],
265         NULL, 0) != tempInt )
266     {
267         x = (x + 1);
268         
269         if( x >= [o_time_color_pop numberOfItems] )
270         {
271             x = 0;
272             return;
273         }
274     }
275     [o_time_color_pop selectItemAtIndex: x];
276     [o_time_stamp_fld setStringValue: [NSString stringWithUTF8String:
277         config_GetPsz( p_intf, "time-format" )]];
278     [o_time_opaque_sld setIntValue: config_GetInt( p_intf, "time-opacity")];
279     /* FIXME: the following line doesn't work with "-1", which is the default
280      * value */
281     [o_time_size_pop selectItemWithTitle: 
282         [[NSNumber numberWithInt: config_GetInt( p_intf, "time-size" )]
283             stringValue]];
284     [o_time_pos_radio selectCellWithTag: config_GetInt( p_intf, "time-position" )];    
285
286     /* retrieve the logo settings */
287     [o_logo_opaque_sld setIntValue: config_GetInt( p_intf, "logo-transparency")];
288     /* in case that no path has been saved yet */
289     if( config_GetPsz( p_intf, "logo-file" ) )
290         [o_logo_image_fld setStringValue: [NSString stringWithUTF8String:
291                                         config_GetPsz( p_intf, "logo-file" )]];
292     else
293         [o_logo_image_fld setStringValue: @""];
294     [o_logo_pos_radio selectCellWithTag: config_GetInt( p_intf, "logo-position" )];
295     
296     /* enable the requested filters */
297     char * psz_subfilters;
298     psz_subfilters = config_GetPsz( p_intf, "sub-filter" );
299     if( psz_subfilters )
300     {
301         [o_marq_enabled_ckb setState: (bool)strstr( psz_subfilters, "marq")];
302         [o_logo_enabled_ckb setState: (bool)strstr( psz_subfilters, "logo")];
303         [o_time_enabled_ckb setState: (bool)strstr( psz_subfilters, "time")];
304     }
305         [self enableMarq];
306         [self enableLogo];
307         [self enableTime];
308 }
309
310 - (IBAction)logo_selectFile:(id)sender
311 {
312     NSOpenPanel * openPanel = [NSOpenPanel openPanel];
313     SEL sel = @selector(logo_getFile:returnCode:contextInfo:);
314     [openPanel beginSheetForDirectory:nil file:nil types: [NSArray
315         arrayWithObjects: @"png", @"PNG", @"'PNGf'", nil] modalForWindow:
316         o_sfilter_win modalDelegate:self didEndSelector:sel contextInfo:nil];
317 }
318
319 - (void)logo_getFile: (NSOpenPanel *)sheet returnCode:
320     (int)returnCode contextInfo: (void *)contextInfo
321 {
322     if (returnCode == NSOKButton)
323     {
324         [o_logo_image_fld setStringValue: [sheet filename]];
325     }
326 }
327
328 - (IBAction)propertyChanged:(id)sender
329 {
330     intf_thread_t * p_intf = VLCIntf;
331     input_thread_t * p_input = (input_thread_t *)vlc_object_find( p_intf,
332         VLC_OBJECT_INPUT, FIND_ANYWHERE );
333
334     vlc_value_t val;
335
336     /* general properties */
337     if( sender == o_sfilter_saveSettings_ckb)
338     {
339         o_save_settings = [o_sfilter_saveSettings_ckb state]; 
340     }
341
342     /* marquee */
343     else if( sender == o_marq_marq_fld )
344     {
345         if( [[o_marq_marq_fld stringValue] length] == 0 )
346         {
347             val.psz_string = "";
348         }
349         else
350         {
351             val.psz_string = (char *)[[o_marq_marq_fld stringValue] UTF8String];
352         }
353
354         if( p_input )
355             var_Set( p_input->p_libvlc_global, "marq-marquee", val );
356
357         config_PutPsz( p_intf, "marq-marquee", val.psz_string );
358     }
359     
360     else if( sender == o_marq_pos_radio )
361     {
362         val.i_int = [[o_marq_pos_radio selectedCell] tag];
363
364         if( p_input )
365             var_Set( p_input->p_libvlc_global, "marq-position", val );
366
367         config_PutInt( p_intf, "marq-position", val.i_int );
368     }
369     
370     else if( sender == o_marq_color_pop )
371     {
372         val.i_int = strtol( [[[o_colors objectAtIndex: [o_marq_color_pop
373             indexOfSelectedItem]] objectAtIndex: 1] UTF8String], NULL, 0 );
374
375         if( p_input )
376             var_Set( p_input->p_libvlc_global, "marq-color", val );
377
378         config_PutInt( p_intf, "marq-color", val.i_int );
379     }
380     
381     else if( sender == o_marq_opaque_sld )
382     {
383         val.i_int = [o_marq_opaque_sld intValue];
384
385         if( p_input )
386             var_Set( p_input->p_libvlc_global, "marq-opacity", val );
387
388         config_PutInt( p_intf, "marq-opacity", val.i_int );
389     }
390     
391     else if( sender == o_marq_size_pop )
392     {
393         val.i_int = [[o_marq_size_pop titleOfSelectedItem] intValue];
394
395         if( p_input )
396             var_Set( p_input->p_libvlc_global, "marq-size", val );
397
398         config_PutInt( p_intf, "marq-size", val.i_int );
399     }
400     
401     else if( sender == o_marq_tmOut_fld && [[sender stringValue] length] > 0 )
402     {
403         val.i_int = [o_marq_tmOut_fld intValue];
404
405         if( p_input )
406             var_Set( p_input->p_libvlc_global, "marq-timeout", val );
407
408         config_PutInt( p_intf, "marq-timeout", val.i_int );
409     }
410     
411     /* time */
412     
413     else if( sender == o_time_stamp_fld )
414     {
415         if( [[o_time_stamp_fld stringValue] length] == 0 )
416         {
417             val.psz_string = "";
418         }
419         else
420         {
421             val.psz_string = (char *)[[o_time_stamp_fld stringValue] UTF8String];
422         }
423
424         if( p_input )
425             var_Set( p_input->p_libvlc_global, "time-format", val );
426
427         config_PutPsz( p_intf, "time-format", val.psz_string );
428     }
429
430     else if( sender == o_time_pos_radio )
431     {
432         val.i_int = [[o_time_pos_radio selectedCell] tag];
433
434         if( p_input )
435             var_Set( p_input->p_libvlc_global, "time-position", val );
436
437         config_PutInt( p_intf, "time-position", val.i_int );
438     }
439     
440     else if( sender == o_time_color_pop )
441     {
442         val.i_int = strtol( [[[o_colors objectAtIndex: [o_time_color_pop
443             indexOfSelectedItem]] objectAtIndex: 1] UTF8String], NULL, 0 );
444
445         if( p_input )
446             var_Set( p_input->p_libvlc_global, "time-color", val );
447
448         config_PutInt( p_intf, "time-color", val.i_int );
449     }
450     
451     else if( sender == o_time_opaque_sld )
452     {
453         val.i_int = [o_time_opaque_sld intValue];
454
455         if( p_input )
456             var_Set( p_input->p_libvlc_global, "time-opacity", val );
457
458         config_PutInt( p_intf, "time-opacity", val.i_int );
459     }
460     
461     else if( sender == o_time_size_pop )
462     {
463         val.i_int = [[o_time_size_pop titleOfSelectedItem] intValue];
464
465         if( p_input )
466             var_Set( p_input->p_libvlc_global, "time-size", val );
467
468         config_PutInt( p_intf, "time-size", val.i_int );
469     }
470
471     /* logo */
472     else if( sender == o_logo_opaque_sld )
473     {
474         val.i_int = [o_logo_opaque_sld intValue];
475
476         if( p_input )
477             var_Set( p_input->p_libvlc_global, "logo-transparency", val );
478
479         config_PutInt( p_intf, "logo-transparency", val.i_int );
480     }
481     
482     else if( sender == o_logo_pos_radio )
483     {
484         val.i_int = [[o_logo_pos_radio selectedCell] tag];
485
486         if( p_input )
487             var_Set( p_input->p_libvlc_global, "logo-position", val );
488
489         config_PutInt( p_intf, "logo-position", val.i_int );
490     }
491     else
492     {
493         /* just in case */
494         msg_Err( p_intf, "couldn't find any action for sender" );
495     }
496
497     /* clean up */
498     if ( p_input )
499     {
500         o_config_changed = YES;
501         vlc_object_release( p_input );
502     }
503 }
504
505 - (IBAction)enableFilter:(id)sender
506 {
507     if( sender == o_marq_enabled_ckb )
508     {
509         if( [o_marq_enabled_ckb state] == NSOnState )
510         {
511             [self changeFiltersString:"marq" onOrOff:VLC_TRUE];
512         }
513         else
514         {
515             [self changeFiltersString:"marq" onOrOff:VLC_FALSE];
516         }
517         [self enableMarq];
518     }
519     if( sender == o_logo_enabled_ckb )
520     {
521         if( [o_logo_enabled_ckb state] == NSOnState )
522         {
523             [self changeFiltersString:"logo" onOrOff:VLC_TRUE];
524         }
525         else
526         {
527             [self changeFiltersString:"logo" onOrOff:VLC_FALSE];
528         }
529         [self enableLogo];
530     }
531     if( sender == o_time_enabled_ckb )
532     {
533         if( [o_time_enabled_ckb state] == NSOnState )
534         {
535             [self changeFiltersString:"time" onOrOff:VLC_TRUE];
536         }
537         else
538         {
539             [self changeFiltersString:"time" onOrOff:VLC_FALSE];
540         }
541         [self enableTime];
542     }    
543 }
544
545 - (void)enableMarq
546 {
547     [o_marq_color_pop setEnabled: [o_marq_enabled_ckb state]];
548     [o_marq_marq_fld setEnabled: [o_marq_enabled_ckb state]];
549     [o_marq_opaque_sld setEnabled: [o_marq_enabled_ckb state]];
550     [o_marq_size_pop setEnabled: [o_marq_enabled_ckb state]];
551     [o_marq_tmOut_fld setEnabled: [o_marq_enabled_ckb state]];
552     [o_marq_pos_radio setEnabled: [o_marq_enabled_ckb state]];
553 }
554
555 - (void)enableTime
556 {
557     [o_time_color_pop setEnabled: [o_time_enabled_ckb state]];
558     [o_time_stamp_fld setEnabled: [o_time_enabled_ckb state]];
559     [o_time_opaque_sld setEnabled: [o_time_enabled_ckb state]];
560     [o_time_size_pop setEnabled: [o_time_enabled_ckb state]];
561     [o_time_pos_radio setEnabled: [o_time_enabled_ckb state]];
562 }
563
564 - (void)enableLogo
565 {
566     [o_logo_image_btn setEnabled: [o_logo_enabled_ckb state]];
567     [o_logo_image_fld setEnabled: [o_logo_enabled_ckb state]];
568     [o_logo_opaque_sld setEnabled: [o_logo_enabled_ckb state]];
569     [o_logo_pos_radio setEnabled: [o_logo_enabled_ckb state]];
570 }
571
572 - (void)changeFiltersString:(char *)psz_name onOrOff:(vlc_bool_t )b_add
573 {
574     /* copied from ../wxwidgets/extrapanel.cpp
575      * renamed to conform with Cocoa's rules
576      * and slightly modified to suit our needs */
577
578     intf_thread_t * p_intf = VLCIntf;
579     
580     char *psz_parser, *psz_string;
581     psz_string = config_GetPsz( p_intf, "sub-filter" );
582     
583     if( !psz_string ) psz_string = strdup("");
584
585     psz_parser = strstr( psz_string, psz_name );
586
587     if( b_add )
588     {
589         if( !psz_parser )
590         {
591             psz_parser = psz_string;
592             asprintf( &psz_string, (*psz_string) ? "%s:%s" : "%s%s",
593                             psz_string, psz_name );
594             free( psz_parser );
595         }
596         else
597         {
598             return;
599         }
600     }
601     else
602     {
603         if( psz_parser )
604         {
605             memmove( psz_parser, psz_parser + strlen(psz_name) +
606                             (*(psz_parser + strlen(psz_name)) == ':' ? 1 : 0 ),
607                             strlen(psz_parser + strlen(psz_name)) + 1 );
608
609             /* Remove trailing : : */
610             if( *(psz_string+strlen(psz_string ) -1 ) == ':' )
611             {
612                 *(psz_string+strlen(psz_string ) -1 ) = '\0';
613             }
614          }
615          else
616          {
617              free( psz_string );
618              return;
619          }
620     }
621     
622     config_PutPsz( p_intf, "sub-filter", psz_string );
623     
624     /* Try to set on the fly */
625     /* FIXME: enable this once we support on-the-fly addition of this kind of
626      * filters... */
627     vout_thread_t *p_vout;
628     p_vout = (vout_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_VOUT,
629                                               FIND_ANYWHERE );
630     if( p_vout )
631     {
632         var_SetString( p_vout, "sub-filter", psz_string );
633         vlc_object_release( p_vout );
634     }
635
636     free( psz_string );
637
638     vlc_object_release( p_intf );
639
640     o_config_changed = YES;
641 }
642 @end