]> git.sesse.net Git - vlc/blob - modules/gui/qt4/dialogs/sout.cpp
35433f1dc18fa21358f7aeedd475bbd1159bd090
[vlc] / modules / gui / qt4 / dialogs / sout.cpp
1 /*****************************************************************************
2  * sout.cpp : Stream output dialog ( old-style )
3  ****************************************************************************
4  * Copyright (C) 2007-2008 the VideoLAN team
5  * Copyright (C) 2007 Société des arts technologiques
6  * Copyright (C) 2007 Savoir-faire Linux
7  *
8  * $Id$
9  *
10  * Authors: Clément Stenac <zorglub@videolan.org>
11  *          Jean-Baptiste Kempf <jb@videolan.org>
12  *          Jean-François Massol <jf.massol -at- gmail.com>
13  *          Pierre-Luc Beaudoin <pierre-luc.beaudoin@savoirfairelinux.com>
14  *
15  * This program is free software; you can redistribute it and/or modify
16  * it under the terms of the GNU General Public License as published by
17  * the Free Software Foundation; either version 2 of the License, or
18  * ( at your option ) any later version.
19  *
20  * This program is distributed in the hope that it will be useful,
21  * but WITHOUT ANY WARRANTY; without even the implied warranty of
22  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
23  * GNU General Public License for more details.
24  *
25  * You should have received a copy of the GNU General Public License
26  * along with this program; if not, write to the Free Software
27  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
28  *****************************************************************************/
29
30 #ifdef HAVE_CONFIG_H
31 # include "config.h"
32 #endif
33
34 #include "dialogs/sout.hpp"
35
36 #include <QString>
37 #include <QFileDialog>
38
39 struct streaming_account_t
40 {
41     char *psz_username; /*< username of account */
42     char *psz_password; /*< password of account */
43 };
44
45 struct sout_gui_descr_t
46 {
47     /* Access types */
48     bool b_local;   /*< local access module */
49     bool b_file;    /*< file access module */
50     bool b_http;    /*< http access module */
51     bool b_mms;     /*< mms access module */
52     bool b_rtp;     /*< rtp access module */
53     bool b_udp;     /*< udp access module */
54     bool b_dump;    /*< dump access module */
55     bool b_icecast; /*< icecast access module */
56
57     char *psz_file;     /*< filename */
58     char *psz_http;     /*< HTTP servername or ipaddress */
59     char *psz_mms;      /*< MMS servername or ipaddress */
60     char *psz_rtp;      /*< RTP servername or ipaddress */
61     char *psz_udp;      /*< UDP servername or ipaddress */
62     char *psz_icecast;  /*< Icecast servername or ipaddress*/
63
64     int32_t i_http;     /*< http port number */
65     int32_t i_mms;      /*< mms port number */
66     int32_t i_rtp;      /*< rtp port number */
67     int32_t i_rtp_audio;      /*< rtp port number */
68     int32_t i_rtp_video;      /*< rtp port number */
69     int32_t i_udp;      /*< udp port number */
70     int32_t i_icecast;  /*< icecast port number */
71
72     /* Mux */
73     char *psz_mux;      /*< name of muxer to use in streaming */
74
75     /* Transcode */
76     bool b_soverlay; /*< enable burning overlay in the video */
77     char *psz_vcodec;   /*< video codec to use in transcoding */
78     char *psz_acodec;   /*< audio codec to use in transcoding */
79     char *psz_scodec;   /*< subtitle codec to use in transcoding */
80     int32_t i_vb;       /*< video bitrate to use in transcoding */
81     int32_t i_ab;       /*< audio bitrate to use in transcoding */
82     int32_t i_channels; /*< number of audio channels to use in transcoding */
83     float f_scale;      /*< scaling factor to use in transcoding */
84
85     /* Misc */
86     bool b_sap;   /*< send SAP announcement */
87     bool b_all_es;/*< send all elementary streams from source stream */
88     char *psz_group;    /*< SAP Group name */
89     char *psz_name;     /*< SAP name */
90     int32_t i_ttl;      /*< Time To Live (TTL) for network traversal */
91
92     /* Icecast */
93     char *psz_icecast_mountpoint;/*< path to Icecast mountpoint */
94     struct streaming_account_t sa_icecast;  /*< Icecast account information */
95 };
96
97 SoutDialog* SoutDialog::instance = NULL;
98
99 SoutDialog::SoutDialog( QWidget *parent, intf_thread_t *_p_intf,
100                      bool _transcode_only ) : QVLCDialog( parent,  _p_intf )
101 {
102     setWindowTitle( qtr( "Stream Output" ) );
103
104     b_transcode_only = _transcode_only;
105
106     /* UI stuff */
107     ui.setupUi( this );
108
109     changeUDPandRTPmess( false );
110
111 /* ADD HERE for new profiles */
112 #define ADD_PROFILE( name, shortname ) ui.profileBox->addItem( qtr( name ), QVariant( QString( shortname ) ) );
113     ADD_PROFILE( "Custom" , "Custom" )
114     ADD_PROFILE( "IPod (mp4/aac)", "IPod" )
115     ADD_PROFILE( "XBox", "XBox" )
116     ADD_PROFILE( "Windows (wmv/asf)", "Windows" )
117     ADD_PROFILE( "PSP", "PSP")
118     ADD_PROFILE( "GSM", "GSM" )
119
120 #define ADD_VCODEC( name, fourcc ) ui.vCodecBox->addItem( name, QVariant( fourcc ) );
121     ADD_VCODEC( "MPEG-1", "mp1v" )
122     ADD_VCODEC( "MPEG-2", "mp2v" )
123     ADD_VCODEC( "MPEG-4", "mp4v" )
124     ADD_VCODEC( "DIVX 1" , "DIV1" )
125     ADD_VCODEC( "DIVX 2" , "DIV2" )
126     ADD_VCODEC( "DIVX 3" , "DIV3" )
127     ADD_VCODEC( "H-263", "H263" )
128     ADD_VCODEC( "H-264", "h264" )
129     ADD_VCODEC( "WMV1", "WMV1" )
130     ADD_VCODEC( "WMV2" , "WMV2" )
131     ADD_VCODEC( "M-JPEG", "MJPG" )
132     ADD_VCODEC( "Theora", "theo" )
133
134 #define ADD_ACODEC( name, fourcc ) ui.aCodecBox->addItem( name, QVariant( fourcc ) );
135     ADD_ACODEC( "MPEG Audio", "mpga" )
136     ADD_ACODEC( "MP3", "mp3" )
137     ADD_ACODEC( "MPEG 4 Audio ( AAC )", "mp4a" )
138     ADD_ACODEC( "A52/AC-3", "a52" )
139     ADD_ACODEC( "Vorbis", "vorb" )
140     ADD_ACODEC( "Flac", "flac" )
141     ADD_ACODEC( "Speex", "spx" )
142     ADD_ACODEC( "WAV", "s16l" )
143     ADD_ACODEC( "WMA", "wma" )
144
145 #define ADD_SCALING( factor ) ui.vScaleBox->addItem( factor );
146     ADD_SCALING( "0.25" )
147     ADD_SCALING( "0.5" )
148     ADD_SCALING( "0.75" )
149     ADD_SCALING( "1" )
150     ADD_SCALING( "1.25" )
151     ADD_SCALING( "1.5" )
152     ADD_SCALING( "1.75" )
153     ADD_SCALING( "2" )
154
155     ui.mrlEdit->setToolTip ( qtr( "Stream output string.\n"
156                 "This is automatically generated "
157                  "when you change the above settings,\n"
158                  "but you can update it manually." ) ) ;
159
160 //     /* Connect everything to the updateMRL function */
161  #define CB( x ) CONNECT( ui.x, toggled( bool ), this, updateMRL() );
162  #define CT( x ) CONNECT( ui.x, textChanged( const QString ), this, updateMRL() );
163  #define CS( x ) CONNECT( ui.x, valueChanged( int ), this, updateMRL() );
164  #define CC( x ) CONNECT( ui.x, currentIndexChanged( int ), this, updateMRL() );
165     /* Output */
166     CB( fileOutput ); CB( HTTPOutput ); CB( localOutput );
167     CB( RTPOutput ); CB( MMSHOutput ); CB( rawInput ); CB( UDPOutput );
168     CT( fileEdit ); CT( HTTPEdit ); CT( RTPEdit ); CT( MMSHEdit ); CT( UDPEdit );
169     CT( IcecastEdit ); CT( IcecastMountpointEdit ); CT( IcecastNamePassEdit );
170     CS( HTTPPort ); CS( RTPPort ); CS( RTPPort2 ); CS( MMSHPort ); CS( UDPPort );
171     /* Transcode */
172     CC( vCodecBox ); CC( subsCodecBox ); CC( aCodecBox ) ;
173     CB( transcodeVideo ); CB( transcodeAudio ); CB( transcodeSubs );
174     /*   CB( sOverlay ); */
175     CS( vBitrateSpin ); CS( aBitrateSpin ); CS( aChannelsSpin ); CC( vScaleBox );
176     /* Mux */
177     CB( PSMux ); CB( TSMux ); CB( MPEG1Mux ); CB( OggMux ); CB( ASFMux );
178     CB( MP4Mux ); CB( MOVMux ); CB( WAVMux ); CB( RAWMux ); CB( FLVMux );
179     /* Misc */
180     CB( soutAll ); CS( ttl ); CT( sapName ); CT( sapGroup );
181
182     CONNECT( ui.profileBox, activated( const QString & ), this, setOptions() );
183     CONNECT( ui.fileSelectButton, clicked() , this, fileBrowse()  );
184     CONNECT( ui.transcodeVideo, toggled( bool ),
185             this, setVTranscodeOptions( bool ) );
186     CONNECT( ui.transcodeAudio, toggled( bool ),
187             this, setATranscodeOptions( bool ) );
188     CONNECT( ui.transcodeSubs, toggled( bool ),
189             this, setSTranscodeOptions( bool ) );
190     CONNECT( ui.rawInput, toggled( bool ), this, setRawOptions( bool ) );
191
192     okButton = new QPushButton( qtr( "&Stream" ) );
193     QPushButton *cancelButton = new QPushButton( qtr( "&Cancel" ) );
194
195     okButton->setDefault( true );
196     ui.acceptButtonBox->addButton( okButton, QDialogButtonBox::AcceptRole );
197     ui.acceptButtonBox->addButton( cancelButton, QDialogButtonBox::RejectRole );
198
199     BUTTONACT( okButton, ok() );
200     BUTTONACT( cancelButton, cancel() );
201
202     CONNECT( ui.UDPOutput, toggled( bool ), this, changeUDPandRTPmess( bool ) );
203     CONNECT( ui.RTPOutput, clicked(bool), this, RTPtoggled( bool ) );
204
205     if( b_transcode_only ) toggleSout();
206 }
207
208 void SoutDialog::fileBrowse()
209 {
210     QString fileName = QFileDialog::getSaveFileName( this, qtr( "Save file" ), "",
211         qtr( "Containers (*.ps *.ts *.mpg *.ogg *.asf *.mp4 *.mov *.wav *.raw *.flv)" ) );
212     ui.fileEdit->setText( fileName );
213     updateMRL();
214 }
215
216 void SoutDialog::setVTranscodeOptions( bool b_trans )
217 {
218     ui.vCodecLabel->setEnabled( b_trans );
219     ui.vCodecBox->setEnabled( b_trans );
220     ui.vBitrateLabel->setEnabled( b_trans );
221     ui.vBitrateSpin->setEnabled( b_trans );
222     ui.vScaleLabel->setEnabled( b_trans );
223     ui.vScaleBox->setEnabled( b_trans );
224 }
225
226 void SoutDialog::setATranscodeOptions( bool b_trans )
227 {
228     ui.aCodecLabel->setEnabled( b_trans );
229     ui.aCodecBox->setEnabled( b_trans );
230     ui.aBitrateLabel->setEnabled( b_trans );
231     ui.aBitrateSpin->setEnabled( b_trans );
232     ui.aChannelsLabel->setEnabled( b_trans );
233     ui.aChannelsSpin->setEnabled( b_trans );
234 }
235
236 void SoutDialog::setSTranscodeOptions( bool b_trans )
237 {
238     ui.subsCodecBox->setEnabled( b_trans );
239     ui.subsOverlay->setEnabled( b_trans );
240 }
241
242 void SoutDialog::setRawOptions( bool b_raw )
243 {
244     ui.localOutput->setEnabled( !b_raw );
245     ui.HTTPOutput->setEnabled( !b_raw );
246     ui.MMSHOutput->setEnabled( !b_raw );
247     ui.UDPOutput->setEnabled( !b_raw );
248     ui.RTPOutput->setEnabled( !b_raw );
249     ui.IcecastOutput->setEnabled( !b_raw );
250     ui.UDPRTPLabel->setEnabled( !b_raw );
251
252     if( b_raw )
253         ui.tabWidget->setDisabled( true );
254     else
255         setOptions();
256 }
257
258 void SoutDialog::setOptions()
259 {
260     QString profileString =
261         ui.profileBox->itemData( ui.profileBox->currentIndex() ).toString();
262     msg_Dbg( p_intf, "Profile Used: %s",  qta( profileString ));
263     int index;
264
265 #define setProfile( muxName, hasVideo, vCodecName, hasAudio, aCodecName ) \
266     { \
267         ui.muxName ##Mux->setChecked( true ); \
268         \
269         ui.transcodeAudio->setChecked( hasAudio ); \
270         index = ui.aCodecBox->findText( vCodecName );  \
271         if( index >= 0 ) ui.aCodecBox->setCurrentIndex( index ); \
272         \
273         ui.transcodeVideo->setChecked( hasVideo ); \
274         index = ui.aCodecBox->findText( vCodecName );  \
275         if( index >=0 ) ui.vCodecBox->setCurrentIndex( index ); \
276     }
277
278     /* ADD HERE the profiles you want and need */
279     /* FIXME */
280     if( profileString == "IPod" ) setProfile( MP4, true, "mp4a", true, "mp4v" )
281     else if( profileString == "XBox" ) setProfile( ASF, true, "wma", true, "WMV2" )
282
283         /* If the profile is not a custom one, then disable the tabWidget */
284         if ( profileString == "Custom" )
285             ui.tabWidget->setEnabled( true );
286         else
287             ui.tabWidget->setDisabled( true );
288
289     /* Update the MRL !! */
290     updateMRL();
291 }
292
293 void SoutDialog::toggleSout()
294 {
295     //Toggle all the streaming options.
296 #define HIDEORSHOW(x) if( b_transcode_only ) x->hide(); else x->show();
297     HIDEORSHOW( ui.HTTPOutput ) ; HIDEORSHOW( ui.RTPOutput ) ; HIDEORSHOW( ui.MMSHOutput ) ; HIDEORSHOW( ui.UDPOutput ) ;
298     HIDEORSHOW( ui.HTTPEdit ) ; HIDEORSHOW( ui.RTPEdit ) ; HIDEORSHOW( ui.MMSHEdit ) ; HIDEORSHOW( ui.UDPEdit ) ;
299     HIDEORSHOW( ui.HTTPLabel ) ; HIDEORSHOW( ui.RTPLabel ) ; HIDEORSHOW( ui.MMSHLabel ) ; HIDEORSHOW( ui.UDPLabel ) ;
300     HIDEORSHOW( ui.HTTPPortLabel ) ; HIDEORSHOW( ui.RTPPortLabel ) ; HIDEORSHOW( ui.MMSHPortLabel ) ; HIDEORSHOW( ui.UDPPortLabel )
301     HIDEORSHOW( ui.HTTPPort ) ; HIDEORSHOW( ui.RTPPort ) ; HIDEORSHOW( ui.MMSHPort ) ; HIDEORSHOW( ui.UDPPort ) ; HIDEORSHOW( ui.RTPPortLabel2 ); HIDEORSHOW( ui.RTPPort2 ); HIDEORSHOW( ui.UDPRTPLabel )
302
303     HIDEORSHOW( ui.sap ); HIDEORSHOW( ui.sapName );
304     HIDEORSHOW( ui.sapGroup ); HIDEORSHOW( ui.sapGroupLabel );
305     HIDEORSHOW( ui.ttlLabel ); HIDEORSHOW( ui.ttl );
306
307     HIDEORSHOW( ui.IcecastOutput ); HIDEORSHOW( ui.IcecastEdit );
308     HIDEORSHOW( ui.IcecastNamePassEdit ); HIDEORSHOW( ui.IcecastMountpointEdit );
309     HIDEORSHOW( ui.IcecastPort ); HIDEORSHOW( ui.IcecastLabel );
310     HIDEORSHOW( ui.IcecastPortLabel );
311     HIDEORSHOW( ui.IcecastMountpointLabel ); HIDEORSHOW( ui.IcecastNameLabel );
312 #undef HIDEORSHOW
313
314     if( b_transcode_only ) okButton->setText( "&Save" );
315     else okButton->setText( "&Stream" );
316
317     setMinimumHeight( 500 );
318     resize( width(), sizeHint().height() );
319 }
320
321 void SoutDialog::changeUDPandRTPmess( bool b_udp )
322 {
323     ui.RTPEdit->setVisible( !b_udp );
324     ui.RTPLabel->setVisible( !b_udp );
325     ui.RTPPort->setVisible( !b_udp );
326     ui.RTPPortLabel->setVisible( !b_udp );
327     ui.UDPEdit->setVisible( b_udp );
328     ui.UDPLabel->setVisible( b_udp );
329     ui.UDPPortLabel->setText( b_udp ? qtr( "Port:") : qtr( "Audio Port:" ) );
330     ui.RTPPort2->setVisible( !b_udp );
331     ui.RTPPortLabel2->setVisible( !b_udp );
332 }
333
334 void SoutDialog::RTPtoggled( bool b_en )
335 {
336     if( !b_en )
337     {
338         if( ui.RTPPort->value() == ui.UDPPort->value() )
339         {
340             ui.UDPPort->setValue( ui.UDPPort->value() + 1 );
341         }
342
343         while( ui.RTPPort2->value() == ui.UDPPort->value() ||
344                 ui.RTPPort2->value() == ui.RTPPort->value() )
345         {
346             ui.RTPPort2->setValue( ui.RTPPort2->value() + 1 );
347         }
348     }
349     ui.sap->setEnabled( b_en );
350     ui.RTPLabel->setEnabled( b_en );
351     ui.RTPEdit->setEnabled( b_en );
352     ui.UDPOutput->setEnabled( b_en );
353     ui.UDPPort->setEnabled( b_en );
354     ui.UDPPortLabel->setEnabled( b_en );
355     ui.RTPPort2->setEnabled( b_en );
356     ui.RTPPortLabel2->setEnabled( b_en );
357 }
358
359 void SoutDialog::ok()
360 {
361     mrl = ui.mrlEdit->text();
362     accept();
363 }
364
365 void SoutDialog::cancel()
366 {
367     mrl.clear();
368     reject();
369 }
370
371 void SoutDialog::updateMRL()
372 {
373     sout_gui_descr_t sout;
374     memset( &sout, 0, sizeof( sout_gui_descr_t ) );
375     unsigned int counter = 0;
376
377     sout.b_local = ui.localOutput->isChecked();
378     sout.b_file = ui.fileOutput->isChecked();
379     sout.b_http = ui.HTTPOutput->isChecked();
380     sout.b_mms = ui.MMSHOutput->isChecked();
381     sout.b_icecast = ui.IcecastOutput->isChecked();
382     sout.b_rtp = ui.RTPOutput->isChecked();
383     sout.b_udp = ui.UDPOutput->isChecked();
384     sout.b_dump = ui.rawInput->isChecked();
385     sout.b_sap = ui.sap->isChecked();
386     sout.b_all_es = ui.soutAll->isChecked();
387     sout.psz_vcodec = strdup( qtu( ui.vCodecBox->itemData( ui.vCodecBox->currentIndex() ).toString() ) );
388     sout.psz_acodec = strdup( qtu( ui.aCodecBox->itemData( ui.aCodecBox->currentIndex() ).toString() ) );
389     sout.psz_scodec = strdup( qtu( ui.subsCodecBox->itemData( ui.subsCodecBox->currentIndex() ).toString() ) );
390     sout.psz_file = strdup( qtu( ui.fileEdit->text() ) );
391     sout.psz_http = strdup( qtu( ui.HTTPEdit->text() ) );
392     sout.psz_mms = strdup( qtu( ui.MMSHEdit->text() ) );
393     sout.psz_rtp = strdup( qtu( ui.RTPEdit->text() ) );
394     sout.psz_udp = strdup( qtu( ui.UDPEdit->text() ) );
395     sout.psz_icecast = strdup( qtu( ui.IcecastEdit->text() ) );
396     sout.sa_icecast.psz_username = strdup( qtu( ui.IcecastNamePassEdit->text() ) );
397     sout.sa_icecast.psz_password = strdup( qtu( ui.IcecastNamePassEdit->text() ) );
398     sout.psz_icecast_mountpoint = strdup( qtu( ui.IcecastMountpointEdit->text() ) );
399     sout.i_http = ui.HTTPPort->value();
400     sout.i_mms = ui.MMSHPort->value();
401     sout.i_rtp = ui.RTPPort->value();
402     sout.i_rtp_audio = sout.i_udp = ui.UDPPort->value();
403     sout.i_rtp_video = ui.RTPPort2->value();
404     sout.i_icecast = ui.IcecastPort->value();
405     sout.i_ab = ui.aBitrateSpin->value();
406     sout.i_vb = ui.vBitrateSpin->value();
407     sout.i_channels = ui.aChannelsSpin->value();
408     sout.f_scale = atof( qta( ui.vScaleBox->currentText() ) );
409     sout.psz_group = strdup( qtu( ui.sapGroup->text() ) );
410     sout.psz_name = strdup( qtu( ui.sapName->text() ) );
411
412     if ( sout.b_local ) counter++ ;
413     if ( sout.b_file ) counter++ ;
414     if ( sout.b_http ) counter++ ;
415     if ( sout.b_mms ) counter++ ;
416     if ( sout.b_rtp ) counter++ ;
417     if ( sout.b_udp ) counter ++;
418     if ( sout.b_icecast ) counter ++;
419
420 #define SMUX( x, txt ) if( ui.x->isChecked() ) sout.psz_mux = strdup( txt );
421     SMUX( PSMux, "ps" );
422     SMUX( TSMux, "ts" );
423     SMUX( MPEG1Mux, "mpeg" );
424     SMUX( OggMux, "ogg" );
425     SMUX( ASFMux, "asf" );
426     SMUX( MP4Mux, "mp4" );
427     SMUX( MOVMux, "mov" );
428     SMUX( WAVMux, "wav" );
429     SMUX( RAWMux, "raw" );
430     SMUX( FLVMux, "flv" );
431     SMUX( MKVMux, "mkv" );
432
433     bool trans = false;
434     bool more = false;
435
436     if ( ui.transcodeVideo->isChecked() || ui.transcodeAudio->isChecked()
437          && !ui.rawInput->isChecked() /*demuxdump speciality*/ )
438     {
439         if ( ui.transcodeVideo->isChecked() )
440         {
441             mrl = ":sout=#transcode{";
442             mrl.append( "vcodec=" );
443             mrl.append( sout.psz_vcodec );
444             mrl.append( "," );
445             mrl.append( "vb=" );
446             mrl.append( QString::number( sout.i_vb,10 ) );
447             mrl.append( "," );
448             mrl.append( "scale=" );
449             mrl.append( QString::number( sout.f_scale ) );
450             trans = true;
451         }
452
453         if ( ui.transcodeAudio->isChecked() )
454         {
455             if ( trans )
456             {
457                 mrl.append( "," );
458             }
459             else
460             {
461                 mrl = ":sout=#transcode{";
462             }
463             mrl.append( "acodec=" );
464             mrl.append( sout.psz_acodec );
465             mrl.append( "," );
466             mrl.append( "ab=" );
467             mrl.append( QString::number( sout.i_ab,10 ) );
468             mrl.append( "," );
469             mrl.append( "channels=" );
470             mrl.append( QString::number( sout.i_channels,10 ) );
471             trans = true;
472         }
473         mrl.append( "}" );
474     }
475
476     /* Special case for demuxdump */
477     if ( sout.b_file && sout.b_dump )
478     {
479         mrl = ":demux=dump :demuxdump-file=";
480         mrl.append( sout.psz_file );
481     }
482     else
483
484
485     /* Protocol output */
486     if ( sout.b_local || sout.b_file || sout.b_http ||
487          sout.b_mms || sout.b_rtp || sout.b_udp || sout.b_icecast )
488     {
489
490 #define ISMORE() if ( more ) mrl.append( "," )
491 #define ATLEASTONE() if ( counter ) mrl.append( "dst=" )
492
493 #define CHECKMUX() \
494        if( sout.psz_mux ) \
495        {                  \
496          mrl.append( ",mux=");\
497          mrl.append( sout.psz_mux ); \
498        }
499
500         if ( trans )
501             mrl.append( ":" );
502         else
503             mrl = ":sout=#";
504
505         if ( counter )
506             mrl.append( "duplicate{" );
507
508         if ( sout.b_local )
509         {
510             ISMORE();
511             ATLEASTONE();
512             mrl.append( "display" );
513             more = true;
514         }
515
516         if ( sout.b_file )
517         {
518             ISMORE();
519             ATLEASTONE();
520             mrl.append( "std{access=file" );
521             CHECKMUX();
522             mrl.append( ",dst=" );
523             mrl.append( sout.psz_file );
524             mrl.append( "}" );
525             more = true;
526         }
527
528         if ( sout.b_http )
529         {
530             ISMORE();
531             ATLEASTONE();
532             mrl.append( "std{access=http" );
533             CHECKMUX();
534             mrl.append( ",dst=" );
535             mrl.append( sout.psz_http );
536             mrl.append( ":" );
537             mrl.append( QString::number( sout.i_http,10 ) );
538             mrl.append( "}" );
539             more = true;
540         }
541
542         if ( sout.b_mms )
543         {
544             ISMORE();
545             ATLEASTONE();
546             mrl.append( "std{access=mmsh" );
547             CHECKMUX();
548             mrl.append( ",dst=" );
549             mrl.append( sout.psz_mms );
550             mrl.append( ":" );
551             mrl.append( QString::number( sout.i_mms,10 ) );
552             mrl.append( "}" );
553             more = true;
554         }
555
556         if ( sout.b_rtp )
557         {
558             ISMORE();
559             ATLEASTONE();
560             if ( sout.b_udp )
561             {
562                 mrl.append( "std{access=udp" );
563                 CHECKMUX();
564                 mrl.append( ",dst=" );
565                 mrl.append( sout.psz_udp );
566                 mrl.append( ":" );
567                 mrl.append( QString::number( sout.i_udp,10 ) );
568             }
569             else
570             {
571                 mrl.append( "rtp{" );
572                 mrl.append( "dst=" );
573                 mrl.append( sout.psz_rtp );
574                 CHECKMUX();
575                 mrl.append( ",port=" );
576                 mrl.append( QString::number( sout.i_rtp,10 ) );
577                 if( !sout.psz_mux || strncmp( sout.psz_mux, "ts", 2 ) )
578                 {
579                     mrl.append( ",port-audio=" );
580                     mrl.append( QString::number( sout.i_rtp_audio, 10 ) );
581                     mrl.append( ",port-video=" );
582                     mrl.append( QString::number( sout.i_rtp_video, 10 ) );
583                 }
584             }
585
586             /* SAP */
587             if ( sout.b_sap )
588             {
589                 mrl.append( ",sap," );
590                 mrl.append( "group=\"" );
591                 mrl.append( sout.psz_group );
592                 mrl.append( "\"," );
593                 mrl.append( "name=\"" );
594                 mrl.append( sout.psz_name );
595                 mrl.append( "\"" );
596             }
597
598             mrl.append( "}" );
599             more = true;
600         }
601
602         if( sout.b_icecast )
603         {
604             ISMORE();
605             ATLEASTONE();
606             mrl.append( "std{access=shout,mux=ogg" );
607             mrl.append( ",dst=" );
608             mrl.append( sout.sa_icecast.psz_username );
609             mrl.append( "@" );
610             mrl.append( sout.psz_icecast );
611             mrl.append( ":" );
612             mrl.append( QString::number( sout.i_icecast, 10 ) );
613             mrl.append( "/" );
614             mrl.append( sout.psz_icecast_mountpoint );
615             mrl.append( "}" );
616             more = true;
617         }
618
619         if ( counter )
620         {
621             mrl.append( "}" );
622         }
623     }
624
625 #undef CHECKMUX
626
627     if ( sout.b_all_es )
628         mrl.append( ":sout-all" );
629
630     ui.mrlEdit->setText( mrl );
631     free( sout.psz_acodec ); free( sout.psz_vcodec ); free( sout.psz_scodec );
632     free( sout.psz_file );free( sout.psz_http ); free( sout.psz_mms );
633     free( sout.psz_rtp ); free( sout.psz_udp ); free( sout.psz_mux );
634     free( sout.psz_name ); free( sout.psz_group );
635     free( sout.psz_icecast ); free( sout.psz_icecast_mountpoint );
636     free( sout.sa_icecast.psz_password ); free( sout.sa_icecast.psz_username );
637 }