]> git.sesse.net Git - vlc/blob - modules/gui/qt4/components/playlist/icon_view.cpp
6a01d6e9fe422d7b702f7ad9cfc030df4c995b73
[vlc] / modules / gui / qt4 / components / playlist / icon_view.cpp
1 /*****************************************************************************
2  * icon_view.cpp : Icon view for the Playlist
3  ****************************************************************************
4  * Copyright © 2010 the VideoLAN team
5  * $Id$
6  *
7  * Authors:         Jean-Baptiste Kempf <jb@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 Foundation, Inc.,
21  * 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23
24 #include "components/playlist/icon_view.hpp"
25 #include "components/playlist/playlist_model.hpp"
26 #include "components/playlist/sorting.h"
27 #include "input_manager.hpp"
28
29 #include <QApplication>
30 #include <QPainter>
31 #include <QRect>
32 #include <QStyleOptionViewItem>
33 #include <QFontMetrics>
34 #include <QPixmapCache>
35 #include <QDrag>
36 #include <QDragMoveEvent>
37
38 #include "assert.h"
39
40 #define ART_SIZE_W          110
41 #define ART_SIZE_H          80
42 #define ART_RADIUS          5
43 #define SPACER              5
44
45 QString AbstractPlViewItemDelegate::getMeta( const QModelIndex & index, int meta ) const
46 {
47     return index.model()->index( index.row(),
48                                   PLModel::columnFromMeta( meta ),
49                                   index.parent() )
50                                 .data().toString();
51 }
52
53 void AbstractPlViewItemDelegate::paintBackground(
54     QPainter *painter, const QStyleOptionViewItem & option, const QModelIndex & index ) const
55 {
56     /* FIXME: This does not indicate item selection in all QStyles, so for the time being we
57        have to draw it ourselves, to ensure visible effect of selection on all platforms */
58     /* QApplication::style()->drawPrimitive( QStyle::PE_PanelItemViewItem, &option,
59                                             painter ); */
60
61     painter->save();
62     QRect r = option.rect.adjusted( 0, 0, -1, -1 );
63     if( option.state & QStyle::State_Selected )
64     {
65         painter->setBrush( option.palette.color( QPalette::Highlight ) );
66         painter->setPen( option.palette.color( QPalette::Highlight ).darker( 150 ) );
67         painter->drawRect( r );
68     }
69     else if( index.data( PLModel::IsCurrentRole ).toBool() )
70     {
71         painter->setBrush( QBrush( Qt::lightGray ) );
72         painter->setPen( QColor( Qt::darkGray ) );
73         painter->drawRect( r );
74     }
75     if( option.state & QStyle::State_MouseOver )
76     {
77         painter->setOpacity( 0.5 );
78         painter->setPen( Qt::NoPen );
79         painter->setBrush( option.palette.color( QPalette::Highlight ).lighter( 150 ) );
80         painter->drawRect( option.rect );
81     }
82     painter->restore();
83 }
84
85 QPixmap AbstractPlViewItemDelegate::getArtPixmap( const QModelIndex & index, const QSize & size ) const
86 {
87     PLItem *item = static_cast<PLItem*>( index.internalPointer() );
88     assert( item );
89
90     QString artUrl = InputManager::decodeArtURL( item->inputItem() );
91
92     if( artUrl.isEmpty() )
93     {
94         for( int i = 0; i < item->childCount(); i++ )
95         {
96             artUrl = InputManager::decodeArtURL( item->child( i )->inputItem() );
97             if( !artUrl.isEmpty() )
98                 break;
99         }
100     }
101
102     QPixmap artPix;
103
104     QString key = artUrl + QString("%1%2").arg(size.width()).arg(size.height());
105
106     if( !QPixmapCache::find( key, artPix ))
107     {
108         if( artUrl.isEmpty() || !artPix.load( artUrl ) )
109         {
110             key = QString("noart%1%2").arg(size.width()).arg(size.height());
111             if( !QPixmapCache::find( key, artPix ) )
112             {
113                 artPix = QPixmap( ":/noart" ).scaled( size,
114                                                       Qt::KeepAspectRatio,
115                                                       Qt::SmoothTransformation );
116                 QPixmapCache::insert( key, artPix );
117             }
118         }
119         else
120         {
121             artPix = artPix.scaled( size, Qt::KeepAspectRatio, Qt::SmoothTransformation );
122             QPixmapCache::insert( key, artPix );
123         }
124     }
125
126     return artPix;
127 }
128
129 void PlIconViewItemDelegate::paint( QPainter * painter, const QStyleOptionViewItem & option, const QModelIndex & index ) const
130 {
131     QString title = getMeta( index, COLUMN_TITLE );
132     QString artist = getMeta( index, COLUMN_ARTIST );
133
134     QPixmap artPix = getArtPixmap( index, QSize( ART_SIZE_W, ART_SIZE_H ) );
135
136     paintBackground( painter, option, index );
137
138     painter->save();
139
140     QRect artRect( option.rect.x() + 5 + ( ART_SIZE_W - artPix.width() ) / 2,
141                    option.rect.y() + 5 + ( ART_SIZE_H - artPix.height() ) / 2,
142                    artPix.width(), artPix.height() );
143
144     // Draw the drop shadow
145     painter->save();
146     painter->setOpacity( 0.7 );
147     painter->setBrush( QBrush( Qt::darkGray ) );
148     painter->setPen( Qt::NoPen );
149     painter->drawRoundedRect( artRect.adjusted( 0, 0, 2, 2 ), ART_RADIUS, ART_RADIUS );
150     painter->restore();
151
152     // Draw the art pixmap
153     QPainterPath artRectPath;
154     artRectPath.addRoundedRect( artRect, ART_RADIUS, ART_RADIUS );
155     painter->setClipPath( artRectPath );
156     painter->drawPixmap( artRect, artPix );
157     painter->setClipping( false );
158
159     if( option.state & QStyle::State_Selected )
160         painter->setPen( option.palette.color( QPalette::HighlightedText ) );
161
162     QFont font( index.data( Qt::FontRole ).value<QFont>() );
163
164     //Draw children indicator
165     if( !index.data( PLModel::IsLeafNodeRole ).toBool() )
166     {
167         painter->setOpacity( 0.75 );
168         QRect r( option.rect );
169         r.setSize( QSize( 25, 25 ) );
170         r.translate( 5, 5 );
171         if( index.data( PLModel::IsCurrentsParentNodeRole ).toBool() )
172             painter->fillRect( r, option.palette.color( QPalette::Highlight ) );
173         else
174             painter->fillRect( r, option.palette.color( QPalette::Mid ) );
175         painter->setOpacity( 1.0 );
176         QPixmap dirPix( ":/type/node" );
177         QRect r2( dirPix.rect() );
178         r2.moveCenter( r.center() );
179         painter->drawPixmap( r2, dirPix );
180     }
181
182     // Draw title
183     font.setItalic( true );
184     painter->setFont( font );
185
186     QFontMetrics fm = painter->fontMetrics();
187     QRect textRect = option.rect.adjusted( 1, ART_SIZE_H + 10, 0, -1 );
188     textRect.setHeight( fm.height() );
189
190     painter->drawText( textRect,
191                       fm.elidedText( title, Qt::ElideRight, textRect.width() ),
192                       QTextOption( Qt::AlignCenter ) );
193
194     // Draw artist
195     painter->setPen( painter->pen().color().lighter( 150 ) );
196     font.setItalic( false );
197     painter->setFont( font );
198     fm = painter->fontMetrics();
199
200     textRect.moveTop( textRect.bottom() + 1 );
201
202     painter->drawText(  textRect,
203                         fm.elidedText( artist, Qt::ElideRight, textRect.width() ),
204                         QTextOption( Qt::AlignCenter ) );
205
206     painter->restore();
207 }
208
209 QSize PlIconViewItemDelegate::sizeHint ( const QStyleOptionViewItem & option, const QModelIndex & index ) const
210 {
211     QFont f( index.data( Qt::FontRole ).value<QFont>() );
212     f.setBold( true );
213     QFontMetrics fm( f );
214     int textHeight = fm.height();
215     QSize sz ( ART_SIZE_W + 2 * SPACER,
216                ART_SIZE_H + 3 * SPACER + 2 * textHeight + 1 );
217     return sz;
218 }
219
220
221 #define LISTVIEW_ART_SIZE 45
222
223 void PlListViewItemDelegate::paint( QPainter * painter, const QStyleOptionViewItem & option, const QModelIndex & index ) const
224 {
225     QModelIndex parent = index.parent();
226     QModelIndex i;
227
228     QString title = getMeta( index, COLUMN_TITLE );
229     QString duration = getMeta( index, COLUMN_DURATION );
230     if( !duration.isEmpty() ) title += QString(" [%1]").arg( duration );
231
232     QString artist = getMeta( index, COLUMN_ARTIST );
233     QString album = getMeta( index, COLUMN_ALBUM );
234     QString trackNum = getMeta( index, COLUMN_TRACK_NUMBER );
235     QString artistAlbum = artist;
236     if( !album.isEmpty() )
237     {
238         if( !artist.isEmpty() ) artistAlbum += ": ";
239         artistAlbum += album;
240         if( !trackNum.isEmpty() ) artistAlbum += QString( " [#%1]" ).arg( trackNum );
241     }
242
243     QPixmap artPix = getArtPixmap( index, QSize( LISTVIEW_ART_SIZE, LISTVIEW_ART_SIZE ) );
244
245     //Draw selection rectangle and current playing item indication
246     paintBackground( painter, option, index );
247
248     QRect artRect( artPix.rect() );
249     artRect.moveCenter( QPoint( artRect.center().x() + 3,
250                                 option.rect.center().y() ) );
251     //Draw album art
252     painter->drawPixmap( artRect, artPix );
253
254     //Start drawing text
255     painter->save();
256
257     if( option.state & QStyle::State_Selected )
258         painter->setPen( option.palette.color( QPalette::HighlightedText ) );
259
260     QTextOption textOpt( Qt::AlignVCenter | Qt::AlignLeft );
261     textOpt.setWrapMode( QTextOption::NoWrap );
262
263     QFont f( index.data( Qt::FontRole ).value<QFont>() );
264
265     //Draw title info
266     f.setItalic( true );
267     painter->setFont( f );
268     QFontMetrics fm( painter->fontMetrics() );
269
270     QRect textRect = option.rect.adjusted( LISTVIEW_ART_SIZE + 10, 0, -10, 0 );
271     if( !artistAlbum.isEmpty() )
272     {
273         textRect.setHeight( fm.height() );
274         textRect.moveBottom( option.rect.center().y() - 2 );
275     }
276
277     //Draw children indicator
278     if( !index.data( PLModel::IsLeafNodeRole ).toBool() )
279     {
280         QPixmap dirPix = QPixmap( ":/type/node" );
281         painter->drawPixmap( QPoint( textRect.x(), textRect.center().y() - dirPix.height() / 2 ),
282                              dirPix );
283         textRect.setLeft( textRect.x() + dirPix.width() + 5 );
284     }
285
286     painter->drawText( textRect,
287                        fm.elidedText( title, Qt::ElideRight, textRect.width() ),
288                        textOpt );
289
290     // Draw artist and album info
291     if( !artistAlbum.isEmpty() )
292     {
293         f.setItalic( false );
294         painter->setFont( f );
295         fm = painter->fontMetrics();
296
297         textRect.moveTop( textRect.bottom() + 4 );
298         textRect.setLeft( textRect.x() + 20 );
299
300         painter->drawText( textRect,
301                            fm.elidedText( artistAlbum, Qt::ElideRight, textRect.width() ),
302                            textOpt );
303     }
304
305     painter->restore();
306 }
307
308 QSize PlListViewItemDelegate::sizeHint ( const QStyleOptionViewItem & option, const QModelIndex & index ) const
309 {
310   QFont f;
311   f.setBold( true );
312   QFontMetrics fm( f );
313   int height = qMax( LISTVIEW_ART_SIZE, 2 * fm.height() + 4 ) + 6;
314   return QSize( 0, height );
315 }
316
317 static void plViewStartDrag( QAbstractItemView *view, const Qt::DropActions & supportedActions )
318 {
319     QDrag *drag = new QDrag( view );
320     drag->setPixmap( QPixmap( ":/noart64" ) );
321     drag->setMimeData( view->model()->mimeData(
322         view->selectionModel()->selectedIndexes() ) );
323     drag->exec( supportedActions );
324 }
325
326 static void plViewDragMoveEvent( QAbstractItemView *view, QDragMoveEvent * event )
327 {
328     if( event->keyboardModifiers() & Qt::ControlModifier &&
329         event->possibleActions() & Qt::CopyAction )
330         event->setDropAction( Qt::CopyAction );
331     else event->acceptProposedAction();
332 }
333
334 PlIconView::PlIconView( PLModel *model, QWidget *parent ) : QListView( parent )
335 {
336     PlIconViewItemDelegate *delegate = new PlIconViewItemDelegate( this );
337
338     setModel( model );
339     setViewMode( QListView::IconMode );
340     setMovement( QListView::Static );
341     setResizeMode( QListView::Adjust );
342     setGridSize( delegate->sizeHint() );
343     setWrapping( true );
344     setUniformItemSizes( true );
345     setSelectionMode( QAbstractItemView::ExtendedSelection );
346     setDragEnabled(true);
347     /* dropping in QListView::IconMode does not seem to work */
348     //setAcceptDrops( true );
349     //setDropIndicatorShown(true);
350
351     setItemDelegate( delegate );
352 }
353
354 void PlIconView::startDrag ( Qt::DropActions supportedActions )
355 {
356     plViewStartDrag( this, supportedActions );
357 }
358
359 void PlIconView::dragMoveEvent ( QDragMoveEvent * event )
360 {
361     plViewDragMoveEvent( this, event );
362     QAbstractItemView::dragMoveEvent( event );
363 }
364
365 PlListView::PlListView( PLModel *model, QWidget *parent ) : QListView( parent )
366 {
367     setModel( model );
368     setViewMode( QListView::ListMode );
369     setUniformItemSizes( true );
370     setSelectionMode( QAbstractItemView::ExtendedSelection );
371     setAlternatingRowColors( true );
372     setDragEnabled(true);
373     setAcceptDrops( true );
374     setDropIndicatorShown(true);
375
376     PlListViewItemDelegate *delegate = new PlListViewItemDelegate( this );
377     setItemDelegate( delegate );
378 }
379
380 void PlListView::startDrag ( Qt::DropActions supportedActions )
381 {
382     plViewStartDrag( this, supportedActions );
383 }
384
385 void PlListView::dragMoveEvent ( QDragMoveEvent * event )
386 {
387     plViewDragMoveEvent( this, event );
388     QAbstractItemView::dragMoveEvent( event );
389 }
390
391 void PlListView::keyPressEvent( QKeyEvent *event )
392 {
393     //If the space key is pressed, override the standard list behaviour to allow pausing
394     //to proceed.
395     if ( event->modifiers() == Qt::NoModifier && event->key() == Qt::Key_Space )
396         QWidget::keyPressEvent( event );
397     //Otherwise, just do as usual.
398     else
399         QListView::keyPressEvent( event );
400 }
401
402 void PlTreeView::startDrag ( Qt::DropActions supportedActions )
403 {
404     plViewStartDrag( this, supportedActions );
405 }
406
407 void PlTreeView::dragMoveEvent ( QDragMoveEvent * event )
408 {
409     plViewDragMoveEvent( this, event );
410     QAbstractItemView::dragMoveEvent( event );
411 }
412
413 void PlTreeView::keyPressEvent( QKeyEvent *event )
414 {
415     //If the space key is pressed, override the standard list behaviour to allow pausing
416     //to proceed.
417     if ( event->modifiers() == Qt::NoModifier && event->key() == Qt::Key_Space )
418         QWidget::keyPressEvent( event );
419     //Otherwise, just do as usual.
420     else
421         QTreeView::keyPressEvent( event );
422 }