]> git.sesse.net Git - vlc/blob - modules/gui/macosx/sfilters.m
Mac OS X gui: Sanitize vlc object retaining/releasing.
[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         if( strstr( psz_subfilters, "marq") )
302             [o_marq_enabled_ckb setState: YES];
303         else
304             [o_marq_enabled_ckb setState: NO];
305         
306         if( strstr( psz_subfilters, "logo") )
307             [o_logo_enabled_ckb setState: YES];
308         else
309             [o_logo_enabled_ckb setState: NO];
310         
311         if( strstr( psz_subfilters, "time") )
312             [o_time_enabled_ckb setState: YES];
313         else
314             [o_time_enabled_ckb setState: NO];
315     }
316     [self enableMarq];
317     [self enableLogo];
318     [self enableTime];
319 }
320
321 - (IBAction)logo_selectFile:(id)sender
322 {
323     NSOpenPanel * openPanel = [NSOpenPanel openPanel];
324     SEL sel = @selector(logo_getFile:returnCode:contextInfo:);
325     [openPanel beginSheetForDirectory:nil file:nil types: [NSArray
326         arrayWithObjects: @"png", @"PNG", @"'PNGf'", nil] modalForWindow:
327         o_sfilter_win modalDelegate:self didEndSelector:sel contextInfo:nil];
328 }
329
330 - (void)logo_getFile: (NSOpenPanel *)sheet returnCode:
331     (int)returnCode contextInfo: (void *)contextInfo
332 {
333     if (returnCode == NSOKButton)
334     {
335         [o_logo_image_fld setStringValue: [sheet filename]];
336     }
337 }
338
339 - (IBAction)propertyChanged:(id)sender
340 {
341     intf_thread_t * p_intf = VLCIntf;
342     input_thread_t * p_input = (input_thread_t *)vlc_object_find( p_intf,
343         VLC_OBJECT_INPUT, FIND_ANYWHERE );
344
345     vlc_value_t val;
346
347     /* general properties */
348     if( sender == o_sfilter_saveSettings_ckb)
349     {
350         o_save_settings = [o_sfilter_saveSettings_ckb state]; 
351     }
352
353     /* marquee */
354     else if( sender == o_marq_marq_fld )
355     {
356         if( [[o_marq_marq_fld stringValue] length] == 0 )
357         {
358             val.psz_string = (char *)"";
359         }
360         else
361         {
362             val.psz_string = (char *)[[o_marq_marq_fld stringValue] UTF8String];
363         }
364
365         if( p_input )
366             var_Set( p_input->p_libvlc_global, "marq-marquee", val );
367
368         config_PutPsz( p_intf, "marq-marquee", val.psz_string );
369     }
370     
371     else if( sender == o_marq_pos_radio )
372     {
373         val.i_int = [[o_marq_pos_radio selectedCell] tag];
374
375         if( p_input )
376             var_Set( p_input->p_libvlc_global, "marq-position", val );
377
378         config_PutInt( p_intf, "marq-position", val.i_int );
379     }
380     
381     else if( sender == o_marq_color_pop )
382     {
383         val.i_int = strtol( [[[o_colors objectAtIndex: [o_marq_color_pop
384             indexOfSelectedItem]] objectAtIndex: 1] UTF8String], NULL, 0 );
385
386         if( p_input )
387             var_Set( p_input->p_libvlc_global, "marq-color", val );
388
389         config_PutInt( p_intf, "marq-color", val.i_int );
390     }
391     
392     else if( sender == o_marq_opaque_sld )
393     {
394         val.i_int = [o_marq_opaque_sld intValue];
395
396         if( p_input )
397             var_Set( p_input->p_libvlc_global, "marq-opacity", val );
398
399         config_PutInt( p_intf, "marq-opacity", val.i_int );
400     }
401     
402     else if( sender == o_marq_size_pop )
403     {
404         val.i_int = [[o_marq_size_pop titleOfSelectedItem] intValue];
405
406         if( p_input )
407             var_Set( p_input->p_libvlc_global, "marq-size", val );
408
409         config_PutInt( p_intf, "marq-size", val.i_int );
410     }
411     
412     else if( sender == o_marq_tmOut_fld && [[sender stringValue] length] > 0 )
413     {
414         val.i_int = [o_marq_tmOut_fld intValue];
415
416         if( p_input )
417             var_Set( p_input->p_libvlc_global, "marq-timeout", val );
418
419         config_PutInt( p_intf, "marq-timeout", val.i_int );
420     }
421     
422     /* time */
423     
424     else if( sender == o_time_stamp_fld )
425     {
426         if( [[o_time_stamp_fld stringValue] length] == 0 )
427         {
428             val.psz_string = (char *)"";
429         }
430         else
431         {
432             val.psz_string = (char *)[[o_time_stamp_fld stringValue] UTF8String];
433         }
434
435         if( p_input )
436             var_Set( p_input->p_libvlc_global, "time-format", val );
437
438         config_PutPsz( p_intf, "time-format", val.psz_string );
439     }
440
441     else if( sender == o_time_pos_radio )
442     {
443         val.i_int = [[o_time_pos_radio selectedCell] tag];
444
445         if( p_input )
446             var_Set( p_input->p_libvlc_global, "time-position", val );
447
448         config_PutInt( p_intf, "time-position", val.i_int );
449     }
450     
451     else if( sender == o_time_color_pop )
452     {
453         val.i_int = strtol( [[[o_colors objectAtIndex: [o_time_color_pop
454             indexOfSelectedItem]] objectAtIndex: 1] UTF8String], NULL, 0 );
455
456         if( p_input )
457             var_Set( p_input->p_libvlc_global, "time-color", val );
458
459         config_PutInt( p_intf, "time-color", val.i_int );
460     }
461     
462     else if( sender == o_time_opaque_sld )
463     {
464         val.i_int = [o_time_opaque_sld intValue];
465
466         if( p_input )
467             var_Set( p_input->p_libvlc_global, "time-opacity", val );
468
469         config_PutInt( p_intf, "time-opacity", val.i_int );
470     }
471     
472     else if( sender == o_time_size_pop )
473     {
474         val.i_int = [[o_time_size_pop titleOfSelectedItem] intValue];
475
476         if( p_input )
477             var_Set( p_input->p_libvlc_global, "time-size", val );
478
479         config_PutInt( p_intf, "time-size", val.i_int );
480     }
481
482     /* logo */
483     else if( sender == o_logo_opaque_sld )
484     {
485         val.i_int = [o_logo_opaque_sld intValue];
486
487         if( p_input )
488             var_Set( p_input->p_libvlc_global, "logo-transparency", val );
489
490         config_PutInt( p_intf, "logo-transparency", val.i_int );
491     }
492     
493     else if( sender == o_logo_pos_radio )
494     {
495         val.i_int = [[o_logo_pos_radio selectedCell] tag];
496
497         if( p_input )
498             var_Set( p_input->p_libvlc_global, "logo-position", val );
499
500         config_PutInt( p_intf, "logo-position", val.i_int );
501     }
502     else
503     {
504         /* just in case */
505         msg_Err( p_intf, "couldn't find any action for sender" );
506     }
507
508     /* clean up */
509     if ( p_input )
510     {
511         o_config_changed = YES;
512         vlc_object_release( p_input );
513     }
514 }
515
516 - (IBAction)enableFilter:(id)sender
517 {
518     if( sender == o_marq_enabled_ckb )
519     {
520         if( [o_marq_enabled_ckb state] == NSOnState )
521         {
522             [self changeFiltersString:(char *)"marq" onOrOff:VLC_TRUE];
523         }
524         else
525         {
526             [self changeFiltersString:(char *)"marq" onOrOff:VLC_FALSE];
527         }
528         [self enableMarq];
529     }
530     if( sender == o_logo_enabled_ckb )
531     {
532         if( [o_logo_enabled_ckb state] == NSOnState )
533         {
534             [self changeFiltersString:(char *)"logo" onOrOff:VLC_TRUE];
535         }
536         else
537         {
538             [self changeFiltersString:(char *)"logo" onOrOff:VLC_FALSE];
539         }
540         [self enableLogo];
541     }
542     if( sender == o_time_enabled_ckb )
543     {
544         if( [o_time_enabled_ckb state] == NSOnState )
545         {
546             [self changeFiltersString:(char *)"time" onOrOff:VLC_TRUE];
547         }
548         else
549         {
550             [self changeFiltersString:(char *)"time" onOrOff:VLC_FALSE];
551         }
552         [self enableTime];
553     }    
554 }
555
556 - (void)enableMarq
557 {
558     [o_marq_color_pop setEnabled: [o_marq_enabled_ckb state]];
559     [o_marq_marq_fld setEnabled: [o_marq_enabled_ckb state]];
560     [o_marq_opaque_sld setEnabled: [o_marq_enabled_ckb state]];
561     [o_marq_size_pop setEnabled: [o_marq_enabled_ckb state]];
562     [o_marq_tmOut_fld setEnabled: [o_marq_enabled_ckb state]];
563     [o_marq_pos_radio setEnabled: [o_marq_enabled_ckb state]];
564 }
565
566 - (void)enableTime
567 {
568     [o_time_color_pop setEnabled: [o_time_enabled_ckb state]];
569     [o_time_stamp_fld setEnabled: [o_time_enabled_ckb state]];
570     [o_time_opaque_sld setEnabled: [o_time_enabled_ckb state]];
571     [o_time_size_pop setEnabled: [o_time_enabled_ckb state]];
572     [o_time_pos_radio setEnabled: [o_time_enabled_ckb state]];
573 }
574
575 - (void)enableLogo
576 {
577     [o_logo_image_btn setEnabled: [o_logo_enabled_ckb state]];
578     [o_logo_image_fld setEnabled: [o_logo_enabled_ckb state]];
579     [o_logo_opaque_sld setEnabled: [o_logo_enabled_ckb state]];
580     [o_logo_pos_radio setEnabled: [o_logo_enabled_ckb state]];
581 }
582
583 - (void)changeFiltersString:(char *)psz_name onOrOff:(vlc_bool_t )b_add
584 {
585     /* copied from ../wxwidgets/extrapanel.cpp
586      * renamed to conform with Cocoa's rules
587      * and slightly modified to suit our needs */
588
589     intf_thread_t * p_intf = VLCIntf;
590     
591     char *psz_parser, *psz_string;
592     psz_string = config_GetPsz( p_intf, "sub-filter" );
593     
594     if( !psz_string ) psz_string = strdup("");
595
596     psz_parser = strstr( psz_string, psz_name );
597
598     if( b_add )
599     {
600         if( !psz_parser )
601         {
602             psz_parser = psz_string;
603             asprintf( &psz_string, (*psz_string) ? "%s:%s" : "%s%s",
604                             psz_string, psz_name );
605             free( psz_parser );
606         }
607         else
608         {
609             return;
610         }
611     }
612     else
613     {
614         if( psz_parser )
615         {
616             memmove( psz_parser, psz_parser + strlen(psz_name) +
617                             (*(psz_parser + strlen(psz_name)) == ':' ? 1 : 0 ),
618                             strlen(psz_parser + strlen(psz_name)) + 1 );
619
620             /* Remove trailing : : */
621             if( *(psz_string+strlen(psz_string ) -1 ) == ':' )
622             {
623                 *(psz_string+strlen(psz_string ) -1 ) = '\0';
624             }
625          }
626          else
627          {
628              free( psz_string );
629              return;
630          }
631     }
632     
633     config_PutPsz( p_intf, "sub-filter", psz_string );
634     
635     /* Try to set on the fly */
636     /* FIXME: enable this once we support on-the-fly addition of this kind of
637      * filters... */
638     vout_thread_t *p_vout;
639     p_vout = (vout_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_VOUT,
640                                               FIND_ANYWHERE );
641     if( p_vout )
642     {
643         var_SetString( p_vout, "sub-filter", psz_string );
644         vlc_object_release( p_vout );
645     }
646
647     free( psz_string );
648
649     o_config_changed = YES;
650 }
651 @end