interface_widgets.cpp
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:15k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * interface_widgets.cpp : Custom widgets for the main interface
  3.  ****************************************************************************
  4.  * Copyright (C) 2006-2008 the VideoLAN team
  5.  * $Id: 5c38426b96b7907ebcb9cf39178f7d6f76005d0f $
  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.  * This program is free software; you can redistribute it and/or modify
  13.  * it under the terms of the GNU General Public License as published by
  14.  * the Free Software Foundation; either version 2 of the License, or
  15.  * ( at your option ) any later version.
  16.  *
  17.  * This program is distributed in the hope that it will be useful,
  18.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  19.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  20.  * GNU General Public License for more details.
  21.  *
  22.  * You should have received a copy of the GNU General Public License
  23.  * along with this program; if not, write to the Free Software
  24.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  25.  *****************************************************************************/
  26. #ifdef HAVE_CONFIG_H
  27. # include "config.h"
  28. #endif
  29. #include "components/interface_widgets.hpp"
  30. #include "menus.hpp"             /* Popup menu on bgWidget */
  31. #include <vlc_vout.h>
  32. #include <QLabel>
  33. #include <QToolButton>
  34. #include <QPalette>
  35. #include <QResizeEvent>
  36. #include <QDate>
  37. #include <QMenu>
  38. #include <QWidgetAction>
  39. #ifdef Q_WS_X11
  40. # include <X11/Xlib.h>
  41. # include <qx11info_x11.h>
  42. static void videoSync( void )
  43. {
  44.     /* Make sure the X server has processed all requests.
  45.      * This protects other threads using distinct connections from getting
  46.      * the video widget window in an inconsistent states. */
  47.     XSync( QX11Info::display(), False );
  48. }
  49. #else
  50. # define videoSync() (void)0
  51. #endif
  52. #include <math.h>
  53. /**********************************************************************
  54.  * Video Widget. A simple frame on which video is drawn
  55.  * This class handles resize issues
  56.  **********************************************************************/
  57. VideoWidget::VideoWidget( intf_thread_t *_p_i ) : QFrame( NULL ), p_intf( _p_i )
  58. {
  59.     /* Init */
  60.     p_vout = NULL;
  61.     videoSize.rwidth() = -1;
  62.     videoSize.rheight() = -1;
  63.     hide();
  64.     /* Set the policy to expand in both directions */
  65. //    setSizePolicy( QSizePolicy::Expanding, QSizePolicy::Expanding );
  66.     /* Black background is more coherent for a Video Widget */
  67.     QPalette plt =  palette();
  68.     plt.setColor( QPalette::Window, Qt::black );
  69.     setPalette( plt );
  70.     setAutoFillBackground(true);
  71.     /* Indicates that the widget wants to draw directly onto the screen.
  72.        Widgets with this attribute set do not participate in composition
  73.        management */
  74.     setAttribute( Qt::WA_PaintOnScreen, true );
  75. }
  76. void VideoWidget::paintEvent(QPaintEvent *ev)
  77. {
  78.     QFrame::paintEvent(ev);
  79. #ifdef Q_WS_X11
  80.     XFlush( QX11Info::display() );
  81. #endif
  82. }
  83. VideoWidget::~VideoWidget()
  84. {
  85.     /* Ensure we are not leaking the video output. This would crash. */
  86.     assert( !p_vout );
  87. }
  88. /**
  89.  * Request the video to avoid the conflicts
  90.  **/
  91. WId VideoWidget::request( vout_thread_t *p_nvout, int *pi_x, int *pi_y,
  92.                           unsigned int *pi_width, unsigned int *pi_height,
  93.                           bool b_keep_size )
  94. {
  95.     msg_Dbg( p_intf, "Video was requested %i, %i", *pi_x, *pi_y );
  96.     if( b_keep_size )
  97.     {
  98.         *pi_width  = size().width();
  99.         *pi_height = size().height();
  100.     }
  101.     if( p_vout )
  102.     {
  103.         msg_Dbg( p_intf, "embedded video already in use" );
  104.         return NULL;
  105.     }
  106.     p_vout = p_nvout;
  107.     videoSync();
  108. #ifndef NDEBUG
  109.     msg_Dbg( p_intf, "embedded video ready (handle %p)", (void *)winId() );
  110. #endif
  111.     return winId();
  112. }
  113. /* Set the Widget to the correct Size */
  114. /* Function has to be called by the parent
  115.    Parent has to care about resizing himself*/
  116. void VideoWidget::SetSizing( unsigned int w, unsigned int h )
  117. {
  118.     msg_Dbg( p_intf, "Video is resizing to: %i %i", w, h );
  119.     videoSize.rwidth() = w;
  120.     videoSize.rheight() = h;
  121.     if( !isVisible() ) show();
  122.     updateGeometry(); // Needed for deinterlace
  123.     videoSync();
  124. }
  125. void VideoWidget::release( void )
  126. {
  127.     msg_Dbg( p_intf, "Video is not needed anymore" );
  128.     p_vout = NULL;
  129.     videoSize.rwidth() = 0;
  130.     videoSize.rheight() = 0;
  131.     updateGeometry();
  132.     hide();
  133. }
  134. QSize VideoWidget::sizeHint() const
  135. {
  136.     return videoSize;
  137. }
  138. /**********************************************************************
  139.  * Background Widget. Show a simple image background. Currently,
  140.  * it's album art if present or cone.
  141.  **********************************************************************/
  142. #define ICON_SIZE 128
  143. #define MAX_BG_SIZE 400
  144. #define MIN_BG_SIZE 128
  145. BackgroundWidget::BackgroundWidget( intf_thread_t *_p_i )
  146.                  :QWidget( NULL ), p_intf( _p_i )
  147. {
  148.     /* We should use that one to take the more size it can */
  149.     setSizePolicy( QSizePolicy::Expanding, QSizePolicy::Expanding);
  150.     /* A dark background */
  151.     setAutoFillBackground( true );
  152.     plt = palette();
  153.     plt.setColor( QPalette::Active, QPalette::Window , Qt::black );
  154.     plt.setColor( QPalette::Inactive, QPalette::Window , Qt::black );
  155.     setPalette( plt );
  156.     /* A cone in the middle */
  157.     label = new QLabel;
  158.     label->setMargin( 5 );
  159.     label->setMaximumHeight( MAX_BG_SIZE );
  160.     label->setMaximumWidth( MAX_BG_SIZE );
  161.     label->setMinimumHeight( MIN_BG_SIZE );
  162.     label->setMinimumWidth( MIN_BG_SIZE );
  163.     if( QDate::currentDate().dayOfYear() >= 354 )
  164.         label->setPixmap( QPixmap( ":/vlc128-christmas.png" ) );
  165.     else
  166.         label->setPixmap( QPixmap( ":/vlc128.png" ) );
  167.     QGridLayout *backgroundLayout = new QGridLayout( this );
  168.     backgroundLayout->addWidget( label, 0, 1 );
  169.     backgroundLayout->setColumnStretch( 0, 1 );
  170.     backgroundLayout->setColumnStretch( 2, 1 );
  171.     CONNECT( THEMIM->getIM(), artChanged( QString ),
  172.              this, updateArt( const QString& ) );
  173. }
  174. BackgroundWidget::~BackgroundWidget()
  175. {}
  176. void BackgroundWidget::resizeEvent( QResizeEvent * event )
  177. {
  178.     if( event->size().height() <= MIN_BG_SIZE )
  179.         label->hide();
  180.     else
  181.         label->show();
  182. }
  183. void BackgroundWidget::updateArt( const QString& url )
  184. {
  185.     if( url.isEmpty() )
  186.     {
  187.         if( QDate::currentDate().dayOfYear() >= 354 )
  188.             label->setPixmap( QPixmap( ":/vlc128-christmas.png" ) );
  189.         else
  190.             label->setPixmap( QPixmap( ":/vlc128.png" ) );
  191.     }
  192.     else
  193.     {
  194.         label->setPixmap( QPixmap( url ) );
  195.     }
  196. }
  197. void BackgroundWidget::contextMenuEvent( QContextMenuEvent *event )
  198. {
  199.     QVLCMenu::PopupMenu( p_intf, true );
  200.     event->accept();
  201. }
  202. #if 0
  203. #include <QPushButton>
  204. #include <QHBoxLayout>
  205. /**********************************************************************
  206.  * Visualization selector panel
  207.  **********************************************************************/
  208. VisualSelector::VisualSelector( intf_thread_t *_p_i ) :
  209.                                 QFrame( NULL ), p_intf( _p_i )
  210. {
  211.     QHBoxLayout *layout = new QHBoxLayout( this );
  212.     layout->setMargin( 0 );
  213.     QPushButton *prevButton = new QPushButton( "Prev" );
  214.     QPushButton *nextButton = new QPushButton( "Next" );
  215.     layout->addWidget( prevButton );
  216.     layout->addWidget( nextButton );
  217.     layout->addStretch( 10 );
  218.     layout->addWidget( new QLabel( qtr( "Current visualization" ) ) );
  219.     current = new QLabel( qtr( "None" ) );
  220.     layout->addWidget( current );
  221.     BUTTONACT( prevButton, prev() );
  222.     BUTTONACT( nextButton, next() );
  223.     setLayout( layout );
  224.     setMaximumHeight( 35 );
  225. }
  226. VisualSelector::~VisualSelector()
  227. {}
  228. void VisualSelector::prev()
  229. {
  230.     char *psz_new = aout_VisualPrev( p_intf );
  231.     if( psz_new )
  232.     {
  233.         current->setText( qfu( psz_new ) );
  234.         free( psz_new );
  235.     }
  236. }
  237. void VisualSelector::next()
  238. {
  239.     char *psz_new = aout_VisualNext( p_intf );
  240.     if( psz_new )
  241.     {
  242.         current->setText( qfu( psz_new ) );
  243.         free( psz_new );
  244.     }
  245. }
  246. #endif
  247. SpeedLabel::SpeedLabel( intf_thread_t *_p_intf, const QString& text,
  248.                         QWidget *parent )
  249.            : QLabel( text, parent ), p_intf( _p_intf )
  250. {
  251.     setToolTip( qtr( "Current playback speed.nClick to adjust" ) );
  252.     /* Create the Speed Control Widget */
  253.     speedControl = new SpeedControlWidget( p_intf, this );
  254.     speedControlMenu = new QMenu( this );
  255.     QWidgetAction *widgetAction = new QWidgetAction( speedControl );
  256.     widgetAction->setDefaultWidget( speedControl );
  257.     speedControlMenu->addAction( widgetAction );
  258.     /* Change the SpeedRate in the Status Bar */
  259.     CONNECT( THEMIM->getIM(), rateChanged( int ), this, setRate( int ) );
  260.     CONNECT( THEMIM, inputChanged( input_thread_t * ),
  261.              speedControl, activateOnState() );
  262. }
  263. SpeedLabel::~SpeedLabel()
  264. {
  265.         delete speedControl;
  266.         delete speedControlMenu;
  267. }
  268. /****************************************************************************
  269.  * Small right-click menu for rate control
  270.  ****************************************************************************/
  271. void SpeedLabel::showSpeedMenu( QPoint pos )
  272. {
  273.     speedControlMenu->exec( QCursor::pos() - pos
  274.                           + QPoint( 0, height() ) );
  275. }
  276. void SpeedLabel::setRate( int rate )
  277. {
  278.     QString str;
  279.     str.setNum( ( 1000 / (double)rate ), 'f', 2 );
  280.     str.append( "x" );
  281.     setText( str );
  282.     setToolTip( str );
  283.     speedControl->updateControls( rate );
  284. }
  285. /**********************************************************************
  286.  * Speed control widget
  287.  **********************************************************************/
  288. SpeedControlWidget::SpeedControlWidget( intf_thread_t *_p_i, QWidget *_parent )
  289.                     : QFrame( _parent ), p_intf( _p_i )
  290. {
  291.     QSizePolicy sizePolicy( QSizePolicy::Maximum, QSizePolicy::Fixed );
  292.     sizePolicy.setHorizontalStretch( 0 );
  293.     sizePolicy.setVerticalStretch( 0 );
  294.     speedSlider = new QSlider( this );
  295.     speedSlider->setSizePolicy( sizePolicy );
  296.     speedSlider->setMaximumSize( QSize( 80, 200 ) );
  297.     speedSlider->setOrientation( Qt::Vertical );
  298.     speedSlider->setTickPosition( QSlider::TicksRight );
  299.     speedSlider->setRange( -34, 34 );
  300.     speedSlider->setSingleStep( 1 );
  301.     speedSlider->setPageStep( 1 );
  302.     speedSlider->setTickInterval( 17 );
  303.     CONNECT( speedSlider, valueChanged( int ), this, updateRate( int ) );
  304.     QToolButton *normalSpeedButton = new QToolButton( this );
  305.     normalSpeedButton->setMaximumSize( QSize( 26, 20 ) );
  306.     normalSpeedButton->setAutoRaise( true );
  307.     normalSpeedButton->setText( "1x" );
  308.     normalSpeedButton->setToolTip( qtr( "Revert to normal play speed" ) );
  309.     CONNECT( normalSpeedButton, clicked(), this, resetRate() );
  310.     QVBoxLayout *speedControlLayout = new QVBoxLayout( this );
  311.     speedControlLayout->setLayoutMargins( 4, 4, 4, 4, 4 );
  312.     speedControlLayout->setSpacing( 4 );
  313.     speedControlLayout->addWidget( speedSlider );
  314.     speedControlLayout->addWidget( normalSpeedButton );
  315.     activateOnState();
  316. }
  317. void SpeedControlWidget::activateOnState()
  318. {
  319.     speedSlider->setEnabled( THEMIM->getIM()->hasInput() );
  320. }
  321. void SpeedControlWidget::updateControls( int rate )
  322. {
  323.     if( speedSlider->isSliderDown() )
  324.     {
  325.         //We don't want to change anything if the user is using the slider
  326.         return;
  327.     }
  328.     double value = 17 * log( (double)INPUT_RATE_DEFAULT / rate ) / log( 2 );
  329.     int sliderValue = (int) ( ( value > 0 ) ? value + .5 : value - .5 );
  330.     if( sliderValue < speedSlider->minimum() )
  331.     {
  332.         sliderValue = speedSlider->minimum();
  333.     }
  334.     else if( sliderValue > speedSlider->maximum() )
  335.     {
  336.         sliderValue = speedSlider->maximum();
  337.     }
  338.     //Block signals to avoid feedback loop
  339.     speedSlider->blockSignals( true );
  340.     speedSlider->setValue( sliderValue );
  341.     speedSlider->blockSignals( false );
  342. }
  343. void SpeedControlWidget::updateRate( int sliderValue )
  344. {
  345.     double speed = pow( 2, (double)sliderValue / 17 );
  346.     int rate = INPUT_RATE_DEFAULT / speed;
  347.     THEMIM->getIM()->setRate(rate);
  348. }
  349. void SpeedControlWidget::resetRate()
  350. {
  351.     THEMIM->getIM()->setRate( INPUT_RATE_DEFAULT );
  352. }
  353. CoverArtLabel::CoverArtLabel( QWidget *parent, intf_thread_t *_p_i )
  354.         : QLabel( parent ), p_intf( _p_i )
  355. {
  356.     setContextMenuPolicy( Qt::ActionsContextMenu );
  357.     CONNECT( this, updateRequested(), this, doUpdate() );
  358.     CONNECT( THEMIM->getIM(), artChanged( QString ),
  359.              this, doUpdate( const QString& ) );
  360.     setMinimumHeight( 128 );
  361.     setMinimumWidth( 128 );
  362.     setMaximumHeight( 128 );
  363.     setMaximumWidth( 128 );
  364.     setScaledContents( true );
  365.     QList< QAction* > artActions = actions();
  366.     QAction *action = new QAction( qtr( "Download cover art" ), this );
  367.     addAction( action );
  368.     CONNECT( action, triggered(), this, doUpdate() );
  369.     doUpdate( "" );
  370. }
  371. CoverArtLabel::~CoverArtLabel()
  372. {
  373.     QList< QAction* > artActions = actions();
  374.     foreach( QAction *act, artActions )
  375.         removeAction( act );
  376. }
  377. void CoverArtLabel::doUpdate( const QString& url )
  378. {
  379.     QPixmap pix;
  380.     if( !url.isEmpty()  && pix.load( url ) )
  381.     {
  382.         setPixmap( pix );
  383.     }
  384.     else
  385.     {
  386.         setPixmap( QPixmap( ":/noart.png" ) );
  387.     }
  388. }
  389. void CoverArtLabel::doUpdate()
  390. {
  391.     THEMIM->getIM()->requestArtUpdate();
  392. }
  393. TimeLabel::TimeLabel( intf_thread_t *_p_intf  ) :QLabel(), p_intf( _p_intf )
  394. {
  395.    b_remainingTime = false;
  396.    setText( " --:--/--:-- " );
  397.    setAlignment( Qt::AlignRight | Qt::AlignVCenter );
  398.    setToolTip( qtr( "Toggle between elapsed and remaining time" ) );
  399.    CONNECT( THEMIM->getIM(), cachingChanged( float ),
  400.             this, setCaching( float ) );
  401.    CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
  402.              this, setDisplayPosition( float, int, int ) );
  403. }
  404. void TimeLabel::setDisplayPosition( float pos, int time, int length )
  405. {
  406.     if( pos == -1.f )
  407.     {
  408.         setText( " --:--/--:-- " );
  409.         return;
  410.     }
  411.     char psz_length[MSTRTIME_MAX_SIZE], psz_time[MSTRTIME_MAX_SIZE];
  412.     secstotimestr( psz_length, length );
  413.     secstotimestr( psz_time, ( b_remainingTime && length ) ? length - time
  414.                                                            : time );
  415.     QString timestr;
  416.     timestr.sprintf( "%s/%s", psz_time,
  417.                             ( !length && time ) ? "--:--" : psz_length );
  418.     /* Add a minus to remaining time*/
  419.     if( b_remainingTime && length ) setText( " -"+timestr+" " );
  420.     else setText( " "+timestr+" " );
  421. }
  422. void TimeLabel::toggleTimeDisplay()
  423. {
  424.     b_remainingTime = !b_remainingTime;
  425. }
  426. void TimeLabel::setCaching( float f_cache )
  427. {
  428.     QString amount;
  429.     amount.setNum( (int)(100 * f_cache) );
  430.     setText( "Buff: " + amount + "%" );
  431. }