]> git.sesse.net Git - vlc/commitdiff
Added widget, fix save, etc.
authorJérome Decoodt <djc@videolan.org>
Thu, 5 May 2005 10:57:55 +0000 (10:57 +0000)
committerJérome Decoodt <djc@videolan.org>
Thu, 5 May 2005 10:57:55 +0000 (10:57 +0000)
Known bugs: changing advanced view, detection of MacOS X version at
            runtime and many others...

modules/gui/macosx/prefs.h
modules/gui/macosx/prefs.m
modules/gui/macosx/prefs_widgets.h
modules/gui/macosx/prefs_widgets.m

index 2991feb044ff2ae692357f2856c7280abc4849f9..ee3ee3ef2905f576b4137e79152b6d080a6022b3 100644 (file)
@@ -28,6 +28,7 @@
     VLCTreeItem *o_parent;
     NSMutableArray *o_children;
     int i_object_category;
+    NSView *o_view;
 }
 
 + (VLCTreeItem *)rootItem;
@@ -36,7 +37,8 @@
 - (int)getObjectID;
 - (NSString *)getName;
 - (BOOL)hasPrefs:(NSString *)o_module_name;
-- (NSView *)showView:(NSScrollView *)o_prefs_view;
+- (NSView *)showView:(NSScrollView *)o_prefs_view advancedView:(vlc_bool_t) b_advanced;
+- (void)applyChanges;
 
 @end
 
index aead51d6a389bc3f829e38980afb63d861e003c8..a8ab3d9cecd193eb29e23e9fa6bb1c243f6cbdc3 100644 (file)
@@ -126,6 +126,7 @@ static VLCPrefs *_o_sharedMainInstance = nil;
 - (IBAction)savePrefs: (id)sender
 {
     /* TODO: call savePrefs on Root item */
+    [[VLCTreeItem rootItem] applyChanges];
     config_SaveConfigFile( p_intf, NULL );
     [o_prefs_window orderOut:self];
 }
@@ -148,10 +149,8 @@ static VLCPrefs *_o_sharedMainInstance = nil;
     if( i_return == NSAlertAlternateReturn )
     {
         config_ResetAll( p_intf );
-        [[o_tree itemAtRow:[o_tree selectedRow]] showView:o_prefs_view];
-/*        [self showViewForID: [[o_tree itemAtRow:[o_tree selectedRow]] getObjectID]
-            andName: [[o_tree itemAtRow:[o_tree selectedRow]] getName]];
-*/
+        [[o_tree itemAtRow:[o_tree selectedRow]] showView:o_prefs_view advancedView:
+            ([o_advanced_ckb state] == NSOnState)?VLC_TRUE:VLC_FALSE];
     }
 }
 
@@ -160,10 +159,8 @@ static VLCPrefs *_o_sharedMainInstance = nil;
     b_advanced = !b_advanced;
     [o_advanced_ckb setState: b_advanced];
     /* refresh the view of the current treeitem */
-    [[o_tree itemAtRow:[o_tree selectedRow]] showView:o_prefs_view];
-/*    [self showViewForID: [[o_tree itemAtRow:[o_tree selectedRow]] getObjectID]
-        andName: [[o_tree itemAtRow:[o_tree selectedRow]] getName]];
-*/
+    [[o_tree itemAtRow:[o_tree selectedRow]] showView:o_prefs_view advancedView:
+            ([o_advanced_ckb state] == NSOnState)?VLC_TRUE:VLC_FALSE];
 }
 
 - (void)loadConfigTree
@@ -177,7 +174,8 @@ static VLCPrefs *_o_sharedMainInstance = nil;
 /* update the document view to the view of the selected tree item */
 - (void)outlineViewSelectionDidChange:(NSNotification *)o_notification
 {
-    [[o_tree itemAtRow:[o_tree selectedRow]] showView: o_prefs_view];
+    [[o_tree itemAtRow:[o_tree selectedRow]] showView: o_prefs_view advancedView:
+            ([o_advanced_ckb state] == NSOnState)?VLC_TRUE:VLC_FALSE];
 }
 
 @end
@@ -189,7 +187,7 @@ static VLCPrefs *_o_sharedMainInstance = nil;
 }
 
 - (BOOL)outlineView:(NSOutlineView *)outlineView isItemExpandable:(id)item {
-    return (item == nil) ? YES : ([item numberOfChildren] != -1);
+    return (item == nil) ? YES : (([item numberOfChildren] != -1) && ([item numberOfChildren] != 0));
 }
 
 - (id)outlineView:(NSOutlineView *)outlineView child:(int)index ofItem:(id)item {
@@ -219,6 +217,7 @@ static VLCTreeItem *o_root_item = nil;
         o_parent = o_parent_item;
         o_children = o_children_array;
         i_object_category = i_category;
+        o_view = nil;
     }
     return( self );
 }
@@ -243,11 +242,11 @@ static VLCTreeItem *o_root_item = nil;
         return o_children;
     if( [ o_children count] == 0 )
     {
-        intf_thread_t *p_intf = VLCIntf;
+        intf_thread_t   *p_intf = VLCIntf;
         vlc_list_t      *p_list;
         module_t        *p_module = NULL;
         module_config_t *p_item;
-        int i_index;
+        int             i_index;
 
         /* List the modules */
         p_list = vlc_list_find( p_intf, VLC_OBJECT_MODULE, FIND_ANYWHERE );
@@ -290,8 +289,10 @@ static VLCTreeItem *o_root_item = nil;
                         break;
                     case CONFIG_SUBCATEGORY:
                         o_child_name = [[VLCMain sharedInstance] localizedString: config_CategoryNameGet(p_item->i_value ) ];
-                        [p_last_category->o_children addObject:[[VLCTreeItem alloc] initWithName: o_child_name
-                            ID: p_item->i_value parent:p_last_category children:[[NSMutableArray alloc] initWithCapacity:10] whithCategory: p_item - p_module->p_config]];
+                        if( p_item->i_value != SUBCAT_VIDEO_GENERAL &&
+                            p_item->i_value != SUBCAT_AUDIO_GENERAL )
+                            [p_last_category->o_children addObject:[[VLCTreeItem alloc] initWithName: o_child_name
+                                ID: p_item->i_value parent:p_last_category children:[[NSMutableArray alloc] initWithCapacity:10] whithCategory: p_item - p_module->p_config]];
                         break;
                     default:
                         break;
@@ -343,7 +344,7 @@ static VLCTreeItem *o_root_item = nil;
 
                 long cookie;
                 vlc_bool_t b_found = VLC_FALSE;
-                int i;
+                unsigned int i;
                 VLCTreeItem* p_category_item, * p_subcategory_item;
                 for (i = 0 ; i < [o_children count] ; i++)
                 {
@@ -431,159 +432,312 @@ static VLCTreeItem *o_root_item = nil;
     return( NO );
 }
 
-- (NSView *)showView:(NSScrollView *)o_prefs_view
+- (NSView *)showView:(NSScrollView *)o_prefs_view advancedView:(vlc_bool_t) b_advanced
 {
 fprintf( stderr, "[%s] showView\n", [o_name UTF8String] );
-    vlc_list_t *p_list;
-    intf_thread_t *p_intf = VLCIntf;
-    module_t *p_parser;
-    module_config_t *p_item, *p_first_item;
-    NSView *o_view;
-    NSRect s_rc;                        /* rect                         */
-    NSTextField *o_text_field;         /* input field / label          */
-    NSRect s_vrc;                       /* view rect                    */
-    NSString *o_module_name;
-    int i_pos, i_module_tag;
-    NSPoint o_pos;
-    vlc_bool_t b_advanced = VLC_TRUE;
-    s_vrc = [[o_prefs_view contentView] bounds];
-    s_vrc.size.height -= 4;
-    
-    o_view = [[NSView alloc] initWithFrame: s_vrc];
-
-    p_list = vlc_list_find( p_intf, VLC_OBJECT_MODULE, FIND_ANYWHERE );
-    
-    /* Get a pointer to the module */
-    if( i_object_category == -1 )
-    {
-        p_parser = (module_t *) vlc_object_get( p_intf, i_object_id );
-        if( !p_parser || p_parser->i_object_type != VLC_OBJECT_MODULE )
-        {
-            /* 0OOoo something went really bad */
-            vlc_list_release( p_list );
-            return o_view;
-        }
-        o_module_name = [NSString stringWithUTF8String: p_parser->psz_object_name];    
-fprintf( stderr, "showView: going to show [%d] %s\n", i_object_id, p_parser->psz_object_name );
-        p_first_item = p_item = p_parser->p_config;
-    }
-    else
+    if( o_view == nil )
     {
-        int i_index;
-        if( !p_list ) return o_view;
-
-        /*
-        * Find the main module
-        */
-        for( i_index = 0; i_index < p_list->i_count; i_index++ )
-        {
-            p_parser = (module_t *)p_list->p_values[i_index].p_object;
-            if( !strcmp( p_parser->psz_object_name, "main" ) )
-                break;
-        }
-        if( p_parser == NULL )
+        intf_thread_t   *p_intf = VLCIntf;
+        vlc_list_t      *p_list;
+        module_t        *p_parser = NULL;
+        module_config_t *p_item;
+        NSRect          s_vrc;
+
+        s_vrc = [[o_prefs_view contentView] bounds]; s_vrc.size.height -= 4;
+        o_view = [[VLCFlippedView alloc] initWithFrame: s_vrc];
+        [o_view setAutoresizingMask: NSViewWidthSizable | NSViewHeightSizable];
+        
+        /* Get a pointer to the module */
+        if( i_object_category == -1 )
         {
-            msg_Err( p_intf, "could not find the main module in our preferences" );
-            return o_view;
-        }
-        p_first_item = p_item = (p_parser->p_config + i_object_category);
-    }
-    o_view = nil;
-    i_module_tag = 3;
-
-/* These defines should come from "Apple Human Interface Guidelines" */
-#define X_ORIGIN 20
-#define Y_ORIGIN 17
-#define Y_INTER 8
-    /* Init View */
-    s_vrc = [[o_prefs_view contentView] bounds]; s_vrc.size.height -= 4;
-    o_view = [[VLCFlippedView alloc] initWithFrame: s_vrc];
-    [o_view setAutoresizingMask: NSViewWidthSizable];
-
-    o_pos.x = X_ORIGIN;
-    o_pos.y = Y_ORIGIN;
-    BOOL b_right_cat = TRUE;
+            p_parser = (module_t *) vlc_object_get( p_intf, i_object_id );
+            if( !p_parser || p_parser->i_object_type != VLC_OBJECT_MODULE )
+            {
+                /* 0OOoo something went really bad */
+                return nil;
+            }
+            p_item = p_parser->p_config;
+            int i = 0;
+            int i_yPos = -2;
+            int i_lastItem = 0;
 
-    if( p_item ) do ; while ( ( p_item->i_type == CONFIG_CATEGORY || p_item->i_type == CONFIG_SUBCATEGORY ) && p_item++ ); 
+            p_item = p_parser->p_config + 1;
 
-    if( p_item ) do
-    {
-fprintf( stderr, "Category: %d\n", p_item->i_type );
-        if( p_item->i_type == CONFIG_HINT_CATEGORY )
-        {
-            if( !strcmp( p_parser->psz_object_name, "main" ) &&
-                [o_name isEqualToString: [[VLCMain sharedInstance] localizedString: p_item->psz_text]] )
-                b_right_cat = TRUE;
-            else if( strcmp( p_parser->psz_object_name, "main" ) )
-                 b_right_cat = TRUE;
-            else b_right_cat = FALSE;
-        }
-        else if( p_item->i_type == CONFIG_HINT_END && !strcmp( p_parser->psz_object_name, "main" ) )
-            b_right_cat = FALSE;
+            do
+            {
+                if( !p_item )
+                {
+                    fprintf( stderr, "Something is going very bad... skipping null item\n" );
+                    break;
+                }
+                switch(p_item->i_type)
+                {
+                case CONFIG_SUBCATEGORY:
+fprintf( stderr, "drawing subcategory %s\n", [o_name UTF8String] );
+                    break;
+                case CONFIG_SECTION:
+fprintf( stderr, "drawing section %s\n", p_item->psz_text );
+                    break;
+                case CONFIG_CATEGORY:
+fprintf( stderr, "drawing category %s\n", [o_name UTF8String] );
+                    break;
+                case CONFIG_HINT_END:
+fprintf( stderr, "end of (sub)category\n" );
+                    break;
+                case CONFIG_HINT_USAGE:
+fprintf( stderr, "skipping hint usage\n" );
+                    break;
+                default:
+fprintf( stderr, "%s (%d) is ", p_item->psz_name, p_item->i_type );
+                {
+                    VLCConfigControl *o_control = nil;
+                    int i_widget = 0;
+                    if( p_item->b_advanced && (! b_advanced) )
+                        break;
+                    switch( p_item->i_type )
+                    {
+                    case CONFIG_ITEM_STRING:
+fprintf( stderr, "CONFIG_ITEM_STRING" );
+                        if( !p_item->i_list )
+                            i_widget = CONFIG_ITEM_STRING;
+                        else
+                            i_widget = CONFIG_ITEM_STRING_LIST;
+                        break;
+                    case CONFIG_ITEM_FILE:
+                    case CONFIG_ITEM_DIRECTORY:
+fprintf( stderr, "CONFIG_ITEM_FILE" );
+                        i_widget = CONFIG_ITEM_FILE;
+                        break;
+                    case CONFIG_ITEM_MODULE:
+                    case CONFIG_ITEM_MODULE_CAT:
+fprintf( stderr, "CONFIG_ITEM_MODULE" );
+                        i_widget = CONFIG_ITEM_MODULE;
+                        break;
+                    case CONFIG_ITEM_INTEGER:
+fprintf( stderr, "CONFIG_ITEM_INTEGER" );
+                        if( p_item->i_list )
+                            i_widget = CONFIG_ITEM_STRING_LIST;
+                        else if( p_item->i_min != 0 || p_item->i_max != 0 )
+                            i_widget = CONFIG_ITEM_RANGED_INTEGER;
+                        else
+                            i_widget = CONFIG_ITEM_INTEGER;
+                        break;
+                    case CONFIG_ITEM_FLOAT:
+fprintf( stderr, "CONFIG_ITEM_FLOAT" );
+                        if( p_item->f_min != 0 || p_item->f_max != 0 )
+                            i_widget = CONFIG_ITEM_RANGED_INTEGER;
+                        else
+                            i_widget = CONFIG_ITEM_INTEGER;
+                        break;
+                    case CONFIG_ITEM_BOOL:
+fprintf( stderr, "CONFIG_ITEM_BOOL" );
+                        i_widget = CONFIG_ITEM_BOOL;
+                        break;
+                    case CONFIG_ITEM_KEY:
+fprintf( stderr, "CONFIG_ITEM_KEY" );
+#define MACOS_VERSION 4
+                        if( MACOS_VERSION < 3 )
+                            i_widget = CONFIG_ITEM_KEY_BEFORE_10_3;
+                        else
+                            i_widget = CONFIG_ITEM_KEY_AFTER_10_3;
+                        break;
+                    case CONFIG_ITEM_MODULE_LIST:
+                    case CONFIG_ITEM_MODULE_LIST_CAT:
+fprintf( stderr, "CONFIG_ITEM_MODULE_LIST" );
+                        i_widget = CONFIG_ITEM_MODULE_LIST;
+                        break;
+                    default:
+fprintf( stderr, "***UNKNOWN***" );
+                    }
+                    if( i_widget != 0 )
+                    {
+                        i_yPos += [VLCConfigControl calcVerticalMargin:i_widget lastItem:i_lastItem];
+                        o_control = [VLCConfigControl newControl:p_item
+                                                      withView:o_view
+                                                      yOffset: i_yPos
+                                                      lastItem: i_lastItem];
+                        if( o_control != nil )
+                        {
+                            i_yPos += [o_control frame].size.height;
+                            i_lastItem = i_widget;
+                            [o_control setAutoresizingMask: NSViewMaxYMargin | NSViewWidthSizable];
+                            [o_view addSubview: o_control];
+                        }
+                    }
+fprintf( stderr, "\n" );
+                    break;
+                }
+                }
+            } while( p_item++->i_type != CONFIG_HINT_END );
 
-        if( (p_item->b_advanced && !b_advanced ) || !b_right_cat )
-        {
-            continue;
+            vlc_object_release( p_parser );
         }
-fprintf( stderr, "Creating view for: %s\n", p_item->psz_name );
-        VLCConfigControl *o_control = nil;
-        switch( p_item->i_type )
+        else
         {
-            case CONFIG_ITEM_STRING:
+            int i_index;
+            p_list = vlc_list_find( p_intf, VLC_OBJECT_MODULE, FIND_ANYWHERE );
+            if( !p_list ) return o_view;
+
+            /*
+            * Find the main module
+            */
+            for( i_index = 0; i_index < p_list->i_count; i_index++ )
             {
-                if( !p_item->i_list )
-                    o_control = [StringConfigControl newControl:p_item
-                                                      withView:o_view withObject:p_intf offset: o_pos];
-                else
-                    o_control = [StringListConfigControl newControl:p_item
-                                                                withView:o_view withObject:p_intf offset: o_pos];
+                p_parser = (module_t *)p_list->p_values[i_index].p_object;
+                if( !strcmp( p_parser->psz_object_name, "main" ) )
+                    break;
             }
-            break;
-            case CONFIG_ITEM_FILE:
-            case CONFIG_ITEM_DIRECTORY:
+            if( p_parser == NULL )
             {
-                o_control = [FileConfigControl newControl: p_item withView: o_view
-                                                    withObject: p_intf offset: o_pos];
+                msg_Err( p_intf, "could not find the main module in our preferences" );
+                return o_view;
             }
-            break;
-            case CONFIG_ITEM_INTEGER:
-            {
-                if( !p_item->i_list )
-                    o_control = [IntegerConfigControl newControl:p_item
-                                                            withView:o_view withObject:p_intf offset: o_pos];
+            p_item = (p_parser->p_config + i_object_category);
+            if( ( p_item->i_type == CONFIG_CATEGORY ) &&
+              ( ( p_item->i_value == CAT_AUDIO )  || ( p_item->i_value == CAT_VIDEO ) ) )
+                p_item++;
 
-                else if( p_item->i_min != 0 || p_item->i_max != 0 )
-                    o_control = [RangedIntegerConfigControl newControl:p_item
-                                                            withView:o_view withObject:p_intf offset: o_pos];
-                else
-                    o_control = [IntegerListConfigControl newControl:p_item
-                                                                withView:o_view withObject:p_intf offset: o_pos];
-            }
-            break;
-            case CONFIG_ITEM_KEY:
+            int i = 0;
+            int i_yPos = -2;
+            int i_lastItem = 0;
+
+            do
             {
-                o_control = [KeyConfigControl newControl:p_item withView:o_view withObject:p_intf offset: o_pos];
-            }
-            break;
-        }
-        if( o_control != nil )
-        {
-            [o_view addSubview: o_control];
-            o_pos.y += [o_control frame].size.height + Y_INTER;
+                p_item++;
+                if( !p_item )
+                {
+                    fprintf( stderr, "Something is going very bad... skipping null item\n" );
+                    break;
+                }
+                switch(p_item->i_type)
+                {
+                case CONFIG_SUBCATEGORY:
+fprintf( stderr, "drawing subcategory %s\n", [o_name UTF8String] );
+                    break;
+                case CONFIG_SECTION:
+fprintf( stderr, "drawing section %s\n", p_item->psz_text );
+                    break;
+                case CONFIG_CATEGORY:
+fprintf( stderr, "drawing category %s\n", [o_name UTF8String] );
+                    break;
+                case CONFIG_HINT_END:
+fprintf( stderr, "end of (sub)category\n" );
+                    break;
+                case CONFIG_HINT_USAGE:
+fprintf( stderr, "skipping hint usage\n" );
+                    break;
+                default:
+fprintf( stderr, "%s (%d) is ", p_item->psz_name, p_item->i_type );
+                {
+                    VLCConfigControl *o_control = nil;
+                    int i_widget = 0;
+                    switch( p_item->i_type )
+                    {
+                    case CONFIG_ITEM_STRING:
+fprintf( stderr, "CONFIG_ITEM_STRING" );
+                        if( !p_item->i_list )
+                            i_widget = CONFIG_ITEM_STRING;
+                        else
+                            i_widget = CONFIG_ITEM_STRING_LIST;
+                        break;
+                    case CONFIG_ITEM_FILE:
+                    case CONFIG_ITEM_DIRECTORY:
+fprintf( stderr, "CONFIG_ITEM_FILE" );
+                        i_widget = CONFIG_ITEM_FILE;
+                        break;
+                    case CONFIG_ITEM_MODULE:
+                    case CONFIG_ITEM_MODULE_CAT:
+fprintf( stderr, "CONFIG_ITEM_MODULE" );
+                        i_widget = CONFIG_ITEM_MODULE;
+                        break;
+                    case CONFIG_ITEM_INTEGER:
+fprintf( stderr, "CONFIG_ITEM_INTEGER" );
+                        if( p_item->i_list )
+                            i_widget = CONFIG_ITEM_STRING_LIST;
+                        else if( p_item->i_min != 0 || p_item->i_max != 0 )
+                            i_widget = CONFIG_ITEM_RANGED_INTEGER;
+                        else
+                            i_widget = CONFIG_ITEM_INTEGER;
+                        break;
+                    case CONFIG_ITEM_FLOAT:
+fprintf( stderr, "CONFIG_ITEM_FLOAT" );
+                        if( p_item->f_min != 0 || p_item->f_max != 0 )
+                            i_widget = CONFIG_ITEM_RANGED_INTEGER;
+                        else
+                            i_widget = CONFIG_ITEM_INTEGER;
+                        break;
+                    case CONFIG_ITEM_BOOL:
+fprintf( stderr, "CONFIG_ITEM_BOOL" );
+                        i_widget = CONFIG_ITEM_BOOL;
+                        break;
+                    case CONFIG_ITEM_KEY:
+fprintf( stderr, "CONFIG_ITEM_KEY" );
+    #define MACOS_VERSION 4
+                        if( MACOS_VERSION < 3 )
+                            i_widget = CONFIG_ITEM_KEY_BEFORE_10_3;
+                        else
+                            i_widget = CONFIG_ITEM_KEY_AFTER_10_3;
+                        break;
+                    case CONFIG_ITEM_MODULE_LIST:
+                    case CONFIG_ITEM_MODULE_LIST_CAT:
+fprintf( stderr, "CONFIG_ITEM_MODULE_LIST" );
+                        i_widget = CONFIG_ITEM_MODULE_LIST;
+                        break;
+                    default:
+fprintf( stderr, "***UNKNOWN***" );
+                    }
+                    if( i_widget != 0 )
+                    {
+                        i_yPos += [VLCConfigControl calcVerticalMargin:i_widget lastItem:i_lastItem];
+                        o_control = [VLCConfigControl newControl:p_item
+                                                      withView:o_view
+                                                      yOffset: i_yPos
+                                                      lastItem: i_lastItem];
+                        if( o_control != nil )
+                        {
+                            i_yPos += [o_control frame].size.height;
+                            i_lastItem = i_widget;
+                            [o_control setAutoresizingMask: NSViewMaxYMargin | NSViewWidthSizable];
+                            [o_view addSubview: o_control];
+                        }
+                    }
+fprintf( stderr, "\n" );
+                    break;
+                }
+                }
+            } while ( ( p_item->i_type != CONFIG_HINT_END ) && ( p_item->i_type != CONFIG_SUBCATEGORY ) );
+
+            vlc_object_release( p_parser );
+            vlc_list_release( p_list );
         }
-    #undef Y_ORIGIN
-    #undef X_ORIGIN
     }
-    while( p_item->i_type != CONFIG_HINT_END && p_item->i_type != CONFIG_CATEGORY && p_item->i_type != CONFIG_SUBCATEGORY && p_item++ );
-
-    vlc_object_release( p_parser );
-    vlc_list_release( p_list );
-
-    [o_prefs_view setDocumentView: o_view];
-    [o_prefs_view setNeedsDisplay: TRUE];
+    else
+    {
+        NSRect s_vrc;
+        s_vrc = [[o_prefs_view contentView] bounds]; s_vrc.size.height -= 4;
+        [o_view setFrame: s_vrc];
+    }
+    if( o_view != nil )
+        [o_prefs_view setDocumentView:o_view];
     return o_view;
 }
 
+- (void)applyChanges
+{
+    unsigned int i;
+    if( o_view != nil )
+    {
+    //Item has been shown
+fprintf( stderr, "[%s] applying changes\n", [o_name cString]);
+        NSArray *o_subviews = [o_view subviews];
+        for( i = 0 ; i < [o_subviews count] ; i++ )
+            [[o_subviews objectAtIndex:i] applyChanges];
+    }
+    if( o_children != IsALeafNode )
+        for( i = 0 ; i < [o_children count] ; i++ )
+            [[o_children objectAtIndex:i] applyChanges];
+}
+
 @end
 
 
index 617e4c5345e948b8aa58cbe8f2bc543ad876eac7..08039b269c4880aafd9f9e0264b743f15e548a08 100644 (file)
  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  *****************************************************************************/
 
+#define CONFIG_ITEM_STRING_LIST (CONFIG_ITEM_STRING + 1)
+#define CONFIG_ITEM_RANGED_INTEGER (CONFIG_ITEM_INTEGER + 1)
+#define CONFIG_ITEM_KEY_BEFORE_10_3 (CONFIG_ITEM_KEY + 1)
+#define CONFIG_ITEM_KEY_AFTER_10_3 (CONFIG_ITEM_KEY + 2)
 
 @interface VLCConfigControl : NSView
 {
-    vlc_object_t    *p_this;
     module_config_t *p_item;
     char            *psz_name;
     NSTextField     *o_label;
     int             i_type;
     vlc_bool_t      b_advanced;
-    NSView          *contentView;
 }
 
-+ (VLCConfigControl *)newControl: (module_config_t *)p_item withView: (NSView *)o_parent_view withObject: (vlc_object_t *)p_this offset:(NSPoint) offset;
-- (id)initWithFrame: (NSRect)frame item: (module_config_t *)p_item withObject: (vlc_object_t *)_p_this;
++ (VLCConfigControl *)newControl: (module_config_t *)_p_item withView: (NSView *)o_parent_view yOffset:(int) i_yPos lastItem: (int) i_lastItem;
+- (id)initWithFrame: (NSRect)frame item: (module_config_t *)p_item;
 - (NSString *)getName;
 - (int)getType;
 - (BOOL)isAdvanced;
 - (int)intValue;
 - (float)floatValue;
 - (char *)stringValue;
+- (void)applyChanges;
+static NSMenu   *o_keys_menu = nil;
 
-@end
-
-@interface KeyConfigControl : VLCConfigControl
-{
-    NSMatrix        *o_matrix;
-    NSComboBox      *o_combo;
-}
++ (int)calcVerticalMargin: (int)i_curItem lastItem:(int)i_lastItem;
 
 @end
-#if 0
 
-@interface ModuleConfigControl : VLCConfigControl
-{
-    NSPopUpButton   *o_popup;
-}
-
-@end
-#endif
 @interface StringConfigControl : VLCConfigControl
 {
     NSTextField     *o_textfield;
 }
 
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
 @end
 
 @interface StringListConfigControl : VLCConfigControl
     NSComboBox      *o_combo;
 }
 
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
 @end
+
 @interface FileConfigControl : VLCConfigControl
 {
     NSTextField     *o_textfield;
     BOOL            b_directory;
 }
 
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
 - (IBAction)openFileDialog: (id)sender;
 - (void)pathChosenInPanel:(NSOpenPanel *)o_sheet withReturn:(int)i_return_code contextInfo:(void  *)o_context_info;
 
 @end
 
+@interface ModuleConfigControl : VLCConfigControl
+{
+    NSPopUpButton   *o_popup;
+}
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
+@end
+
 @interface IntegerConfigControl : VLCConfigControl
 {
     NSTextField     *o_textfield;
     NSStepper       *o_stepper;
 }
 
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
 - (IBAction)stepperChanged:(id)sender;
 - (void)textfieldChanged:(NSNotification *)o_notification;
 
     NSComboBox      *o_combo;
 }
 
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
 @end
 
 @interface RangedIntegerConfigControl : VLCConfigControl
     NSTextField     *o_textfield_max;
 }
 
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
 - (IBAction)sliderChanged:(id)sender;
 - (void)textfieldChanged:(NSNotification *)o_notification;
 
 @end
-#if 0
+
+@interface BoolConfigControl : VLCConfigControl
+{
+    NSButton        *o_checkbox;
+}
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
+@end
 
 @interface FloatConfigControl : VLCConfigControl
 {
     NSTextField     *o_textfield;
+    NSStepper       *o_stepper;
 }
 
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+- (IBAction)stepperChanged:(id)sender;
+- (void)textfieldChanged:(NSNotification *)o_notification;
+
 @end
 
 @interface RangedFloatConfigControl : VLCConfigControl
     NSTextField     *o_textfield_max;
 }
 
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
 - (IBAction)sliderChanged:(id)sender;
 - (void)textfieldChanged:(NSNotification *)o_notification;
 
 @end
 
+@interface KeyConfigControlBefore103 : VLCConfigControl
+{
+    NSButton        *o_cmd_checkbox;
+    NSButton        *o_ctrl_checkbox;
+    NSButton        *o_alt_checkbox;
+    NSButton        *o_shift_checkbox;
+    NSPopUpButton   *o_popup;
+}
 
-@interface BoolConfigControl : VLCConfigControl
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
+@end
+
+@interface KeyConfigControlAfter103 : VLCConfigControl
 {
-    NSButton        *o_checkbox;
+    NSPopUpButton   *o_popup;
+}
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
+@end
+
+@interface ModuleListConfigControl : VLCConfigControl
+{
+    NSTextField     *o_textfield;
+    NSScrollView    *o_scrollview;
+    NSMutableArray  *o_modulearray;
 }
 
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem;
+
 @end
-#endif
\ No newline at end of file
+
+//#undef CONFIG_ITEM_LIST_STRING
+//#undef CONFIG_ITEM_RANGED_INTEGER
+//#undef CONFIG_ITEM_KEY_BEFORE_10_3
+//#undef CONFIG_ITEM_KEY_AFTER_10_3
+
index 1ac1157f81e0f7f3252a3593fa3954e3683883c9..383f5388b2457f9caa36bd5cd44ddb12948dcd9e 100644 (file)
@@ -4,7 +4,8 @@
  * Copyright (C) 2002-2003 VideoLAN
  * $Id$
  *
- * Authors: Derk-Jan Hartman <hartman at videolan.org> 
+ * Authors: Derk-Jan Hartman <hartman at videolan.org>
+ *          Jérôme Decoodt <djc at videolan.org>
  *
  * This program is free software; you can redistribute it and/or modify
  * it under the terms of the GNU General Public License as published by
 #define OFFSET_RIGHT 20
 #define OFFSET_BETWEEN 2
 
+#define LEFTMARGIN  18
+#define RIGHTMARGIN 18
+
+#define UPWARDS_WHITE_ARROW                 "\xE2\x87\xA7" 
+#define OPTION_KEY                          "\xE2\x8C\xA5"
+#define UP_ARROWHEAD                        "\xE2\x8C\x83"
+#define PLACE_OF_INTEREST_SIGN              "\xE2\x8C\x98"
+
+#define MACOS_VERSION 3
+
+#define POPULATE_A_KEY( o_menu, string, value )                             \
+{                                                                           \
+    NSMenuItem *o_mi;                                                       \
+/*  Normal */                                                               \
+    o_mi = [[NSMenuItem alloc] initWithTitle:string action:nil keyEquivalent:@""]; \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        0];                                                                 \
+    [o_mi setAlternate: NO];                                                \
+    [o_mi setTag:                                                           \
+        ( value )];                                                         \
+    [o_menu addItem: o_mi];                                                 \
+if( MACOS_VERSION >= 3 )                                                    \
+{                                                                           \
+/*  Ctrl */                                                                 \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD                                                    \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSControlKeyMask];                                                  \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        KEY_MODIFIER_CTRL | ( value )];                                     \
+    [o_menu addItem: o_mi];                                                 \
+/* Ctrl+Alt */                                                              \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD OPTION_KEY                                         \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSControlKeyMask | NSAlternateKeyMask];                             \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_CTRL | KEY_MODIFIER_ALT) | ( value )];                \
+    [o_menu addItem: o_mi];                                                 \
+/* Ctrl+Shift */                                                            \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD UPWARDS_WHITE_ARROW                                \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+       NSControlKeyMask | NSShiftKeyMask];                                  \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_CTRL | KEY_MODIFIER_SHIFT) | ( value )];              \
+    [o_menu addItem: o_mi];                                                 \
+/* Ctrl+Apple */                                                            \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD PLACE_OF_INTEREST_SIGN                             \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSControlKeyMask | NSCommandKeyMask];                               \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_CTRL | KEY_MODIFIER_COMMAND) | ( value )];            \
+    [o_menu addItem: o_mi];                                                 \
+/* Ctrl+Alt+Shift */                                                        \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD OPTION_KEY UPWARDS_WHITE_ARROW                     \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSControlKeyMask | NSAlternateKeyMask | NSShiftKeyMask];            \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_CTRL | KEY_MODIFIER_ALT | KEY_MODIFIER_SHIFT) | ( value )]; \
+    [o_menu addItem: o_mi];                                                 \
+/* Ctrl+Alt+Apple */                                                        \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD OPTION_KEY PLACE_OF_INTEREST_SIGN                  \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSControlKeyMask | NSAlternateKeyMask | NSCommandKeyMask];          \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_CTRL | KEY_MODIFIER_ALT | KEY_MODIFIER_COMMAND) | ( value )]; \
+    [o_menu addItem: o_mi];                                                 \
+/* Ctrl+Shift+Apple */                                                      \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD UPWARDS_WHITE_ARROW PLACE_OF_INTEREST_SIGN         \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSControlKeyMask | NSShiftKeyMask | NSCommandKeyMask];              \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_CTRL | KEY_MODIFIER_SHIFT | KEY_MODIFIER_COMMAND) | ( value )]; \
+    [o_menu addItem: o_mi];                                                 \
+/* Ctrl+Alt+Shift+Apple */                                                  \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UP_ARROWHEAD OPTION_KEY UPWARDS_WHITE_ARROW PLACE_OF_INTEREST_SIGN  \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSControlKeyMask | NSAlternateKeyMask | NSShiftKeyMask | NSCommandKeyMask]; \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_CTRL | KEY_MODIFIER_ALT | KEY_MODIFIER_SHIFT | KEY_MODIFIER_COMMAND) | ( value )]; \
+    [o_menu addItem: o_mi];                                                 \
+/* Alt */                                                                   \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            OPTION_KEY                                                      \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSAlternateKeyMask];                                                \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        KEY_MODIFIER_ALT | ( value )];                                      \
+    [o_menu addItem: o_mi];                                                 \
+/* Alt+Shift */                                                             \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            OPTION_KEY UPWARDS_WHITE_ARROW                                  \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSAlternateKeyMask | NSShiftKeyMask];                               \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_ALT | KEY_MODIFIER_SHIFT) | ( value )];               \
+    [o_menu addItem: o_mi];                                                 \
+/* Alt+Apple */                                                             \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            OPTION_KEY PLACE_OF_INTEREST_SIGN                               \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSAlternateKeyMask | NSCommandKeyMask];                             \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_ALT | KEY_MODIFIER_COMMAND) | ( value )];             \
+    [o_menu addItem: o_mi];                                                 \
+/* Alt+Shift+Apple */                                                       \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            OPTION_KEY UPWARDS_WHITE_ARROW PLACE_OF_INTEREST_SIGN           \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSAlternateKeyMask | NSShiftKeyMask | NSCommandKeyMask];            \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_ALT | KEY_MODIFIER_SHIFT | KEY_MODIFIER_COMMAND) | ( value )]; \
+    [o_menu addItem: o_mi];                                                 \
+/* Shift */                                                                 \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UPWARDS_WHITE_ARROW                                             \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSShiftKeyMask];                                                    \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        KEY_MODIFIER_SHIFT | ( value )];                                    \
+    [o_menu addItem: o_mi];                                                 \
+/* Shift+Apple */                                                           \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+            UPWARDS_WHITE_ARROW PLACE_OF_INTEREST_SIGN                      \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSShiftKeyMask | NSCommandKeyMask];                                 \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        (KEY_MODIFIER_SHIFT | KEY_MODIFIER_COMMAND) | ( value )];           \
+    [o_menu addItem: o_mi];                                                 \
+/* Apple */                                                                 \
+    o_mi = [[NSMenuItem alloc] initWithTitle:                               \
+        [[NSString stringWithUTF8String:                                    \
+        PLACE_OF_INTEREST_SIGN                                              \
+        ] stringByAppendingString: string]                                  \
+        action:nil keyEquivalent:@""];                                      \
+    [o_mi setKeyEquivalentModifierMask:                                     \
+        NSCommandKeyMask];                                                  \
+    [o_mi setAlternate: YES];                                               \
+    [o_mi setTag:                                                           \
+        KEY_MODIFIER_COMMAND | ( value )];                                  \
+    [o_menu addItem: o_mi];                                                 \
+}                                                                           \
+}
+
+#define ADD_LABEL( o_label, superFrame, x_offset, my_y_offset, label )    \
+{                                                                         \
+    NSRect s_rc = superFrame;                                             \
+    s_rc.size.height = 17;                                                \
+    s_rc.origin.x = x_offset - 3;                                         \
+    s_rc.origin.y = superFrame.size.height - 17 + my_y_offset;            \
+    o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];         \
+    [o_label setDrawsBackground: NO];                                     \
+    [o_label setBordered: NO];                                            \
+    [o_label setEditable: NO];                                            \
+    [o_label setSelectable: NO];                                          \
+    [o_label setStringValue: label];                                      \
+    [o_label setFont:[NSFont systemFontOfSize:0]];                        \
+    [o_label sizeToFit];                                                  \
+}
+
+#define ADD_TEXTFIELD( o_textfield, superFrame, x_offset, my_y_offset, my_width, tooltip, init_value ) \
+{                                                                               \
+    NSRect s_rc = superFrame;                                                   \
+    s_rc.origin.x = x_offset;                                                   \
+    s_rc.origin.y = my_y_offset;                                                \
+    s_rc.size.height = 22;                                                      \
+    s_rc.size.width = my_width;                                                 \
+    o_textfield = [[[NSTextField alloc] initWithFrame: s_rc] retain];           \
+    [o_textfield setFont:[NSFont systemFontOfSize:0]];                          \
+    [o_textfield setToolTip: tooltip];                                          \
+    [o_textfield setStringValue: init_value];                                   \
+}
+
+#define ADD_COMBO( o_combo, superFrame, x_offset, my_y_offset, x2_offset, tooltip ) \
+{                                                                               \
+    NSRect s_rc = superFrame;                                                   \
+    s_rc.origin.x = x_offset + 2;                                               \
+    s_rc.origin.y = my_y_offset;                                                \
+    s_rc.size.height = 26;                                                      \
+    s_rc.size.width = superFrame.size.width + 2 - s_rc.origin.x - (x2_offset);  \
+    o_combo = [[[NSComboBox alloc] initWithFrame: s_rc] retain];                \
+    [o_combo setFont:[NSFont systemFontOfSize:0]];                              \
+    [o_combo setToolTip: tooltip];                                              \
+    [o_combo setUsesDataSource:TRUE];                                           \
+    [o_combo setDataSource:self];                                               \
+    [o_combo setNumberOfVisibleItems:10];                                       \
+    [o_combo setCompletes:YES];                                                 \
+}
+
+#define ADD_RIGHT_BUTTON( o_button, superFrame, x_offset, my_y_offset, tooltip, title )   \
+{                                                                                      \
+    NSRect s_rc = superFrame;                                                          \
+    o_button = [[[NSButton alloc] initWithFrame: s_rc] retain];                        \
+    [o_button setButtonType: NSMomentaryPushInButton];                                 \
+    [o_button setBezelStyle: NSRoundedBezelStyle];                                     \
+    [o_button setTitle: title];                                                        \
+    [o_button setFont:[NSFont systemFontOfSize:0]];                                    \
+    [o_button sizeToFit];                                                              \
+    s_rc = [o_button frame];                                                           \
+    s_rc.origin.x = superFrame.size.width - [o_button frame].size.width - 6;           \
+    s_rc.origin.y = my_y_offset - 6;                                                   \
+    s_rc.size.width += 12;                                                             \
+    [o_button setFrame: s_rc];                                                         \
+    [o_button setToolTip: tooltip];                                                    \
+    [o_button setTarget: self];                                                        \
+    [o_button setAction: @selector(openFileDialog:)];                                  \
+}
+
+#define ADD_POPUP( o_popup, superFrame, x_offset, my_y_offset, x2_offset, tooltip ) \
+{                                                                               \
+    NSRect s_rc = superFrame;                                                   \
+    s_rc.origin.x = x_offset - 1;                                               \
+    s_rc.origin.y = my_y_offset;                                                \
+    s_rc.size.height = 26;                                                      \
+    s_rc.size.width = superFrame.size.width + 2 - s_rc.origin.x - (x2_offset);  \
+    o_popup = [[[NSPopUpButton alloc] initWithFrame: s_rc] retain];             \
+    [o_popup setFont:[NSFont systemFontOfSize:0]];                              \
+    [o_popup setToolTip: tooltip];                                              \
+}
+
+#define ADD_STEPPER( o_stepper, superFrame, x_offset, my_y_offset, tooltip, lower, higher ) \
+{                                                                               \
+    NSRect s_rc = superFrame;                                                   \
+    s_rc.origin.x = x_offset;                                                   \
+    s_rc.origin.y = my_y_offset;                                                \
+    s_rc.size.height = 23;                                                      \
+    s_rc.size.width = 23;                                                       \
+    o_stepper = [[[NSStepper alloc] initWithFrame: s_rc] retain];               \
+    [o_stepper setFont:[NSFont systemFontOfSize:0]];                            \
+    [o_stepper setToolTip: tooltip];                                            \
+    [o_stepper setMaxValue: higher];                                            \
+    [o_stepper setMinValue: lower];                                             \
+    [o_stepper setTarget: self];                                                \
+    [o_stepper setAction: @selector(stepperChanged:)];                          \
+    [o_stepper sendActionOn:NSLeftMouseUpMask|NSLeftMouseDownMask|NSLeftMouseDraggedMask]; \
+}
+
+#define ADD_SLIDER( o_slider, superFrame, x_offset, my_y_offset, my_width, tooltip, lower, higher ) \
+{                                                                               \
+    NSRect s_rc = superFrame;                                                   \
+    s_rc.origin.x = x_offset;                                                   \
+    s_rc.origin.y = my_y_offset;                                                \
+    s_rc.size.height = 21;                                                      \
+    s_rc.size.width = my_width;                                                 \
+    o_slider = [[[NSSlider alloc] initWithFrame: s_rc] retain];                 \
+    [o_slider setFont:[NSFont systemFontOfSize:0]];                             \
+    [o_slider setToolTip: tooltip];                                             \
+    [o_slider setMaxValue: higher];                                             \
+    [o_slider setMinValue: lower];                                              \
+}
+
+#define ADD_CHECKBOX( o_checkbox, superFrame, x_offset, my_y_offset, label, tooltip, init_value, position )    \
+{                                                                         \
+    NSRect s_rc = superFrame;                                             \
+    s_rc.size.height = 18;                                                \
+    s_rc.origin.x = x_offset - 2;                                         \
+    s_rc.origin.y = superFrame.size.height - 18 + my_y_offset;            \
+    o_checkbox = [[[NSButton alloc] initWithFrame: s_rc] retain];         \
+    [o_checkbox setFont:[NSFont systemFontOfSize:0]];                     \
+    [o_checkbox setButtonType: NSSwitchButton];                           \
+    [o_checkbox setImagePosition: position];                              \
+    [o_checkbox setIntValue: init_value];                                 \
+    [o_checkbox setTitle: label];                                         \
+    [o_checkbox setToolTip: tooltip];                                     \
+    [o_checkbox sizeToFit];                                               \
+}
+
+#define ADD_CHECKBOX( o_checkbox, superFrame, x_offset, my_y_offset, label, tooltip, init_value, position )    \
+{                                                                         \
+    NSRect s_rc = superFrame;                                             \
+    s_rc.size.height = 18;                                                \
+    s_rc.origin.x = x_offset - 2;                                         \
+    s_rc.origin.y = superFrame.size.height - 18 + my_y_offset;            \
+    o_checkbox = [[[NSButton alloc] initWithFrame: s_rc] retain];         \
+    [o_checkbox setFont:[NSFont systemFontOfSize:0]];                     \
+    [o_checkbox setButtonType: NSSwitchButton];                           \
+    [o_checkbox setImagePosition: position];                              \
+    [o_checkbox setIntValue: init_value];                                 \
+    [o_checkbox setTitle: label];                                         \
+    [o_checkbox setToolTip: tooltip];                                     \
+    [o_checkbox sizeToFit];                                               \
+}
+
 @implementation VLCConfigControl
 
 - (id)initWithFrame: (NSRect)frame
 {
     return [self initWithFrame: frame
-                    item: nil
-                    withObject: nil];
+                    item: nil];
 }
 
 - (id)initWithFrame: (NSRect)frame
         item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
 {
     self = [super initWithFrame: frame];
 
     if( self != nil )
     {
-        p_this = _p_this;
         p_item = _p_item;
-        o_label = NULL;
         psz_name = strdup( p_item->psz_name );
+        o_label = NULL;
         i_type = p_item->i_type;
         b_advanced = p_item->b_advanced;
         [self setAutoresizingMask:NSViewWidthSizable | NSViewMinYMargin ];
     [super dealloc];
 }
 
++ (int)calcVerticalMargin: (int)i_curItem lastItem: (int)i_lastItem
+{
+    int i_margin;
+    switch( i_curItem )
+    {
+    case CONFIG_ITEM_STRING:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 4;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 8;
+            break;
+        default:
+            i_margin = 20;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_STRING_LIST:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 4;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 8;
+            break;
+        default:
+            i_margin = 20;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_FILE:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 13;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 10;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 9;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 9;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 10;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 10;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 10;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 9;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 11;
+            break;
+        default:
+            i_margin = 23;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_MODULE:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 9;
+            break;
+        default:
+            i_margin = 20;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_INTEGER:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 4;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 8;
+            break;
+        default:
+            i_margin = 20;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_RANGED_INTEGER:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 4;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 8;
+            break;
+        default:
+            i_margin = 20;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_BOOL:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 4;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 2;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 3;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 3;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 3;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 2;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 6;
+            break;
+        default:
+            i_margin = 18;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_KEY_BEFORE_10_3:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 4;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 2;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 3;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 3;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 10;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 6;
+            break;
+        default:
+            i_margin = 18;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_KEY_AFTER_10_3:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 8;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 10;
+            break;
+        default:
+            i_margin = 20;
+            break;
+        }
+        break;
+    case CONFIG_ITEM_MODULE_LIST:
+        switch( i_lastItem )
+        {
+        case CONFIG_ITEM_STRING:
+            i_margin = 10;
+            break;
+        case CONFIG_ITEM_STRING_LIST:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_FILE:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_MODULE:
+            i_margin = 6;
+            break;
+        case CONFIG_ITEM_INTEGER:
+            i_margin = 9;
+            break;
+        case CONFIG_ITEM_RANGED_INTEGER:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_BOOL:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_BEFORE_10_3:
+            i_margin = 7;
+            break;
+        case CONFIG_ITEM_KEY_AFTER_10_3:
+            i_margin = 5;
+            break;
+        case CONFIG_ITEM_MODULE_LIST:
+            i_margin = 8;
+            break;
+        default:
+            i_margin = 20;
+            break;
+        }
+        break;
+    default:
+        i_margin = 20;
+        break;
+    }
+    return i_margin;
+}
 
-+ (VLCConfigControl *)newControl: (module_config_t *)_p_item withView: (NSView *)o_parent_view withObject: (vlc_object_t *)_p_this offset:(NSPoint) offset
++ (VLCConfigControl *)newControl: (module_config_t *)_p_item
+                      withView: (NSView *)o_parent_view
+                      yOffset:(int) i_yPos
+                      lastItem: (int) i_lastItem
 {
     VLCConfigControl *p_control = NULL;
-    NSRect frame = [o_parent_view frame];
-/*FIXME: Why do we need to divide by two ??? */
-    frame.origin.x=offset.x;
-    frame.origin.y=offset.y;
-    frame.size.width-=OFFSET_RIGHT;
     switch( _p_item->i_type )
     {
-#if 0
-    case CONFIG_ITEM_MODULE:
-        p_control = [[ModuleConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
-        break;
-#endif
     case CONFIG_ITEM_STRING:
         if( !_p_item->i_list )
         {
-            p_control = [[StringConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+            p_control = [[StringConfigControl alloc] initWithItem: _p_item
+                    withView: o_parent_view
+                    withVerticalOffset: i_yPos
+                    withLastItem: i_lastItem];
         }
         else
         {
-            p_control = [[StringListConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+            p_control = [[StringListConfigControl alloc] initWithItem: _p_item
+                    withView: o_parent_view
+                    withVerticalOffset: i_yPos
+                    withLastItem: i_lastItem];
         }
         break;
     case CONFIG_ITEM_FILE:
     case CONFIG_ITEM_DIRECTORY:
-        p_control = [[FileConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+        p_control = [[FileConfigControl alloc] initWithItem: _p_item
+                    withView: o_parent_view
+                    withVerticalOffset: i_yPos
+                    withLastItem: i_lastItem];
+        break;
+    case CONFIG_ITEM_MODULE:
+    case CONFIG_ITEM_MODULE_CAT:
+        p_control = [[ModuleConfigControl alloc] initWithItem: _p_item
+                    withView: o_parent_view
+                    withVerticalOffset: i_yPos
+                    withLastItem: i_lastItem];
         break;
     case CONFIG_ITEM_INTEGER:
         if( _p_item->i_list )
         {
-            p_control = [[IntegerListConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+            p_control = [[IntegerListConfigControl alloc] initWithItem: _p_item
+                        withView: o_parent_view
+                        withVerticalOffset: i_yPos
+                        withLastItem: i_lastItem];
         }
         else if( _p_item->i_min != 0 || _p_item->i_max != 0 )
         {
-            p_control = [[RangedIntegerConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+            p_control = [[RangedIntegerConfigControl alloc] initWithItem: _p_item
+                        withView: o_parent_view
+                        withVerticalOffset: i_yPos
+                        withLastItem: i_lastItem];
         }
         else
         {
-            p_control = [[IntegerConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+            p_control = [[IntegerConfigControl alloc] initWithItem: _p_item
+                        withView: o_parent_view
+                        withVerticalOffset: i_yPos
+                        withLastItem: i_lastItem];
         }
         break;
-    case CONFIG_ITEM_KEY:
-        p_control = [[KeyConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+    case CONFIG_ITEM_BOOL:
+        p_control = [[BoolConfigControl alloc] initWithItem: _p_item
+                    withView: o_parent_view
+                    withVerticalOffset: i_yPos
+                    withLastItem: i_lastItem];
         break;
-#if 0
     case CONFIG_ITEM_FLOAT:
         if( _p_item->f_min != 0 || _p_item->f_max != 0 )
         {
-            p_control = [[RangedFloatConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+            p_control = [[RangedFloatConfigControl alloc] initWithItem: _p_item
+                        withView: o_parent_view
+                        withVerticalOffset: i_yPos
+                        withLastItem: i_lastItem];
         }
         else
         {
-            p_control = [[FloatConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+            p_control = [[FloatConfigControl alloc] initWithItem: _p_item
+                        withView: o_parent_view
+                        withVerticalOffset: i_yPos
+                        withLastItem: i_lastItem];
         }
         break;
-
-    case CONFIG_ITEM_BOOL:
-        p_control = [[BoolConfigControl alloc] initWithFrame: frame item: _p_item withObject: _p_this ];
+    case CONFIG_ITEM_KEY:
+        if( MACOS_VERSION < 3 )
+        {
+            p_control = [[KeyConfigControlBefore103 alloc] initWithItem: _p_item
+                        withView: o_parent_view
+                        withVerticalOffset: i_yPos
+                        withLastItem: i_lastItem];
+        }
+        else
+        {
+            p_control = [[KeyConfigControlAfter103 alloc] initWithItem: _p_item
+                        withView: o_parent_view
+                        withVerticalOffset: i_yPos
+                        withLastItem: i_lastItem];
+        }
+        break;
+    case CONFIG_ITEM_MODULE_LIST:
+    case CONFIG_ITEM_MODULE_LIST_CAT:
+        p_control = [[ModuleListConfigControl alloc] initWithItem: _p_item
+                    withView: o_parent_view
+                    withVerticalOffset: i_yPos
+                    withLastItem: i_lastItem];
         break;
-#endif        
     default:
         break;
     }
     return NULL;
 }
 
-@end
-
-@implementation KeyConfigControl
-
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
-{
-    frame.size.height = 80;
-    if( ( self = [super initWithFrame: frame item: _p_item
-                withObject: _p_this] ) )
-    {
-        NSRect s_rc = frame;
-        unsigned int i;
-
-        o_matrix = [[[NSMatrix alloc] initWithFrame: s_rc mode: NSHighlightModeMatrix cellClass: [NSButtonCell class] numberOfRows:2 numberOfColumns:2] retain];
-        NSArray *o_cells = [o_matrix cells];
-        for( i = 0; i < [o_cells count]; i++ )
-        {
-            NSButtonCell *o_current_cell = [o_cells objectAtIndex:i];
-            [o_current_cell setButtonType: NSSwitchButton];
-            [o_current_cell setControlSize: NSSmallControlSize];
-            [o_matrix setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext] toWidth: PREFS_WRAP] forCell: o_current_cell];
-
-            switch( i )
-            {
-                case 0:
-                    [o_current_cell setTitle:_NS("Command")];
-                    [o_current_cell setState: p_item->i_value & KEY_MODIFIER_COMMAND];
-                    break;
-                case 1:
-                    [o_current_cell setTitle:_NS("Control")];
-                    [o_current_cell setState: p_item->i_value & KEY_MODIFIER_CTRL];
-                    break;
-                case 2:
-                    [o_current_cell setTitle:_NS("Option/Alt")];
-                    [o_current_cell setState: p_item->i_value & KEY_MODIFIER_ALT];
-                    break;
-                case 3:
-                    [o_current_cell setTitle:_NS("Shift")];
-                    [o_current_cell setState: p_item->i_value & KEY_MODIFIER_SHIFT];
-                    break;
-            }
-        }
-        [o_matrix sizeToCells];
-        [o_matrix setAutoresizingMask:NSViewMaxXMargin ];
-        [self addSubview: o_matrix];
-
-        /* add the combo box */
-        s_rc.origin.x += [o_matrix frame].size.width + OFFSET_BETWEEN;
-        s_rc.size.height = 22;
-        s_rc.size.width = 100;
-
-        o_combo = [[[NSComboBox alloc] initWithFrame: s_rc] retain];
-        [o_combo setAutoresizingMask:NSViewMaxXMargin ];
-        [o_combo setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext] toWidth: PREFS_WRAP]];
-        
-        for( i = 0; i < sizeof(vlc_keys) / sizeof(key_descriptor_t); i++ )
-        {
-            
-            if( vlc_keys[i].psz_key_string && *vlc_keys[i].psz_key_string )
-            [o_combo addItemWithObjectValue: [[VLCMain sharedInstance] localizedString:vlc_keys[i].psz_key_string]];
-        }
-        
-        [o_combo setStringValue: [[VLCMain sharedInstance] localizedString:KeyToString(( ((unsigned int)p_item->i_value) & ~KEY_MODIFIER ))]];
-        [self addSubview: o_combo];
-        
-        /* add the label */
-        s_rc.origin.y += 50;
-        
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
-        if ( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
-        [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-    }
-    return self;
-}
-
-- (void)dealloc
-{
-    [o_matrix release];
-    [o_combo release];
-    [super dealloc];
-}
-
-- (int)getIntValue
+- (void)applyChanges
 {
-    unsigned int i, i_new_key = 0;
-    NSButtonCell *o_current_cell;
-    NSArray *o_cells = [o_matrix cells];
-
-    for( i = 0; i < [o_cells count]; i++ )
+    vlc_value_t val;
+    switch( p_item->i_type )
     {
-        o_current_cell = [o_cells objectAtIndex:i];
-        if( [[o_current_cell title] isEqualToString:_NS("Command")] && 
-            [o_current_cell state] == NSOnState )
-        {
-            i_new_key |= KEY_MODIFIER_COMMAND;
-        }
-        if( [[o_current_cell title] isEqualToString:_NS("Control")] && 
-            [o_current_cell state] == NSOnState )
-        {
-            i_new_key |= KEY_MODIFIER_CTRL;
-        }
-        if( [[o_current_cell title] isEqualToString:_NS("Option/Alt")] && 
-            [o_current_cell state] == NSOnState )
-        {
-            i_new_key |= KEY_MODIFIER_ALT;
-        }
-        if( [[o_current_cell title] isEqualToString:_NS("Shift")] && 
-            [o_current_cell state] == NSOnState )
-        {
-            i_new_key |= KEY_MODIFIER_SHIFT;
-        }
+    case CONFIG_ITEM_STRING:
+    case CONFIG_ITEM_FILE:
+    case CONFIG_ITEM_DIRECTORY:
+    case CONFIG_ITEM_MODULE:
+    case CONFIG_ITEM_MODULE_LIST:
+    case CONFIG_ITEM_MODULE_LIST_CAT:
+fprintf( stderr, "Applying %s to %s\n" , [self stringValue], psz_name );
+        config_PutPsz( VLCIntf, psz_name, [self stringValue] );
+        break;
+    case CONFIG_ITEM_KEY:
+        /* So you don't need to restart to have the changes take effect */
+fprintf( stderr, "Applying %d to %s\n" , [self intValue], psz_name );
+        val.i_int = [self intValue];
+        var_Set( VLCIntf->p_vlc, psz_name, val );
+    case CONFIG_ITEM_INTEGER:
+    case CONFIG_ITEM_BOOL:
+fprintf( stderr, "Applying %d to %s\n" , [self intValue], psz_name );
+        config_PutInt( VLCIntf, psz_name, [self intValue] );
+        break;
+    case CONFIG_ITEM_FLOAT:
+fprintf( stderr, "Applying %f to %s\n" , [self floatValue], psz_name );
+        config_PutFloat( VLCIntf, psz_name, [self floatValue] );
+        break;
     }
-    i_new_key |= StringToKey([[o_combo stringValue] cString]);
-    return i_new_key;
 }
 
-
 @end
-#if 0
-
-@implementation ModuleConfigControl
-
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)p_item
-        withObject: (vlc_object_t *)_p_this
-{
-    frame.size.height = 20;
-    if( self = [super initWithFrame: frame item: p_item
-                withObject: _p_this] )
-    {
-        vlc_list_t *p_list;
-        module_t *p_parser;
-        NSRect s_rc = frame;
-        int i_index;
-
-        /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
-        if ( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
-        [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-        
-        /* build the popup */
-        s_rc.origin.x = s_rc.size.width - 200 - OFFSET_RIGHT;
-        s_rc.size.width = 200;
-        
-        o_popup = [[[NSPopUpButton alloc] initWithFrame: s_rc] retain];
-        [self addSubview: o_popup];
-        [o_popup setAutoresizingMask:NSViewMinXMargin ];
-
-        [o_popup setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
-        [o_popup addItemWithTitle: _NS("Default")];
-        [[o_popup lastItem] setTag: -1];
-        [o_popup selectItem: [o_popup lastItem]];
-        
-        /* build a list of available modules */
-        p_list = vlc_list_find( p_this, VLC_OBJECT_MODULE, FIND_ANYWHERE );
-        for( i_index = 0; i_index < p_list->i_count; i_index++ )
-        {
-            p_parser = (module_t *)p_list->p_values[i_index].p_object ;
-
-            if( !strcmp( p_parser->psz_capability,
-                        p_item->psz_type ) )
-            {
-                NSString *o_description = [NSApp
-                    localizedString: p_parser->psz_longname];
-                [o_popup addItemWithTitle: o_description];
-
-                if( p_item->psz_value &&
-                    !strcmp( p_item->psz_value, p_parser->psz_object_name ) )
-                {
-                    [o_popup selectItem:[o_popup lastItem]];
-                }
-            }
-        }
-        vlc_list_release( p_list );
-    }
-    return self;
-}
-
-- (void)dealloc
-{
-    [o_popup release];
-    [super dealloc];
-}
-
-- (char *)stringValue
-{
-    NSString *newval = [o_popup stringValue];
-    char *returnval;
-    int i_index;
-    vlc_list_t *p_list;
-    module_t *p_parser;
-    module_config_t *p_item;
-
-    p_item = config_FindConfig( p_this, psz_name );
-    p_list = vlc_list_find( p_this, VLC_OBJECT_MODULE, FIND_ANYWHERE );
-    for( i_index = 0; i_index < p_list->i_count; i_index++ )
-    {
-        p_parser = (module_t *)p_list->p_values[i_index].p_object ;
 
-        if( !strcmp( p_parser->psz_capability,
-                    p_item->psz_type ) )
-        {
-            NSString *o_description = [NSApp
-                localizedString: p_parser->psz_longname];
-            
-            if( [newval isEqualToString: o_description] )
-            {
-                returnval = strdup(p_parser->psz_object_name);
-                break;
-            }
-        }
-    }
-    vlc_list_release( p_list );
-    return returnval;
-}
 
-@end
-#endif
 @implementation StringConfigControl
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 22;
-    if( ( self = [super initWithFrame: frame item: _p_item
-                withObject: _p_this] ) )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_textfieldString, *o_textfieldTooltip;
+    mainFrame.size.height = 22;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
-        s_rc.size.height = 17;
-        s_rc.origin.x = 0;
-        s_rc.origin.y = 3;
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -3, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-        
-        /* build the textfield */
-        s_rc.origin.x = s_rc.size.width - 200 - OFFSET_RIGHT;
-        s_rc.origin.y = 0;
-        s_rc.size.height = 22;
-        s_rc.size.width = 200;
-        
-        o_textfield = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield setAutoresizingMask:NSViewMinXMargin | NSViewWidthSizable ];
 
-        [o_textfield setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        /* build the textfield */
+        o_textfieldTooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
         if( p_item->psz_value )
-            [o_textfield setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_value]];
+            o_textfieldString = [[VLCMain sharedInstance] localizedString: p_item->psz_value];
+        else
+            o_textfieldString = [NSString stringWithString: @""];
+        ADD_TEXTFIELD( o_textfield, mainFrame, [o_label frame].size.width + 2,
+                        0, mainFrame.size.width - [o_label frame].size.width - 2,
+                        o_textfieldTooltip, o_textfieldString )
+        [o_textfield setAutoresizingMask:NSViewWidthSizable ];
         [self addSubview: o_textfield];
     }
     return self;
 
 - (void)dealloc
 {
+    [o_label release];
     [o_textfield release];
     [super dealloc];
 }
 
 - (char *)stringValue
 {
-    return strdup( [NSApp delocalizeString:[o_textfield stringValue]] );
+    return strdup( [[VLCMain sharedInstance] delocalizeString:[o_textfield stringValue]] );
 }
 
 @end
 
 @implementation StringListConfigControl
 
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 24;
-    if( ( self = [super initWithFrame: frame item: _p_item
-                withObject: _p_this] ) )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_textfieldTooltip;
+    mainFrame.size.height = 22;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN + 1;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
         int i_index;
-        s_rc.size.height = 17;
-        s_rc.origin.x = 0;
-        s_rc.origin.y = 5;
-
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -3, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-        
-        /* build the textfield */
-        s_rc.origin.x = s_rc.size.width - 200 - OFFSET_RIGHT;
-        s_rc.origin.y = 0;
-        s_rc.size.height = 26;
-        s_rc.size.width = 200;
-        
-        o_combo = [[[NSComboBox alloc] initWithFrame: s_rc] retain];
-        [o_combo setAutoresizingMask:NSViewMinXMargin | NSViewWidthSizable ];
 
-        [o_combo setUsesDataSource:TRUE];
-        [o_combo setDataSource:self];
-        [o_combo setNumberOfVisibleItems:10];
-        [o_combo setCompletes:YES];
+        /* build the textfield */
+        o_textfieldTooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_COMBO( o_combo, mainFrame, [o_label frame].size.width,
+            -2, 0, o_textfieldTooltip )
+        [o_combo setAutoresizingMask:NSViewWidthSizable ];
         for( i_index = 0; i_index < p_item->i_list; i_index++ )
         {
             if( p_item->psz_value && !strcmp( p_item->psz_value, p_item->ppsz_list[i_index] ) )
                 [o_combo selectItemAtIndex: i_index];
             }
         }
-
-        [o_combo setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
         [self addSubview: o_combo];
     }
     return self;
     if( [o_combo indexOfSelectedItem] >= 0 )
         return strdup( p_item->ppsz_list[[o_combo indexOfSelectedItem]] );
     else
-        return strdup( [NSApp delocalizeString: [o_combo stringValue]] );
+        return strdup( [[VLCMain sharedInstance] delocalizeString: [o_combo stringValue]] );
 }
 
 @end
 }
 
 @end
+
 @implementation FileConfigControl
 
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 53;
-    if( ( self = [super initWithFrame: frame item: _p_item
-                withObject: _p_this] ) )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_buttonTooltip, *o_textfieldString, *o_textfieldTooltip;
+    mainFrame.size.height = 46;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
-        s_rc.size.height = 17;
-        s_rc.origin.x = 0;
-        s_rc.origin.y = 36;
-
         /* is it a directory */
         b_directory = ( [self getType] == CONFIG_ITEM_DIRECTORY ) ? YES : NO;
 
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, 3, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-        
+
         /* build the button */
-        s_rc.origin.y = 0;
-        s_rc.size.height = 32;
-
-        o_button = [[[NSButton alloc] initWithFrame: s_rc] retain];
-        [o_button setButtonType: NSMomentaryPushInButton];
-        [o_button setBezelStyle: NSRoundedBezelStyle];
-        [o_button setTitle: _NS("Browse...")];
-/*TODO: enlarge a bit the button...*/
-        [o_button sizeToFit];
-        [o_button setAutoresizingMask:NSViewMinXMargin];
-        [o_button setFrameOrigin: NSMakePoint( s_rc.size.width - 
-                    [o_button frame].size.width - OFFSET_RIGHT / 2, s_rc.origin.y)];
-        [o_button setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
-
-        [o_button setTarget: self];
-        [o_button setAction: @selector(openFileDialog:)];
-
-        s_rc.origin.x = 15;
-        s_rc.origin.y = 6;
-        s_rc.size.height = 22;
-        s_rc.size.width = s_rc.size.width - OFFSET_BETWEEN - [o_button frame].size.width - OFFSET_RIGHT / 2 - s_rc.origin.x;
-        
-        o_textfield = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        
-        if( p_item->psz_value )
-            [o_textfield setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_value]];
-        [o_textfield setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        o_buttonTooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_RIGHT_BUTTON( o_button, mainFrame, 0, 0, o_buttonTooltip, _NS("Browse...") )
+        [o_button setAutoresizingMask:NSViewMinXMargin ];
+        [self addSubview: o_button];
 
-        [o_textfield setAutoresizingMask:NSViewWidthSizable];
-               
+        /* build the textfield */
+        o_textfieldTooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        if( p_item->psz_value )
+            o_textfieldString = [[VLCMain sharedInstance] localizedString: p_item->psz_value];
+        else
+            o_textfieldString = [NSString stringWithString: @""];
+        ADD_TEXTFIELD( o_textfield, mainFrame, 12,
+                        2, mainFrame.size.width - 8 - [o_button frame].size.width,
+                        o_textfieldTooltip, o_textfieldString )
+        [o_textfield setAutoresizingMask:NSViewWidthSizable ];
         [self addSubview: o_textfield];
-        [self addSubview: o_button];
     }
     return self;
 }
 
 - (char *)stringValue
 {
-    return strdup( [[o_textfield stringValue] fileSystemRepresentation] );
+    if( [[o_textfield stringValue] length] != 0)
+        return strdup( [[o_textfield stringValue] fileSystemRepresentation] );
+    else
+        return NULL;
 }
 
 @end
 
-@implementation IntegerConfigControl
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
+@implementation ModuleConfigControl
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 25;
-    if( ( self = [super initWithFrame: frame item: _p_item
-                withObject: _p_this] ) )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_popupTooltip;
+    mainFrame.size.height = 22;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN + 1;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
-        s_rc.size.height = 17;
-        s_rc.origin.x = 0;
-        s_rc.origin.y = 6;
+        int i_index;
+        vlc_list_t *p_list;
+        module_t *p_parser;
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -1, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-        
-        /* build the stepper */
-        s_rc.origin.x = s_rc.size.width - 16 - OFFSET_RIGHT;
-        s_rc.origin.y = 0;
-        s_rc.size.height = 21;
-        
-        o_stepper = [[[NSStepper alloc] initWithFrame: s_rc] retain];
-        [o_stepper sizeToFit];
-        [o_stepper setAutoresizingMask:NSViewMinXMargin];
-
-        [o_stepper setMaxValue: 1600];
-        [o_stepper setMinValue: -1600];
-        [o_stepper setIntValue: p_item->i_value];
-        [o_stepper setTarget: self];
-        [o_stepper setAction: @selector(stepperChanged:)];
-        [o_stepper sendActionOn:NSLeftMouseUpMask|NSLeftMouseDownMask|NSLeftMouseDraggedMask];
-        [o_stepper setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
-        [self addSubview: o_stepper];
-    
-        /* build the textfield */
-        s_rc.origin.x = s_rc.size.width - 42 - OFFSET_BETWEEN - 19 - OFFSET_RIGHT;
-        s_rc.origin.y = 3;
-        s_rc.size.width = 42;
-        s_rc.size.height = 22;
-        
-        o_textfield = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield setAutoresizingMask:NSViewMinXMargin | NSViewWidthSizable ];
 
+        /* build the popup */
+        o_popupTooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_POPUP( o_popup, mainFrame, [o_label frame].size.width,
+            -2, 0, o_popupTooltip )
+        [o_popup setAutoresizingMask:NSViewWidthSizable ];
+        [o_popup addItemWithTitle: _NS("Default")];
+        [[o_popup lastItem] setTag: -1];
+        [o_popup selectItem: [o_popup lastItem]];
+        
+        /* build a list of available modules */
+        p_list = vlc_list_find( VLCIntf, VLC_OBJECT_MODULE, FIND_ANYWHERE );
+        for( i_index = 0; i_index < p_list->i_count; i_index++ )
+        {
+            p_parser = (module_t *)p_list->p_values[i_index].p_object ;
+            if( p_item->i_type == CONFIG_ITEM_MODULE )
+            {
+                if( !strcmp( p_parser->psz_capability,
+                            p_item->psz_type ) )
+                {
+                    NSString *o_description = [[VLCMain sharedInstance]
+                        localizedString: p_parser->psz_longname];
+                    [o_popup addItemWithTitle: o_description];
+
+                    if( p_item->psz_value &&
+                        !strcmp( p_item->psz_value, p_parser->psz_object_name ) )
+                    {
+                        [o_popup selectItem:[o_popup lastItem]];
+                    }
+                }
+            }
+            else
+            {
+                module_config_t *p_config;
+                if( !strcmp( p_parser->psz_object_name, "main" ) )
+                      continue;
+
+                p_config = p_parser->p_config;
+                if( p_config ) do
+                {
+                    /* Hack: required subcategory is stored in i_min */
+                    if( p_config->i_type == CONFIG_SUBCATEGORY &&
+                        p_config->i_value == p_item->i_min )
+                    {
+                        NSString *o_description = [[VLCMain sharedInstance]
+                            localizedString: p_parser->psz_longname];
+                        [o_popup addItemWithTitle: o_description];
+                        
+                        if( p_item->psz_value && !strcmp(p_item->psz_value,
+                                                p_parser->psz_object_name) )
+                            [o_popup selectItem:[o_popup lastItem]];
+                    }
+                } while( p_config->i_type != CONFIG_HINT_END && p_config++ );
+            }
+        }
+        vlc_list_release( p_list );
+        [self addSubview: o_popup];
+    }
+    return self;
+}
+
+- (void)dealloc
+{
+    [o_popup release];
+    [super dealloc];
+}
+
+- (char *)stringValue
+{
+    NSString *newval = [o_popup titleOfSelectedItem];
+    char *returnval = NULL;
+    int i_index;
+    vlc_list_t *p_list;
+    module_t *p_parser;
+
+    p_list = vlc_list_find( VLCIntf, VLC_OBJECT_MODULE, FIND_ANYWHERE );
+    for( i_index = 0; i_index < p_list->i_count; i_index++ )
+    {
+        p_parser = (module_t *)p_list->p_values[i_index].p_object ;
+        if( p_item->i_type == CONFIG_ITEM_MODULE )
+        {
+            if( !strcmp( p_parser->psz_capability,
+                    p_item->psz_type ) )
+            {
+                NSString *o_description = [[VLCMain sharedInstance]
+                    localizedString: p_parser->psz_longname];
+                if( [newval isEqualToString: o_description] )
+                {
+                    returnval = strdup(p_parser->psz_object_name);
+                    break;
+                }
+            }
+        }
+        else
+        {
+            module_config_t *p_config;
+            if( !strcmp( p_parser->psz_object_name, "main" ) )
+                  continue;
+
+            p_config = p_parser->p_config;
+            if( p_config ) do
+            {
+                /* Hack: required subcategory is stored in i_min */
+                if( p_config->i_type == CONFIG_SUBCATEGORY &&
+                    p_config->i_value == p_item->i_min )
+                {
+                    NSString *o_description = [[VLCMain sharedInstance]
+                        localizedString: p_parser->psz_longname];
+                    if( [newval isEqualToString: o_description] )
+                    {
+                        returnval = strdup(p_parser->psz_object_name);
+                        break;
+                    }
+                }
+            } while( p_config->i_type != CONFIG_HINT_END && p_config++ );
+        }
+    }
+    vlc_list_release( p_list );
+    return returnval;
+}
+
+@end
+
+@implementation IntegerConfigControl
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
+{
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_tooltip, *o_textfieldString;
+    mainFrame.size.height = 23;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN + 1;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
+    {
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+
+        /* add the label */
+        if( p_item->psz_text )
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -2, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
+        [self addSubview: o_label];
+
+        /* build the stepper */
+        ADD_STEPPER( o_stepper, mainFrame, mainFrame.size.width - 19,
+            0, o_tooltip, -1600, 1600)
+        [o_stepper setIntValue: p_item->i_value];
+        [o_stepper setAutoresizingMask:NSViewMinXMargin ];
+        [self addSubview: o_stepper];
+
+        /* build the textfield */
+        if( p_item->psz_value )
+            o_textfieldString = [[VLCMain sharedInstance] localizedString: p_item->psz_value];
+        else
+            o_textfieldString = [NSString stringWithString: @""];
+        ADD_TEXTFIELD( o_textfield, mainFrame, mainFrame.size.width - 19 - 52,
+            1, 49, o_tooltip, @"" )
         [o_textfield setIntValue: p_item->i_value];
-        [o_textfield setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
         [o_textfield setDelegate: self];
         [[NSNotificationCenter defaultCenter] addObserver: self
             selector: @selector(textfieldChanged:)
             name: NSControlTextDidChangeNotification
             object: o_textfield];
+        [o_textfield setAutoresizingMask:NSViewMinXMargin ];
         [self addSubview: o_textfield];
     }
     return self;
     [o_stepper setIntValue: [o_textfield intValue]];
 }
 
-- (int)getIntValue
+- (int)intValue
 {
-    return [o_stepper intValue];
+    return [o_textfield intValue];
 }
 
 @end
 
 @implementation IntegerListConfigControl
 
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 24;
-    if( ( self = [super initWithFrame: frame item: _p_item
-                withObject: _p_this] ) )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_textfieldTooltip;
+    mainFrame.size.height = 22;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN + 1;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
         int i_index;
-        s_rc.size.height = 17;
-        s_rc.origin.x = 0;
-        s_rc.origin.y = 5;
-
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -3, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-        
-        /* build the textfield */
-        s_rc.origin.x = s_rc.size.width - 200 - OFFSET_RIGHT;
-        s_rc.origin.y = 0;
-        s_rc.size.height = 26;
-        s_rc.size.width = 200;
-        
-        o_combo = [[[NSComboBox alloc] initWithFrame: s_rc] retain];
-        [o_combo setAutoresizingMask:NSViewMinXMargin | NSViewWidthSizable ];
 
-        [o_combo setUsesDataSource:TRUE];
-        [o_combo setDataSource:self];
-        [o_combo setNumberOfVisibleItems:10];
-        [o_combo setCompletes:YES];
+        /* build the textfield */
+        o_textfieldTooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_COMBO( o_combo, mainFrame, [o_label frame].size.width,
+            -2, 0, o_textfieldTooltip )
+        [o_combo setAutoresizingMask:NSViewWidthSizable ];
         for( i_index = 0; i_index < p_item->i_list; i_index++ )
         {
             if( p_item->i_value == p_item->pi_list[i_index] )
                 [o_combo selectItemAtIndex: i_index];
             }
         }
-
-        [o_combo setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
         [self addSubview: o_combo];
     }
     return self;
 
 @implementation RangedIntegerConfigControl
 
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)_p_item
-        withObject: (vlc_object_t *)_p_this
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 51;
-    if( ( self = [super initWithFrame: frame item: _p_item
-                withObject: _p_this] ) )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_tooltip;
+    mainFrame.size.height = 50;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
-        s_rc.size.height = 17;
-        s_rc.origin.x = 0;
-        s_rc.origin.y = 32;
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -3, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
-
-        /* current value textfield */
-        s_rc.size.height = 22;
-        s_rc.size.width = 40;
-        s_rc.origin.x = [o_label frame].size.width + OFFSET_RIGHT;
-        s_rc.origin.y = 29;
-        o_textfield = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield setAutoresizingMask:NSViewMinXMargin];
 
+        /* build the textfield */
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_TEXTFIELD( o_textfield, mainFrame, [o_label frame].size.width + 2,
+            28, 49, o_tooltip, @"" )
         [o_textfield setIntValue: p_item->i_value];
-        [o_textfield setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        [o_textfield setAutoresizingMask:NSViewMaxXMargin ];
         [o_textfield setDelegate: self];
         [[NSNotificationCenter defaultCenter] addObserver: self
             selector: @selector(textfieldChanged:)
             object: o_textfield];
         [self addSubview: o_textfield];
 
-        /* build the slider */
-        /* min value textfield */
-        s_rc.origin.x = 15;
-        s_rc.origin.y = 0;
-        s_rc.size.width = 40;
-
-        o_textfield_min = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield_min setAutoresizingMask:NSViewMaxXMargin];
-        [o_textfield_min setDrawsBackground: NO];
-        [o_textfield_min setBordered: NO];
-        [o_textfield_min setEditable: NO];
-        [o_textfield_min setSelectable: NO];
+        /* build the mintextfield */
+        ADD_LABEL( o_textfield_min, mainFrame, 12, -30, @"-8888" )
         [o_textfield_min setIntValue: p_item->i_min];
-        [o_textfield_min setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        [o_textfield_min setAutoresizingMask:NSViewMaxXMargin ];
         [self addSubview: o_textfield_min];
 
-        /* max value textfield */
-        s_rc.size.width = 40;
-        s_rc.origin.x = [self bounds].size.width - OFFSET_RIGHT - 40;
-        
-        o_textfield_max = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield_max setAutoresizingMask:NSViewMinXMargin ];
-
-        [o_textfield_max setDrawsBackground: NO];
-        [o_textfield_max setBordered: NO];
-        [o_textfield_max setEditable: NO];
-        [o_textfield_max setSelectable: NO];
-        [o_textfield_max setAlignment: NSRightTextAlignment];
-
+        /* build the maxtextfield */
+        ADD_LABEL( o_textfield_max, mainFrame, mainFrame.size.width - 31, -30, @"8888" )
         [o_textfield_max setIntValue: p_item->i_max];
-        [o_textfield_max setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        [o_textfield_max setAutoresizingMask:NSViewMinXMargin ];
         [self addSubview: o_textfield_max];
-        
-        /* the slider */
-        s_rc.size.height = 21;
-        s_rc.size.width = [self bounds].size.width - OFFSET_RIGHT - 2*OFFSET_BETWEEN - 2 * 40;
-        s_rc.origin.x = 40 + OFFSET_BETWEEN;
-        s_rc.origin.y = 1;
 
-        o_slider = [[[NSSlider alloc] initWithFrame: s_rc] retain];
-        [o_slider setAutoresizingMask:NSViewWidthSizable];
-
-        [o_slider setMaxValue: p_item->i_max];
-        [o_slider setMinValue: p_item->i_min];
+        /* build the slider */
+        ADD_SLIDER( o_slider, mainFrame, [o_textfield_min frame].origin.x + [o_textfield_min frame].size.width + 6,
+            -1, mainFrame.size.width - [o_textfield_max frame].size.width - [o_textfield_max frame].size.width - 14
+            - [o_textfield_min frame].origin.x, o_tooltip, p_item->i_min, p_item->i_max )
         [o_slider setIntValue: p_item->i_value];
+        [o_slider setAutoresizingMask:NSViewWidthSizable ];
         [o_slider setTarget: self];
         [o_slider setAction: @selector(sliderChanged:)];
         [o_slider sendActionOn:NSLeftMouseUpMask|NSLeftMouseDownMask|NSLeftMouseDraggedMask];
-        [o_slider setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
         [self addSubview: o_slider];
-       
+
     }
     return self;
 }
 }
 
 @end
-#if 0
 
 @implementation FloatConfigControl
 
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)p_item
-        withObject: (vlc_object_t *)_p_this
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 20;
-    if( self = [super initWithFrame: frame item: p_item
-                withObject: _p_this] )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_tooltip, *o_textfieldString;
+    mainFrame.size.height = 23;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN + 1;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
 
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -2, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
 
-        /* build the textfield */
-        s_rc.origin.x = s_rc.size.width - 60 - OFFSET_RIGHT;
-        s_rc.size.width = 60;
-        
-        o_textfield = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield setAutoresizingMask:NSViewMinXMargin | NSViewWidthSizable ];
+        /* build the stepper */
+        ADD_STEPPER( o_stepper, mainFrame, mainFrame.size.width - 19,
+            0, o_tooltip, -1600, 1600)
+        [o_stepper setFloatValue: p_item->f_value];
+        [o_stepper setAutoresizingMask:NSViewMinXMargin ];
+        [self addSubview: o_stepper];
 
+        /* build the textfield */
+        if( p_item->psz_value )
+            o_textfieldString = [[VLCMain sharedInstance] localizedString: p_item->psz_value];
+        else
+            o_textfieldString = [NSString stringWithString: @""];
+        ADD_TEXTFIELD( o_textfield, mainFrame, mainFrame.size.width - 19 - 52,
+            1, 49, o_tooltip, @"" )
         [o_textfield setFloatValue: p_item->f_value];
-        [o_textfield setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        [o_textfield setDelegate: self];
+        [[NSNotificationCenter defaultCenter] addObserver: self
+            selector: @selector(textfieldChanged:)
+            name: NSControlTextDidChangeNotification
+            object: o_textfield];
+        [o_textfield setAutoresizingMask:NSViewMinXMargin ];
         [self addSubview: o_textfield];
     }
     return self;
 
 - (void)dealloc
 {
+    [o_stepper release];
     [o_textfield release];
     [super dealloc];
 }
 
-- (float)floatValue
+- (IBAction)stepperChanged:(id)sender
+{
+    [o_textfield setFloatValue: [o_stepper floatValue]];
+}
+
+- (void)textfieldChanged:(NSNotification *)o_notification
+{
+    [o_stepper setFloatValue: [o_textfield floatValue]];
+}
+
+- (int)floatValue
 {
-    return [o_textfield floatValue];
+    return [o_stepper floatValue];
 }
 
 @end
 
 @implementation RangedFloatConfigControl
 
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)p_item
-        withObject: (vlc_object_t *)_p_this
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 50;
-    if( self = [super initWithFrame: frame item: p_item
-                withObject: _p_this] )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_tooltip;
+    mainFrame.size.height = 50;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
-        s_rc.size.height = 20;
-        s_rc.origin.y = 30;
-    
         /* add the label */
-        o_label = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_label setDrawsBackground: NO];
-        [o_label setBordered: NO];
-        [o_label setEditable: NO];
-        [o_label setSelectable: NO];
         if( p_item->psz_text )
-            [o_label setStringValue: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-
-        [o_label sizeToFit];
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -3, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_label];
-        [o_label setAutoresizingMask:NSViewMaxXMargin ];
 
-        /* build the slider */
-        /* min value textfield */
-        s_rc.origin.y = 0;
-        s_rc.origin.x = 0;
-        s_rc.size.width = 40;
-
-        o_textfield_min = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield_min setAutoresizingMask:NSViewMaxXMargin];
-        [o_textfield_min setDrawsBackground: NO];
-        [o_textfield_min setBordered: NO];
-        [o_textfield_min setEditable: NO];
-        [o_textfield_min setSelectable: NO];
+        /* build the textfield */
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_TEXTFIELD( o_textfield, mainFrame, [o_label frame].size.width + 2,
+            28, 49, o_tooltip, @"" )
+        [o_textfield setFloatValue: p_item->f_value];
+        [o_textfield setAutoresizingMask:NSViewMaxXMargin ];
+        [o_textfield setDelegate: self];
+        [[NSNotificationCenter defaultCenter] addObserver: self
+            selector: @selector(textfieldChanged:)
+            name: NSControlTextDidChangeNotification
+            object: o_textfield];
+        [self addSubview: o_textfield];
 
+        /* build the mintextfield */
+        ADD_LABEL( o_textfield_min, mainFrame, 12, -30, @"-8888" )
         [o_textfield_min setFloatValue: p_item->f_min];
-        [o_textfield_min setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        [o_textfield_min setAutoresizingMask:NSViewMaxXMargin ];
         [self addSubview: o_textfield_min];
 
-        /* the slider */
-        s_rc.size.width = [self bounds].size.width - OFFSET_RIGHT - 2*OFFSET_BETWEEN - 3*40;
-        s_rc.origin.x = 40 + OFFSET_BETWEEN;
-
-        o_slider = [[[NSStepper alloc] initWithFrame: s_rc] retain];
-        [o_slider setAutoresizingMask:NSViewWidthSizable];
+        /* build the maxtextfield */
+        ADD_LABEL( o_textfield_max, mainFrame, mainFrame.size.width - 31, -30, @"8888" )
+        [o_textfield_max setFloatValue: p_item->f_max];
+        [o_textfield_max setAutoresizingMask:NSViewMinXMargin ];
+        [self addSubview: o_textfield_max];
 
-        [o_slider setMaxValue: p_item->f_max];
-        [o_slider setMinValue: p_item->f_min];
+        /* build the slider */
+        ADD_SLIDER( o_slider, mainFrame, [o_textfield_min frame].origin.x + [o_textfield_min frame].size.width + 6,
+            -1, mainFrame.size.width - [o_textfield_max frame].size.width - [o_textfield_max frame].size.width - 14
+            - [o_textfield_min frame].origin.x, o_tooltip, p_item->f_min, p_item->f_max )
         [o_slider setFloatValue: p_item->f_value];
+        [o_slider setAutoresizingMask:NSViewWidthSizable ];
         [o_slider setTarget: self];
         [o_slider setAction: @selector(sliderChanged:)];
         [o_slider sendActionOn:NSLeftMouseUpMask|NSLeftMouseDownMask|NSLeftMouseDraggedMask];
-        [o_slider setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
         [self addSubview: o_slider];
-        
-        /* max value textfield */
-        s_rc.size.width = 40;
-        s_rc.origin.x = [self bounds].size.width - OFFSET_RIGHT - OFFSET_BETWEEN - 2*40;
-        
-        o_textfield_max = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield_max setAutoresizingMask:NSViewMinXMargin ];
-
-        [o_textfield_max setDrawsBackground: NO];
-        [o_textfield_max setBordered: NO];
-        [o_textfield_max setEditable: NO];
-        [o_textfield_max setSelectable: NO];
-
-        [o_textfield_max setFloatValue: p_item->f_max];
-        [o_textfield_max setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
-        [self addSubview: o_textfield_max];
-        
-        /* current value textfield */
-        s_rc.size.width = 40;
-        s_rc.origin.x = [self bounds].size.width - OFFSET_RIGHT - 40;
-
-        o_textfield = [[[NSTextField alloc] initWithFrame: s_rc] retain];
-        [o_textfield setAutoresizingMask:NSViewMinXMargin];
-
-        [o_textfield setFloatValue: p_item->f_value];
-        [o_textfield setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
-        [o_textfield setDelegate: self];
-        [[NSNotificationCenter defaultCenter] addObserver: self
-            selector: @selector(textfieldChanged:)
-            name: NSControlTextDidChangeNotification
-            object: o_textfield];
-        [self addSubview: o_textfield];
 
     }
     return self;
     [o_slider setFloatValue: [o_textfield floatValue]];
 }
 
-- (float)floatValue
+- (int)floatValue
 {
     return [o_slider floatValue];
 }
 
 @end
 
-
 @implementation BoolConfigControl
 
-- (id)initWithFrame: (NSRect)frame
-        item: (module_config_t *)p_item
-        withObject: (vlc_object_t *)_p_this
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
 {
-    frame.size.height = 20;
-    if( self = [super initWithFrame: frame item: p_item
-            withObject: _p_this] )
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_tooltip;
+    mainFrame.size.height = 17;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
     {
-        NSRect s_rc = frame;
-        s_rc.size.height = 20;
-        
-        o_checkbox = [[[NSButton alloc] initWithFrame: s_rc] retain];
-        [o_checkbox setButtonType: NSSwitchButton];
-        [o_checkbox setIntValue: p_item->i_value];
-        [o_checkbox setTitle: [[VLCMain sharedInstance] localizedString: p_item->psz_text]];
-        [o_checkbox setToolTip: [[VLCMain sharedInstance] wrapString: [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP]];
+        /* add the checkbox */
+        if( p_item->psz_text )
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_CHECKBOX( o_checkbox, mainFrame, 0, 0, o_labelString, o_tooltip, p_item->i_value, NSImageRight)
+        [o_checkbox setAutoresizingMask:NSViewNotSizable ];
         [self addSubview: o_checkbox];
     }
     return self;
 
 - (int)intValue
 {
-    [o_checkbox intValue];
+    return [o_checkbox intValue];
 }
 
 @end
-#endif
 
+@implementation KeyConfigControlBefore103
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
+{
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_tooltip;
+    mainFrame.size.height = 37;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN + 1;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
+    {
+        /* add the label */
+        if( p_item->psz_text )
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -10, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
+        [self addSubview: o_label];
+
+        /* add the checkboxes */
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_CHECKBOX( o_cmd_checkbox, mainFrame,
+            [o_label frame].size.width + 2, 0,
+            [NSString stringWithUTF8String:PLACE_OF_INTEREST_SIGN],
+            o_tooltip, ((((unsigned int)p_item->i_value) & KEY_MODIFIER_COMMAND)?YES:NO), NSImageLeft )
+        [o_cmd_checkbox setState: p_item->i_value & KEY_MODIFIER_COMMAND];
+        ADD_CHECKBOX( o_ctrl_checkbox, mainFrame,
+            [o_cmd_checkbox frame].size.width + [o_cmd_checkbox frame].origin.x + 6, 0,
+            [NSString stringWithUTF8String:UP_ARROWHEAD],
+            o_tooltip, ((((unsigned int)p_item->i_value) & KEY_MODIFIER_CTRL)?YES:NO), NSImageLeft )
+        [o_ctrl_checkbox setState: p_item->i_value & KEY_MODIFIER_CTRL];
+        ADD_CHECKBOX( o_alt_checkbox, mainFrame,
+            [o_label frame].size.width + 2, -2 - [o_cmd_checkbox frame].size.height,
+            [NSString stringWithUTF8String:OPTION_KEY],
+            o_tooltip, ((((unsigned int)p_item->i_value) & KEY_MODIFIER_ALT)?YES:NO), NSImageLeft )
+        [o_alt_checkbox setState: p_item->i_value & KEY_MODIFIER_ALT];
+        ADD_CHECKBOX( o_shift_checkbox, mainFrame,
+            [o_cmd_checkbox frame].size.width + [o_cmd_checkbox frame].origin.x + 6,
+            -2 - [o_cmd_checkbox frame].size.height,
+            [NSString stringWithUTF8String:UPWARDS_WHITE_ARROW],
+            o_tooltip, ((((unsigned int)p_item->i_value) & KEY_MODIFIER_SHIFT)?YES:NO), NSImageLeft )
+        [o_shift_checkbox setState: p_item->i_value & KEY_MODIFIER_SHIFT];
+        [self addSubview: o_cmd_checkbox];
+        [self addSubview: o_ctrl_checkbox];
+        [self addSubview: o_alt_checkbox];
+        [self addSubview: o_shift_checkbox];
+
+        /* build the popup */
+        ADD_POPUP( o_popup, mainFrame, [o_shift_checkbox frame].origin.x + [o_shift_checkbox frame].size.width + 4,
+            4, 0, o_tooltip )
+        [o_popup setAutoresizingMask:NSViewWidthSizable ];
+    
+        if( o_keys_menu == nil )
+        {
+            unsigned int i;
+            o_keys_menu = [[NSMenu alloc] initWithTitle: @"Keys Menu"];
+            for ( i = 0; i < sizeof(vlc_keys) / sizeof(key_descriptor_t); i++)
+                if( vlc_keys[i].psz_key_string && *vlc_keys[i].psz_key_string )
+                    POPULATE_A_KEY( o_keys_menu, [NSString stringWithCString:vlc_keys[i].psz_key_string]
+                                     , vlc_keys[i].i_key_code)
+        }
+        [o_popup setMenu:[o_keys_menu copyWithZone:nil]];
+        [o_popup selectItemWithTitle: [[VLCMain sharedInstance] localizedString:KeyToString(( ((unsigned int)p_item->i_value) & ~KEY_MODIFIER ))]];
+        [self addSubview: o_popup];
+
+    }
+    return self;
+}
+
+- (void)dealloc
+{
+    [o_cmd_checkbox release];
+    [o_ctrl_checkbox release];
+    [o_alt_checkbox release];
+    [o_shift_checkbox release];
+    [o_popup release];
+    [super dealloc];
+}
+
+- (int)intValue
+{
+    unsigned int i_new_key = 0;
+
+    i_new_key |= ([o_cmd_checkbox state] == NSOnState) ? KEY_MODIFIER_COMMAND : 0;
+    i_new_key |= ([o_ctrl_checkbox state] == NSOnState) ? KEY_MODIFIER_CTRL : 0;
+    i_new_key |= ([o_alt_checkbox state] == NSOnState) ? KEY_MODIFIER_ALT : 0;
+    i_new_key |= ([o_shift_checkbox state] == NSOnState) ? KEY_MODIFIER_SHIFT : 0;
+
+    i_new_key |= StringToKey([[[o_popup selectedItem] title] cString]);
+    return i_new_key;
+}
+
+@end
+
+@implementation KeyConfigControlAfter103
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
+{
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_tooltip;
+    mainFrame.size.height = 22;
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN + 1;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
+    {
+        /* add the label */
+        if( p_item->psz_text )
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -1, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
+        [self addSubview: o_label];
+
+        /* build the popup */
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        ADD_POPUP( o_popup, mainFrame, [o_label frame].origin.x + [o_label frame].size.width + 3,
+            -2, 0, o_tooltip )
+        [o_popup setAutoresizingMask:NSViewWidthSizable ];
+    
+        if( o_keys_menu == nil )
+        {
+            unsigned int i;
+            o_keys_menu = [[NSMenu alloc] initWithTitle: @"Keys Menu"];
+            for ( i = 0; i < sizeof(vlc_keys) / sizeof(key_descriptor_t); i++)
+                if( vlc_keys[i].psz_key_string && *vlc_keys[i].psz_key_string )
+                    POPULATE_A_KEY( o_keys_menu, [NSString stringWithCString:vlc_keys[i].psz_key_string]
+                                     , vlc_keys[i].i_key_code)
+        }
+        [o_popup setMenu:[o_keys_menu copyWithZone:nil]];
+        [o_popup selectItemWithTag: p_item->i_value];
+        [self addSubview: o_popup];
+
+    }
+    return self;
+}
+
+- (void)dealloc
+{
+    [o_popup release];
+    [super dealloc];
+}
+
+- (int)intValue
+{
+    return [o_popup selectedTag];
+}
+
+@end
+
+@implementation ModuleListConfigControl
+
+- (id) initWithItem: (module_config_t *)_p_item
+           withView: (NSView *)o_parent_view
+           withVerticalOffset: (int)i_yPos
+           withLastItem: (int)i_lastItem
+{
+if( _p_item->i_type == CONFIG_ITEM_MODULE_LIST )
+//TODO....
+        return nil;
+
+//Fill our array to know how may items we have...
+    vlc_list_t *p_list;
+    module_t *p_parser;
+    int i_index;
+    NSRect mainFrame = [o_parent_view frame];
+    NSString *o_labelString, *o_textfieldString, *o_tooltip;
+
+    o_modulearray = [[NSMutableArray alloc] initWithCapacity:10];
+    /* build a list of available modules */
+    p_list = vlc_list_find( VLCIntf, VLC_OBJECT_MODULE, FIND_ANYWHERE );
+    for( i_index = 0; i_index < p_list->i_count; i_index++ )
+    {
+        p_parser = (module_t *)p_list->p_values[i_index].p_object ;
+
+        if( !strcmp( p_parser->psz_object_name, "main" ) )
+            continue;
+
+        module_config_t *p_config = p_parser->p_config;
+        if( p_config ) do
+        {
+            NSString *o_modulelongname, *o_modulename;
+            NSNumber *o_moduleenabled = nil;
+            /* Hack: required subcategory is stored in i_min */
+            if( p_config->i_type == CONFIG_SUBCATEGORY &&
+                p_config->i_value == _p_item->i_min )
+            {
+                o_modulelongname = [NSString stringWithCString:p_parser->psz_longname];
+                o_modulename = [NSString stringWithCString:p_parser->psz_object_name];
+
+                if( _p_item->psz_value &&
+                    strstr( _p_item->psz_value, p_parser->psz_object_name ) )
+                    o_moduleenabled = [NSNumber numberWithBool:YES];
+                else
+                    o_moduleenabled = [NSNumber numberWithBool:NO];
+
+                [o_modulearray addObject:[NSMutableArray arrayWithObjects: o_modulename, o_modulelongname, o_moduleenabled, nil]];
+            }
+        } while( p_config->i_type != CONFIG_HINT_END && p_config++ );
+    }
+    vlc_list_release( p_list );
+
+    mainFrame.size.height = 30 + 18 * [o_modulearray count];
+    mainFrame.size.width = mainFrame.size.width - LEFTMARGIN - RIGHTMARGIN;
+    mainFrame.origin.x = LEFTMARGIN;
+    mainFrame.origin.y = i_yPos;
+    if( [super initWithFrame: mainFrame item: _p_item] != nil )
+    {
+        /* add the label */
+        if( p_item->psz_text )
+            o_labelString = [[VLCMain sharedInstance] localizedString: p_item->psz_text];
+        else
+            o_labelString = [NSString stringWithString:@""];
+        ADD_LABEL( o_label, mainFrame, 0, -3, o_labelString )
+        [o_label setAutoresizingMask:NSViewNotSizable ];
+        [self addSubview: o_label];
+
+        /* build the textfield */
+        o_tooltip = [[VLCMain sharedInstance] wrapString:
+            [[VLCMain sharedInstance] localizedString: p_item->psz_longtext ] toWidth: PREFS_WRAP];
+        if( p_item->psz_value )
+            o_textfieldString = [[VLCMain sharedInstance] localizedString: p_item->psz_value];
+        else
+            o_textfieldString = [NSString stringWithString: @""];
+        ADD_TEXTFIELD( o_textfield, mainFrame, [o_label frame].size.width + 2,
+            mainFrame.size.height - 22, mainFrame.size.width - [o_label frame].size.width - 2,
+            o_tooltip, o_textfieldString )
+        [o_textfield setAutoresizingMask:NSViewWidthSizable ];
+        [self addSubview: o_textfield];
+
+
+{
+    NSRect s_rc = mainFrame;
+    s_rc.size.height = mainFrame.size.height - 30;
+    s_rc.size.width = mainFrame.size.width - 12;
+    s_rc.origin.x = 12;
+    s_rc.origin.y = 0;
+    o_scrollview = [[[NSScrollView alloc] initWithFrame: s_rc] retain];
+    [o_scrollview setDrawsBackground: NO];
+    [o_scrollview setBorderType: NSBezelBorder];
+    [o_scrollview setAutohidesScrollers:YES];
+
+    NSTableView *o_tableview;
+    o_tableview = [[NSTableView alloc] initWithFrame : s_rc];
+    [o_tableview setUsesAlternatingRowBackgroundColors:YES];
+    [o_tableview setHeaderView:nil];
+/* TODO: find a good way to fix the row height and text size*/
+/* FIXME: support for multiple selection... */
+//    [o_tableview setAllowsMultipleSelection:YES];
+    
+    NSCell *o_headerCell = [[NSCell alloc] initTextCell:@"Enabled"];
+    NSCell *o_dataCell = [[NSButtonCell alloc] init];
+    [(NSButtonCell*)o_dataCell setButtonType:NSSwitchButton];
+    [o_dataCell setTitle:@""];
+    [o_dataCell setFont:[NSFont systemFontOfSize:0]];
+    NSTableColumn *o_tableColumn = [[NSTableColumn alloc]initWithIdentifier:[NSString stringWithCString: "Enabled"]];
+    [o_tableColumn setHeaderCell: o_headerCell];
+    [o_tableColumn setDataCell: o_dataCell];
+    [o_tableColumn setWidth:17];
+    [o_tableview addTableColumn: o_tableColumn];
+
+    o_headerCell = [[NSCell alloc] initTextCell:@"Module Name"];
+    o_dataCell = [[NSTextFieldCell alloc] init];
+    [o_dataCell setFont:[NSFont systemFontOfSize:12]];
+    o_tableColumn = [[NSTableColumn alloc]initWithIdentifier:[NSString stringWithCString: "Module"]];
+    [o_tableColumn setHeaderCell: o_headerCell];
+    [o_tableColumn setDataCell: o_dataCell];
+    [o_tableColumn setWidth:388 - 17];
+    [o_tableview addTableColumn: o_tableColumn];
+       [o_tableview registerForDraggedTypes:[NSArray arrayWithObjects:
+            @"VLC media player module", nil]];
+
+    [o_tableview setDataSource:self];
+    [o_tableview setTarget: self];
+    [o_tableview setAction: @selector(tableChanged:)];
+    [o_tableview sendActionOn:NSLeftMouseUpMask|NSLeftMouseDownMask|NSLeftMouseDraggedMask];
+    [o_scrollview setDocumentView: o_tableview];
+}
+    [o_scrollview setAutoresizingMask:NSViewWidthSizable ];
+    [self addSubview: o_scrollview];
+
+
+    }
+    return self;
+}
+
+- (IBAction)tableChanged:(id)sender
+{
+    NSString *o_newstring = @"";
+    unsigned int i;
+    for( i = 0 ; i < [o_modulearray count] ; i++ )
+        if( [[[o_modulearray objectAtIndex:i] objectAtIndex:2] boolValue] != NO )
+        {
+            o_newstring = [o_newstring stringByAppendingString:[[o_modulearray objectAtIndex:i] objectAtIndex:0]];
+            o_newstring = [o_newstring stringByAppendingString:@","];
+        }
+    
+    [o_textfield setStringValue: [o_newstring substringToIndex: ([o_newstring length])?[o_newstring length] - 1:0]];
+}
+
+- (void)dealloc
+{
+    [o_scrollview release];
+    [super dealloc];
+}
+
+
+- (char *)stringValue
+{
+    return strdup( [[o_textfield stringValue] cString] );
+}
+
+@end
+
+@implementation ModuleListConfigControl (NSTableDataSource)
+
+- (BOOL)tableView:(NSTableView*)table writeRows:(NSArray*)rows toPasteboard:(NSPasteboard*)pb
+{
+       // We only want to allow dragging of selected rows.
+       NSEnumerator    *iter = [rows objectEnumerator];
+       NSNumber                *row;
+       while ((row = [iter nextObject]) != nil)
+       {
+               if (![table isRowSelected:[row intValue]])
+                       return NO;
+       }
+       
+       [pb declareTypes:[NSArray arrayWithObject:@"VLC media player module"] owner:nil];
+       [pb setPropertyList:rows forType:@"VLC media player module"];
+       return YES;
+
+}
+
+- (NSDragOperation)tableView:(NSTableView*)table validateDrop:(id <NSDraggingInfo>)info proposedRow:(int)row proposedDropOperation:(NSTableViewDropOperation)op
+{
+       // Make drops at the end of the table go to the end.
+       if (row == -1)
+       {
+               row = [table numberOfRows];
+               op = NSTableViewDropAbove;
+               [table setDropRow:row dropOperation:op];
+       }
+       
+       // We don't ever want to drop onto a row, only between rows.
+       if (op == NSTableViewDropOn)
+               [table setDropRow:(row+1) dropOperation:NSTableViewDropAbove];
+       return NSTableViewDropAbove;
+}
+
+- (BOOL)tableView:(NSTableView*)table acceptDrop:(id <NSDraggingInfo>)info row:(int)dropRow dropOperation:(NSTableViewDropOperation)op;
+{
+       NSPasteboard    *pb = [info draggingPasteboard];
+       NSDragOperation srcMask = [info draggingSourceOperationMask];
+       BOOL accepted = NO;
+
+       NS_DURING
+                       
+        NSArray        *array;
+
+        // Intra-table drag - data is the array of rows.
+        if (!accepted && (array = [pb propertyListForType:@"VLC media player module"]) != NULL)
+        {
+            NSEnumerator *     iter = nil;
+            id val;
+                       BOOL isCopy = (srcMask & NSDragOperationMove) ? NO:YES;
+            // Move the modules
+                       iter = [array objectEnumerator];
+                       while ((val = [iter nextObject]) != NULL)
+                       {
+                NSArray *o_tmp = [[o_modulearray objectAtIndex:[val intValue]] mutableCopyWithZone:nil];
+                [o_modulearray removeObject:o_tmp];
+                [o_modulearray insertObject:o_tmp atIndex:(dropRow>[val intValue]) ? dropRow - 1 : dropRow];
+                dropRow++;
+            }
+
+            // Select the newly-dragged items.
+            iter = [array objectEnumerator];
+//TODO...
+                       [table deselectAll:self];
+
+            [self tableChanged:self];
+            [table setNeedsDisplay:YES];
+                       // Indicate that we finished the drag.
+                       accepted = YES;
+               }
+        [table reloadData];
+        [table setNeedsDisplay:YES];
+                       
+               NS_HANDLER
+               
+                       // An exception occurred. Uh-oh. Update the track table so that
+                       //      it stays consistent, and re-raise the exception.
+                       [table reloadData];
+                       [localException raise];
+            [table setNeedsDisplay:YES];
+    NS_ENDHANDLER
+               
+       return accepted;
+}
+
+- (int)numberOfRowsInTableView:(NSTableView *)aTableView
+{
+    return [o_modulearray count];
+}
+
+- (id)tableView:(NSTableView *)aTableView objectValueForTableColumn:(NSTableColumn *)aTableColumn row:(int)rowIndex
+{
+    if( [[aTableColumn identifier] isEqualToString:[NSString stringWithCString:"Enabled"]] )
+        return [[o_modulearray objectAtIndex:rowIndex] objectAtIndex:2];
+    if( [[aTableColumn identifier] isEqualToString:[NSString stringWithCString:"Module"]] )
+        return [[o_modulearray objectAtIndex:rowIndex] objectAtIndex:1];
+        
+    return nil;
+}
+
+- (void)tableView:(NSTableView *)aTableView setObjectValue:(id)anObject forTableColumn:(NSTableColumn *)aTableColumn row:(int)rowIndex
+{
+    [[o_modulearray objectAtIndex:rowIndex] replaceObjectAtIndex:2 withObject: anObject];
+}
+@end