interface_widgets.cpp 16.3 KB
Newer Older
1
/*****************************************************************************
2
 * interface_widgets.cpp : Custom widgets for the main interface
3
 ****************************************************************************
4
 * Copyright (C) 2006-2008 the VideoLAN team
5
 * $Id$
6 7
 *
 * Authors: Clément Stenac <zorglub@videolan.org>
8
 *          Jean-Baptiste Kempf <jb@videolan.org>
9
 *          Rafaël Carré <funman@videolanorg>
10
 *          Ilkka Ollakka <ileoo@videolan.org>
11 12 13 14
 *
 * 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
 * the Free Software Foundation; either version 2 of the License, or
15
 * ( at your option ) any later version.
16 17 18 19 20 21 22 23 24 25
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
 *****************************************************************************/
26

27 28 29
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
30

31
#include "components/interface_widgets.hpp"
32 33 34 35

#include "menus.hpp"             /* Popup menu on bgWidget */

#include <vlc_vout.h>
36

Clément Stenac's avatar
Clément Stenac committed
37
#include <QLabel>
38
#include <QToolButton>
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
39 40
#include <QPalette>
#include <QResizeEvent>
41
#include <QDate>
42 43
#include <QMenu>
#include <QWidgetAction>
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
44

45 46 47 48
#ifdef Q_WS_X11
# include <X11/Xlib.h>
# include <qx11info_x11.h>
#endif
49

50 51
#include <math.h>

52 53 54 55
/**********************************************************************
 * Video Widget. A simple frame on which video is drawn
 * This class handles resize issues
 **********************************************************************/
56

57
VideoWidget::VideoWidget( intf_thread_t *_p_i ) : QFrame( NULL ), p_intf( _p_i )
58
{
59
    /* Init */
60
    reparentable = NULL;
61 62
    videoSize.rwidth() = -1;
    videoSize.rheight() = -1;
63 64 65 66

    hide();

    /* Set the policy to expand in both directions */
67
//    setSizePolicy( QSizePolicy::Expanding, QSizePolicy::Expanding );
68

69 70 71
    /* Indicates that the widget wants to draw directly onto the screen.
       Widgets with this attribute set do not participate in composition
       management */
72
    setAttribute( Qt::WA_PaintOnScreen, true );
73 74 75 76

    layout = new QHBoxLayout( this );
    layout->setContentsMargins( 0, 0, 0, 0 );
    setLayout( layout );
77 78
}

79 80 81
void VideoWidget::paintEvent(QPaintEvent *ev)
{
    QFrame::paintEvent(ev);
82 83 84
#ifdef Q_WS_X11
    XFlush( QX11Info::display() );
#endif
85 86
}

87 88
VideoWidget::~VideoWidget()
{
89
    /* Ensure we are not leaking the video output. This would crash. */
90
    assert( reparentable == NULL );
91 92
}

93
/**
94
 * Request the video to avoid the conflicts
95
 **/
96
WId VideoWidget::request( int *pi_x, int *pi_y,
97 98
                          unsigned int *pi_width, unsigned int *pi_height,
                          bool b_keep_size )
99
{
100
    msg_Dbg( p_intf, "Video was requested %i, %i", *pi_x, *pi_y );
101

102 103 104 105 106
    if( reparentable != NULL )
    {
        msg_Dbg( p_intf, "embedded video already in use" );
        return NULL;
    }
107 108 109 110 111 112
    if( b_keep_size )
    {
        *pi_width  = size().width();
        *pi_height = size().height();
    }

113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
    /* The Qt4 UI needs a fixed a widget ("this"), so that the parent layout is
     * not messed up when we the video is reparented. Hence, we create an extra
     * reparentable widget, that will be within the VideoWidget in windowed
     * mode, and within the root window (NULL parent) in full-screen mode.
     */
    reparentable = new QWidget();
    QLayout *innerLayout = new QHBoxLayout( reparentable );
    innerLayout->setContentsMargins( 0, 0, 0, 0 );

    /* The owner of the video window needs a stable handle (WinId). Reparenting
     * in Qt4-X11 changes the WinId of the widget, so we need to create another
     * dummy widget that stays within the reparentable widget. */
    QWidget *stable = new QWidget();
    QPalette plt = palette();
    plt.setColor( QPalette::Window, Qt::black );
    stable->setPalette( plt );
    stable->setAutoFillBackground(true);
    stable->setAttribute( Qt::WA_PaintOnScreen, true );

    innerLayout->addWidget( stable );

    reparentable->setLayout( innerLayout );
    layout->addWidget( reparentable );
    updateGeometry();

138
#ifndef NDEBUG
139 140
    msg_Dbg( p_intf, "embedded video ready (handle %p)",
             (void *)stable->winId() );
141
#endif
142
    return stable->winId();
143 144
}

145
/* Set the Widget to the correct Size */
146
/* Function has to be called by the parent
Rémi Denis-Courmont's avatar
typo  
Rémi Denis-Courmont committed
147
   Parent has to care about resizing itself */
148
void VideoWidget::SetSizing( unsigned int w, unsigned int h )
149
{
150 151 152
    msg_Dbg( p_intf, "Video is resizing to: %i %i", w, h );
    videoSize.rwidth() = w;
    videoSize.rheight() = h;
153
    if( !isVisible() ) show();
154
    updateGeometry(); // Needed for deinterlace
155 156
}

157 158 159 160
void VideoWidget::SetFullScreen( bool b_fs )
{
    const Qt::WindowStates curstate = reparentable->windowState();
    Qt::WindowStates newstate = curstate;
161
    Qt::WindowFlags  newflags = reparentable->windowFlags();
162 163

    if( b_fs )
164
    {
165
        newstate |= Qt::WindowFullScreen;
166 167
        newflags |= Qt::WindowStaysOnTopHint;
    }
168
    else
169
    {
170
        newstate &= ~Qt::WindowFullScreen;
171 172
        newflags &= ~Qt::WindowStaysOnTopHint;
    }
173 174 175 176 177 178
    if( newstate == curstate )
        return; /* no changes needed */

    if( b_fs )
    {   /* Go full-screen */
        reparentable->setWindowState( newstate );
179
        reparentable->setParent( NULL );
180
        reparentable->setWindowFlags( newflags );
181 182 183 184
        reparentable->show();
    }
    else
    {   /* Go windowed */
185
        reparentable->setWindowFlags( newflags );
186
        layout->addWidget( reparentable );
187
        reparentable->setWindowState( newstate );
188 189 190
    }
}

191
void VideoWidget::release( void )
192
{
Rémi Denis-Courmont's avatar
Typo  
Rémi Denis-Courmont committed
193
    msg_Dbg( p_intf, "Video is not needed anymore" );
194 195 196
    //layout->removeWidget( reparentable );
    delete reparentable;
    reparentable = NULL;
197 198
    videoSize.rwidth() = 0;
    videoSize.rheight() = 0;
199
    updateGeometry();
200
    hide();
201
}
202

203 204 205 206
QSize VideoWidget::sizeHint() const
{
    return videoSize;
}
207

208 209
/**********************************************************************
 * Background Widget. Show a simple image background. Currently,
210
 * it's album art if present or cone.
211
 **********************************************************************/
212 213
#define ICON_SIZE 128
#define MAX_BG_SIZE 400
214
#define MIN_BG_SIZE 128
215

216 217
BackgroundWidget::BackgroundWidget( intf_thread_t *_p_i )
                 :QWidget( NULL ), p_intf( _p_i )
218
{
219
    /* We should use that one to take the more size it can */
220
    setSizePolicy( QSizePolicy::Expanding, QSizePolicy::Expanding);
221

222
    /* A dark background */
Clément Stenac's avatar
Clément Stenac committed
223
    setAutoFillBackground( true );
224
    plt = palette();
225 226 227 228
    plt.setColor( QPalette::Active, QPalette::Window , Qt::black );
    plt.setColor( QPalette::Inactive, QPalette::Window , Qt::black );
    setPalette( plt );

229
    /* A cone in the middle */
230
    label = new QLabel;
231
    label->setMargin( 5 );
232 233
    label->setMaximumHeight( MAX_BG_SIZE );
    label->setMaximumWidth( MAX_BG_SIZE );
234 235
    label->setMinimumHeight( MIN_BG_SIZE );
    label->setMinimumWidth( MIN_BG_SIZE );
236
    if( QDate::currentDate().dayOfYear() >= 354 )
237
        label->setPixmap( QPixmap( ":/logo/vlc128-christmas.png" ) );
238
    else
239
        label->setPixmap( QPixmap( ":/logo/vlc128.png" ) );
240

241 242 243 244
    QGridLayout *backgroundLayout = new QGridLayout( this );
    backgroundLayout->addWidget( label, 0, 1 );
    backgroundLayout->setColumnStretch( 0, 1 );
    backgroundLayout->setColumnStretch( 2, 1 );
245

246
    CONNECT( THEMIM->getIM(), artChanged( QString ),
247
             this, updateArt( const QString& ) );
248 249
}

Clément Stenac's avatar
Clément Stenac committed
250
BackgroundWidget::~BackgroundWidget()
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
251
{}
Ilkka Ollakka's avatar
 
Ilkka Ollakka committed
252

253 254 255 256 257 258 259 260
void BackgroundWidget::resizeEvent( QResizeEvent * event )
{
    if( event->size().height() <= MIN_BG_SIZE )
        label->hide();
    else
        label->show();
}

261
void BackgroundWidget::updateArt( const QString& url )
262
{
263
    if( url.isEmpty() )
Ilkka Ollakka's avatar
 
Ilkka Ollakka committed
264
    {
265
        if( QDate::currentDate().dayOfYear() >= 354 )
266
            label->setPixmap( QPixmap( ":/logo/vlc128-christmas.png" ) );
267
        else
268
            label->setPixmap( QPixmap( ":/logo/vlc128.png" ) );
Ilkka Ollakka's avatar
 
Ilkka Ollakka committed
269
    }
270
    else
Ilkka Ollakka's avatar
 
Ilkka Ollakka committed
271
    {
272
        label->setPixmap( QPixmap( url ) );
Ilkka Ollakka's avatar
 
Ilkka Ollakka committed
273
    }
274 275
}

276 277 278
void BackgroundWidget::contextMenuEvent( QContextMenuEvent *event )
{
    QVLCMenu::PopupMenu( p_intf, true );
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
279
    event->accept();
280
}
281

Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
282
#if 0
283 284 285
#include <QPushButton>
#include <QHBoxLayout>

Clément Stenac's avatar
Clément Stenac committed
286 287 288 289
/**********************************************************************
 * Visualization selector panel
 **********************************************************************/
VisualSelector::VisualSelector( intf_thread_t *_p_i ) :
290
                                QFrame( NULL ), p_intf( _p_i )
Clément Stenac's avatar
Clément Stenac committed
291 292
{
    QHBoxLayout *layout = new QHBoxLayout( this );
293
    layout->setMargin( 0 );
Clément Stenac's avatar
Clément Stenac committed
294
    QPushButton *prevButton = new QPushButton( "Prev" );
295
    QPushButton *nextButton = new QPushButton( "Next" );
Clément Stenac's avatar
Clément Stenac committed
296 297
    layout->addWidget( prevButton );
    layout->addWidget( nextButton );
Clément Stenac's avatar
Clément Stenac committed
298

299
    layout->addStretch( 10 );
300
    layout->addWidget( new QLabel( qtr( "Current visualization" ) ) );
Clément Stenac's avatar
Clément Stenac committed
301 302 303 304 305 306 307

    current = new QLabel( qtr( "None" ) );
    layout->addWidget( current );

    BUTTONACT( prevButton, prev() );
    BUTTONACT( nextButton, next() );

Clément Stenac's avatar
Clément Stenac committed
308
    setLayout( layout );
309
    setMaximumHeight( 35 );
Clément Stenac's avatar
Clément Stenac committed
310 311 312
}

VisualSelector::~VisualSelector()
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
313
{}
Clément Stenac's avatar
Clément Stenac committed
314

Clément Stenac's avatar
Clément Stenac committed
315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333
void VisualSelector::prev()
{
    char *psz_new = aout_VisualPrev( p_intf );
    if( psz_new )
    {
        current->setText( qfu( psz_new ) );
        free( psz_new );
    }
}

void VisualSelector::next()
{
    char *psz_new = aout_VisualNext( p_intf );
    if( psz_new )
    {
        current->setText( qfu( psz_new ) );
        free( psz_new );
    }
}
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
334
#endif
Clément Stenac's avatar
Clément Stenac committed
335

Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
336 337 338
SpeedLabel::SpeedLabel( intf_thread_t *_p_intf, const QString& text,
                        QWidget *parent )
           : QLabel( text, parent ), p_intf( _p_intf )
339
{
340
    setToolTip( qtr( "Current playback speed.\nClick to adjust" ) );
341 342

    /* Create the Speed Control Widget */
343
    speedControl = new SpeedControlWidget( p_intf, this );
344 345 346 347 348 349 350 351 352
    speedControlMenu = new QMenu( this );

    QWidgetAction *widgetAction = new QWidgetAction( speedControl );
    widgetAction->setDefaultWidget( speedControl );
    speedControlMenu->addAction( widgetAction );

    /* Change the SpeedRate in the Status Bar */
    CONNECT( THEMIM->getIM(), rateChanged( int ), this, setRate( int ) );

353
    CONNECT( THEMIM, inputChanged( input_thread_t * ),
354
             speedControl, activateOnState() );
355

356
}
357 358 359 360 361
SpeedLabel::~SpeedLabel()
{
        delete speedControl;
        delete speedControlMenu;
}
362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380
/****************************************************************************
 * Small right-click menu for rate control
 ****************************************************************************/
void SpeedLabel::showSpeedMenu( QPoint pos )
{
    speedControlMenu->exec( QCursor::pos() - pos
                          + QPoint( 0, height() ) );
}

void SpeedLabel::setRate( int rate )
{
    QString str;
    str.setNum( ( 1000 / (double)rate ), 'f', 2 );
    str.append( "x" );
    setText( str );
    setToolTip( str );
    speedControl->updateControls( rate );
}

381 382 383
/**********************************************************************
 * Speed control widget
 **********************************************************************/
384 385
SpeedControlWidget::SpeedControlWidget( intf_thread_t *_p_i, QWidget *_parent )
                    : QFrame( _parent ), p_intf( _p_i )
386
{
387
    QSizePolicy sizePolicy( QSizePolicy::Maximum, QSizePolicy::Fixed );
388 389
    sizePolicy.setHorizontalStretch( 0 );
    sizePolicy.setVerticalStretch( 0 );
390

Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
391
    speedSlider = new QSlider( this );
392 393 394 395 396
    speedSlider->setSizePolicy( sizePolicy );
    speedSlider->setMaximumSize( QSize( 80, 200 ) );
    speedSlider->setOrientation( Qt::Vertical );
    speedSlider->setTickPosition( QSlider::TicksRight );

397
    speedSlider->setRange( -34, 34 );
398 399
    speedSlider->setSingleStep( 1 );
    speedSlider->setPageStep( 1 );
400
    speedSlider->setTickInterval( 17 );
401

402
    CONNECT( speedSlider, valueChanged( int ), this, updateRate( int ) );
403

404
    QToolButton *normalSpeedButton = new QToolButton( this );
405
    normalSpeedButton->setMaximumSize( QSize( 26, 20 ) );
406
    normalSpeedButton->setAutoRaise( true );
407
    normalSpeedButton->setText( "1x" );
408
    normalSpeedButton->setToolTip( qtr( "Revert to normal play speed" ) );
409

410
    CONNECT( normalSpeedButton, clicked(), this, resetRate() );
411

Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
412
    QVBoxLayout *speedControlLayout = new QVBoxLayout( this );
413 414
    speedControlLayout->setLayoutMargins( 4, 4, 4, 4, 4 );
    speedControlLayout->setSpacing( 4 );
415 416
    speedControlLayout->addWidget( speedSlider );
    speedControlLayout->addWidget( normalSpeedButton );
417

418 419
    activateOnState();
}
420

421
void SpeedControlWidget::activateOnState()
422
{
423
    speedSlider->setEnabled( THEMIM->getIM()->hasInput() );
424 425
}

426 427
void SpeedControlWidget::updateControls( int rate )
{
428 429 430 431 432
    if( speedSlider->isSliderDown() )
    {
        //We don't want to change anything if the user is using the slider
        return;
    }
433

434
    double value = 17 * log( (double)INPUT_RATE_DEFAULT / rate ) / log( 2 );
435
    int sliderValue = (int) ( ( value > 0 ) ? value + .5 : value - .5 );
436

437
    if( sliderValue < speedSlider->minimum() )
438
    {
439
        sliderValue = speedSlider->minimum();
440
    }
441
    else if( sliderValue > speedSlider->maximum() )
442
    {
443
        sliderValue = speedSlider->maximum();
444
    }
445

446 447 448 449
    //Block signals to avoid feedback loop
    speedSlider->blockSignals( true );
    speedSlider->setValue( sliderValue );
    speedSlider->blockSignals( false );
450 451 452 453
}

void SpeedControlWidget::updateRate( int sliderValue )
{
454
    double speed = pow( 2, (double)sliderValue / 17 );
455
    int rate = INPUT_RATE_DEFAULT / speed;
456

457
    THEMIM->getIM()->setRate(rate);
458 459 460 461
}

void SpeedControlWidget::resetRate()
{
462
    THEMIM->getIM()->setRate( INPUT_RATE_DEFAULT );
463
}
464

465
CoverArtLabel::CoverArtLabel( QWidget *parent, intf_thread_t *_p_i )
466
              : QLabel( parent ), p_intf( _p_i )
467 468
{
    setContextMenuPolicy( Qt::ActionsContextMenu );
469
    CONNECT( this, updateRequested(), this, askForUpdate() );
470
    CONNECT( THEMIM->getIM(), artChanged( QString ),
471
             this, showArtUpdate( const QString& ) );
472 473 474 475 476 477

    setMinimumHeight( 128 );
    setMinimumWidth( 128 );
    setMaximumHeight( 128 );
    setMaximumWidth( 128 );
    setScaledContents( true );
478

479 480
    QList< QAction* > artActions = actions();
    QAction *action = new QAction( qtr( "Download cover art" ), this );
481
    CONNECT( action, triggered(), this, askForUpdate() );
482
    addAction( action );
483

484
    showArtUpdate( "" );
485 486
}

487 488 489 490 491 492 493
CoverArtLabel::~CoverArtLabel()
{
    QList< QAction* > artActions = actions();
    foreach( QAction *act, artActions )
        removeAction( act );
}

494
void CoverArtLabel::showArtUpdate( const QString& url )
495
{
496 497
    QPixmap pix;
    if( !url.isEmpty()  && pix.load( url ) )
498
    {
499 500 501 502 503
        setPixmap( pix );
    }
    else
    {
        setPixmap( QPixmap( ":/noart.png" ) );
504 505 506
    }
}

507
void CoverArtLabel::askForUpdate()
508
{
509
    THEMIM->getIM()->requestArtUpdate();
510 511
}

512 513 514 515 516 517 518 519
TimeLabel::TimeLabel( intf_thread_t *_p_intf  ) :QLabel(), p_intf( _p_intf )
{
   b_remainingTime = false;
   setText( " --:--/--:-- " );
   setAlignment( Qt::AlignRight | Qt::AlignVCenter );
   setToolTip( qtr( "Toggle between elapsed and remaining time" ) );


520 521
   CONNECT( THEMIM->getIM(), cachingChanged( float ),
            this, setCaching( float ) );
522 523 524 525 526 527
   CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
             this, setDisplayPosition( float, int, int ) );
}

void TimeLabel::setDisplayPosition( float pos, int time, int length )
{
528
    if( pos == -1.f )
529 530 531 532
    {
        setText( " --:--/--:-- " );
        return;
    }
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
533

534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551
    char psz_length[MSTRTIME_MAX_SIZE], psz_time[MSTRTIME_MAX_SIZE];
    secstotimestr( psz_length, length );
    secstotimestr( psz_time, ( b_remainingTime && length ) ? length - time
                                                           : time );

    QString timestr;
    timestr.sprintf( "%s/%s", psz_time,
                            ( !length && time ) ? "--:--" : psz_length );

    /* Add a minus to remaining time*/
    if( b_remainingTime && length ) setText( " -"+timestr+" " );
    else setText( " "+timestr+" " );
}

void TimeLabel::toggleTimeDisplay()
{
    b_remainingTime = !b_remainingTime;
}
552

553 554
void TimeLabel::setCaching( float f_cache )
{
555 556 557
    QString amount;
    amount.setNum( (int)(100 * f_cache) );
    msg_Dbg( p_intf, "New caching: %d", (int)(100*f_cache));
558
    setText( "Buff: " + amount + "%" );
559 560
}

561