]> git.sesse.net Git - vlc/blob - modules/gui/qt4/components/complete_preferences.cpp
Qt: remove code duplication.
[vlc] / modules / gui / qt4 / components / complete_preferences.cpp
1 /*****************************************************************************
2  * preferences.cpp : "Normal preferences"
3  ****************************************************************************
4  * Copyright (C) 2006-2007 the VideoLAN team
5  * $Id$
6  *
7  * Authors: ClĂ©ment Stenac <zorglub@videolan.org>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23 #ifdef HAVE_CONFIG_H
24 # include "config.h"
25 #endif
26
27 #include <QApplication>
28 #include <QLabel>
29 #include <QTreeWidget>
30 #include <QTreeWidgetItem>
31 #include <QVariant>
32 #include <QString>
33 #include <QFont>
34 #include <QGroupBox>
35 #include <QScrollArea>
36 #include <QVBoxLayout>
37 #include <QHBoxLayout>
38 #include <QGridLayout>
39 #include <QHeaderView>
40 #include <QPalette>
41 #include <QColor>
42
43 #include "components/complete_preferences.hpp"
44 #include "components/preferences_widgets.hpp"
45
46 #include <vlc_config_cat.h>
47 #include <vlc_intf_strings.h>
48 #include <assert.h>
49
50 #define ITEM_HEIGHT 25
51
52 /*********************************************************************
53  * The Tree
54  *********************************************************************/
55 PrefsTree::PrefsTree( intf_thread_t *_p_intf, QWidget *_parent ) :
56                             QTreeWidget( _parent ), p_intf( _p_intf )
57 {
58     /* General Qt options */
59     setColumnCount( 1 );
60     setAlternatingRowColors( true );
61     header()->hide();
62
63     setIconSize( QSize( ITEM_HEIGHT,ITEM_HEIGHT ) );
64     setTextElideMode( Qt::ElideNone );
65
66     /* Nice icons */
67 #define BI( a,b) QIcon a##_icon = QIcon( QPixmap( b ))
68     BI( audio, ":/advprefs_audio" );
69     BI( video, ":/advprefs_video" );
70     BI( input, ":/advprefs_codec" );
71     BI( sout, ":/advprefs_sout" );
72     BI( advanced, ":/advprefs_extended" );
73     BI( playlist, ":/advprefs_playlist" );
74     BI( interface, ":/advprefs_intf" );
75 #undef BI
76
77     /* Build the tree for the main module */
78     module_t *p_module = module_get_main();
79
80     /* Initialisation and get the confsize */
81     PrefsItemData *data = NULL;
82     PrefsItemData *data_sub = NULL;
83     QTreeWidgetItem *current_item = NULL;
84     unsigned confsize;
85     module_config_t *const p_config = module_config_get (p_module, &confsize);
86
87     /* Go through the list of conf */
88     for( size_t i = 0; i < confsize; i++ )
89     {
90         const char *psz_help;
91         QIcon icon;
92
93         /* Work on a new item */
94         module_config_t *p_item = p_config + i;
95
96         switch( p_item->i_type )
97         {
98         /* This is a category */
99         case CONFIG_CATEGORY:
100             if( p_item->value.i == -1 ) break;
101
102             /* PrefsItemData Init */
103             data = new PrefsItemData();
104             data->name = qtr( config_CategoryNameGet( p_item->value.i ) );
105             psz_help = config_CategoryHelpGet( p_item->value.i );
106             if( psz_help )
107                 data->help = qtr( psz_help );
108             else
109                 data->help.clear();
110             data->i_type = TYPE_CATEGORY;
111             data->i_object_id = p_item->value.i;
112
113             /* This is a category, put a nice icon */
114             switch( p_item->value.i )
115             {
116 #define CI(a,b) case a: icon = b##_icon;break
117             CI( CAT_AUDIO, audio );
118             CI( CAT_VIDEO, video );
119             CI( CAT_INPUT, input );
120             CI( CAT_SOUT, sout );
121             CI( CAT_ADVANCED, advanced );
122             CI( CAT_PLAYLIST, playlist );
123             CI( CAT_INTERFACE, interface );
124             }
125 #undef CI
126
127             /* Create a new QTreeItem to display it in the tree at top level */
128             current_item = new QTreeWidgetItem();
129             current_item->setText( 0, data->name );
130             current_item->setIcon( 0 , icon );
131             current_item->setSizeHint( 0, QSize( -1, ITEM_HEIGHT ) );
132             current_item->setData( 0, Qt::UserRole,
133                                    qVariantFromValue( data ) );
134             addTopLevelItem( current_item );
135             break;
136
137         /* This is a subcategory */
138         case CONFIG_SUBCATEGORY:
139             if( p_item->value.i == -1 ) break;
140
141             /* Special cases: move the main subcategories to the parent cat*/
142             if( data &&
143                 ( p_item->value.i == SUBCAT_VIDEO_GENERAL ||
144                   p_item->value.i == SUBCAT_ADVANCED_MISC ||
145                   p_item->value.i == SUBCAT_INPUT_GENERAL ||
146                   p_item->value.i == SUBCAT_INTERFACE_GENERAL ||
147                   p_item->value.i == SUBCAT_SOUT_GENERAL||
148                   p_item->value.i == SUBCAT_PLAYLIST_GENERAL||
149                   p_item->value.i == SUBCAT_AUDIO_GENERAL ) )
150             {
151                 /* Data still contains the correct thing */
152                 data->i_type = TYPE_CATSUBCAT;
153                 data->i_subcat_id = p_item->value.i;
154                 data->name = qtr( config_CategoryNameGet( p_item->value.i ) );
155                 psz_help = config_CategoryHelpGet( p_item->value.i );
156                 if( psz_help )
157                     data->help = qtr( psz_help );
158                 else
159                     data->help.clear();
160                 current_item->setData( 0, Qt::UserRole,
161                                        QVariant::fromValue( data ) );
162                 continue;
163             }
164
165             /* Normal Subcategories */
166
167             /* Process the Data */
168             data_sub = new PrefsItemData();
169             data_sub->name = qtr( config_CategoryNameGet( p_item->value.i) );
170             psz_help = config_CategoryHelpGet( p_item->value.i );
171             if( psz_help )
172                 data_sub->help = qtr( psz_help );
173             else
174                 data_sub->help.clear();
175             data_sub->i_type = TYPE_SUBCATEGORY;
176             data_sub->i_object_id = p_item->value.i;
177
178             /* Create a new TreeWidget */
179             QTreeWidgetItem *subcat_item = new QTreeWidgetItem();
180             subcat_item->setText( 0, data_sub->name );
181             subcat_item->setData( 0, Qt::UserRole,
182                                   qVariantFromValue( data_sub ) );
183             subcat_item->setSizeHint( 0, QSize( -1, ITEM_HEIGHT ) );
184
185             /* Add it to the parent */
186             assert( current_item );
187             current_item->addChild( subcat_item );
188             break;
189
190         /* Other items don't need yet a place on the tree */
191         }
192     }
193     module_config_free( p_config );
194     module_release( p_module );
195
196
197     module_t **p_list = module_list_get( NULL );
198     /* Build the tree of plugins */
199     for( size_t i = 0; (p_module = p_list[i]) != NULL; i++ )
200     {
201         // Main module excluded
202         if( module_is_main( p_module) ) continue;
203
204         unsigned  confsize;
205         int i_subcategory = 0, i_category = 0;
206
207         bool b_options = false;
208         module_config_t *const p_config = module_config_get (p_module, &confsize);
209
210         /* Loop through the configurations items */
211         for (size_t i = 0; i < confsize; i++)
212         {
213             const module_config_t *p_item = p_config + i;
214
215             if( p_item->i_type == CONFIG_CATEGORY )
216                 i_category = p_item->value.i;
217             else if( p_item->i_type == CONFIG_SUBCATEGORY )
218                 i_subcategory = p_item->value.i;
219
220             if( p_item->i_type & CONFIG_ITEM )
221                 b_options = true;
222
223             if( b_options && i_category && i_subcategory )
224                 break;
225         }
226         module_config_free (p_config);
227
228         /* Dummy item, please proceed */
229         if( !b_options || i_category == 0 || i_subcategory == 0 ) continue;
230
231
232         // Locate the category item;
233         QTreeWidgetItem *subcat_item = NULL;
234         bool b_found = false;
235
236         for( int i_cat_index = 0 ; i_cat_index < topLevelItemCount();
237                                    i_cat_index++ )
238         {
239             /* Get the treeWidgetItem that correspond to the category */
240             QTreeWidgetItem *cat_item = topLevelItem( i_cat_index );
241             PrefsItemData *data = cat_item->data( 0, Qt::UserRole ).
242                                              value<PrefsItemData *>();
243
244             /* If we match the good category */
245             if( data->i_object_id == i_category )
246             {
247                 for( int i_sc_index = 0; i_sc_index < cat_item->childCount();
248                          i_sc_index++ )
249                 {
250                     subcat_item = cat_item->child( i_sc_index );
251                     PrefsItemData *sc_data = subcat_item->data(0, Qt::UserRole).
252                                                 value<PrefsItemData *>();
253                     if( sc_data && sc_data->i_object_id == i_subcategory )
254                     {
255                         b_found = true;
256                         break;
257                     }
258                 }
259                 if( !b_found )
260                 {
261                     subcat_item = cat_item;
262                     b_found = true;
263                 }
264                 break;
265             }
266         }
267         if( !b_found ) continue;
268
269         PrefsItemData *module_data = new PrefsItemData();
270         module_data->i_type = TYPE_MODULE;
271         module_data->psz_name = strdup( module_get_object( p_module ) );
272         module_data->help.clear();
273         QTreeWidgetItem *module_item = new QTreeWidgetItem();
274         module_item->setText( 0, qtr( module_get_name( p_module, false ) ) );
275         module_item->setData( 0, Qt::UserRole,
276                               QVariant::fromValue( module_data) );
277         module_item->setSizeHint( 0, QSize( -1, ITEM_HEIGHT ) );
278         subcat_item->addChild( module_item );
279     }
280
281     /* We got everything, just sort a bit */
282     sortItems( 0, Qt::AscendingOrder );
283
284     module_list_free( p_list );
285 }
286
287 PrefsTree::~PrefsTree() {}
288
289 void PrefsTree::applyAll()
290 {
291     doAll( false );
292 }
293
294 void PrefsTree::cleanAll()
295 {
296     doAll( true );
297 }
298
299 void PrefsTree::doAll( bool doclean )
300 {
301     for( int i_cat_index = 0 ; i_cat_index < topLevelItemCount();
302              i_cat_index++ )
303     {
304         QTreeWidgetItem *cat_item = topLevelItem( i_cat_index );
305         for( int i_sc_index = 0; i_sc_index < cat_item->childCount();
306                  i_sc_index++ )
307         {
308             QTreeWidgetItem *sc_item = cat_item->child( i_sc_index );
309             for( int i_module = 0 ; i_module < sc_item->childCount();
310                      i_module++ )
311             {
312                 PrefsItemData *data = sc_item->child( i_module )->
313                                data( 0, Qt::UserRole).value<PrefsItemData *>();
314                 if( data->panel && doclean )
315                 {
316                     delete data->panel;
317                     data->panel = NULL;
318                 }
319                 else if( data->panel )
320                     data->panel->apply();
321             }
322             PrefsItemData *data = sc_item->data( 0, Qt::UserRole).
323                                             value<PrefsItemData *>();
324             if( data->panel && doclean )
325             {
326                 delete data->panel;
327                 data->panel = NULL;
328             }
329             else if( data->panel )
330                 data->panel->apply();
331         }
332         PrefsItemData *data = cat_item->data( 0, Qt::UserRole).
333                                             value<PrefsItemData *>();
334         if( data->panel && doclean )
335         {
336             delete data->panel;
337             data->panel = NULL;
338         }
339         else if( data->panel )
340             data->panel->apply();
341     }
342 }
343
344 /*********************************************************************
345  * The Panel
346  *********************************************************************/
347 AdvPrefsPanel::AdvPrefsPanel( QWidget *_parent ) : QWidget( _parent )
348 {}
349
350 AdvPrefsPanel::AdvPrefsPanel( intf_thread_t *_p_intf, QWidget *_parent,
351                         PrefsItemData * data ) :
352                         QWidget( _parent ), p_intf( _p_intf )
353 {
354     /* Find our module */
355     module_t *p_module = NULL;
356     if( data->i_type == TYPE_CATEGORY )
357         return;
358     else if( data->i_type == TYPE_MODULE )
359         p_module = module_find( data->psz_name );
360     else
361     {
362         p_module = module_get_main();
363         assert( p_module );
364     }
365
366     unsigned confsize;
367     module_config_t *const p_config = module_config_get (p_module, &confsize),
368                     *p_item = p_config,
369                     *p_end = p_config + confsize;
370
371     if( data->i_type == TYPE_SUBCATEGORY || data->i_type ==  TYPE_CATSUBCAT )
372     {
373         while (p_item < p_end)
374         {
375             if(  p_item->i_type == CONFIG_SUBCATEGORY &&
376                             ( ( data->i_type == TYPE_SUBCATEGORY &&
377                               p_item->value.i == data->i_object_id ) ||
378                             ( data->i_type == TYPE_CATSUBCAT &&
379                               p_item->value.i == data->i_subcat_id ) ) )
380                 break;
381             p_item++;
382         }
383     }
384
385     /* Widgets now */
386     global_layout = new QVBoxLayout();
387     global_layout->setMargin( 2 );
388     QString head;
389     QString help;
390
391     help = QString( data->help );
392
393     if( data->i_type == TYPE_SUBCATEGORY || data->i_type ==  TYPE_CATSUBCAT )
394     {
395         head = QString( data->name );
396         p_item++; // Why that ?
397     }
398     else
399     {
400         const char *psz_help = module_get_help (p_module);
401         head = QString( qtr( module_GetLongName( p_module ) ) );
402         if( psz_help )
403         {
404             help.append( "\n" );
405             help.append( qtr( psz_help ) );
406         }
407     }
408
409     QLabel *titleLabel = new QLabel( head );
410     QFont titleFont = QApplication::font( static_cast<QWidget*>(0) );
411     titleFont.setPointSize( titleFont.pointSize() + 6 );
412     titleFont.setFamily( "Verdana" );
413     titleLabel->setFont( titleFont );
414
415     // Title <hr>
416     QFrame *title_line = new QFrame;
417     title_line->setFrameShape(QFrame::HLine);
418     title_line->setFrameShadow(QFrame::Sunken);
419
420     QLabel *helpLabel = new QLabel( help, this );
421     helpLabel->setWordWrap( true );
422
423     global_layout->addWidget( titleLabel );
424     global_layout->addWidget( title_line );
425     global_layout->addWidget( helpLabel );
426
427     QGroupBox *box = NULL;
428     QGridLayout *boxlayout = NULL;
429
430     QScrollArea *scroller= new QScrollArea;
431     scroller->setFrameStyle( QFrame::NoFrame );
432     QWidget *scrolled_area = new QWidget;
433
434     QGridLayout *layout = new QGridLayout();
435     int i_line = 0, i_boxline = 0;
436     bool has_hotkey = false;
437
438     if( p_item ) do
439     {
440         if( ( ( data->i_type == TYPE_SUBCATEGORY &&
441                 p_item->value.i != data->i_object_id ) ||
442               ( data->i_type == TYPE_CATSUBCAT  &&
443                 p_item->value.i != data->i_subcat_id ) ) &&
444             ( p_item->i_type == CONFIG_CATEGORY ||
445               p_item->i_type == CONFIG_SUBCATEGORY ) )
446             break;
447         if( p_item->b_internal == true ) continue;
448
449         if( p_item->i_type == CONFIG_SECTION )
450         {
451             if( box )
452             {
453                 box->setLayout( boxlayout );
454                 box->show();
455                 layout->addWidget( box, i_line, 0, 1, -1 );
456                 i_line++;
457             }
458             box = new QGroupBox( qtr( p_item->psz_text ), this );
459             box->hide();
460             boxlayout = new QGridLayout();
461         }
462         /* Only one hotkey control */
463         if( has_hotkey && p_item->i_type & CONFIG_ITEM && p_item->psz_name &&
464                                          strstr( p_item->psz_name, "key-" ) )
465             continue;
466         if( p_item->i_type & CONFIG_ITEM && p_item->psz_name &&
467                                             strstr( p_item->psz_name, "key-" ) )
468             has_hotkey = true;
469
470         ConfigControl *control;
471         if( ! box )
472             control = ConfigControl::createControl( VLC_OBJECT( p_intf ),
473                                         p_item, this, layout, i_line );
474         else
475             control = ConfigControl::createControl( VLC_OBJECT( p_intf ),
476                                     p_item, this, boxlayout, i_boxline );
477         if( !control )
478             continue;
479
480         if( box ) i_boxline++;
481         else i_line++;
482         controls.append( control );
483     }
484     while( !( ( data->i_type == TYPE_SUBCATEGORY ||
485                data->i_type == TYPE_CATSUBCAT ) &&
486              ( p_item->i_type == CONFIG_CATEGORY ||
487                p_item->i_type == CONFIG_SUBCATEGORY ) )
488         && ( ++p_item < p_end ) );
489
490     if( box )
491     {
492         box->setLayout( boxlayout );
493         box->show();
494         layout->addWidget( box, i_line, 0, 1, -1 );
495     }
496
497     module_release (p_module);
498
499     scrolled_area->setSizePolicy( QSizePolicy::Preferred,QSizePolicy::Fixed );
500     scrolled_area->setLayout( layout );
501     scroller->setWidget( scrolled_area );
502     scroller->setWidgetResizable( true );
503     global_layout->addWidget( scroller );
504     setLayout( global_layout );
505 }
506
507 void AdvPrefsPanel::apply()
508 {
509     QList<ConfigControl *>::Iterator i;
510     for( i = controls.begin() ; i != controls.end() ; i++ )
511     {
512         ConfigControl *c = qobject_cast<ConfigControl *>(*i);
513         c->doApply( p_intf );
514     }
515 }
516
517 void AdvPrefsPanel::clean()
518 {}
519
520 AdvPrefsPanel::~AdvPrefsPanel()
521 {
522     qDeleteAll( controls ); controls.clear();
523 }
524