1
0
mirror of https://github.com/tomahawk-player/tomahawk.git synced 2025-03-19 15:29:42 +01:00

* Added ColumnView.

This commit is contained in:
Christian Muehlhaeuser 2013-06-12 06:15:48 +02:00
parent d5eec1ae8f
commit 7c1ac3f75d
2 changed files with 603 additions and 0 deletions

View File

@ -0,0 +1,484 @@
/* === This file is part of Tomahawk Player - <http://tomahawk-player.org> ===
*
* Copyright 2010-2013, Christian Muehlhaeuser <muesli@tomahawk-player.org>
*
* Tomahawk 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 3 of the License, or
* (at your option) any later version.
*
* Tomahawk 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 Tomahawk. If not, see <http://www.gnu.org/licenses/>.
*/
#include "ColumnView.h"
#include "audio/AudioEngine.h"
#include "context/ContextWidget.h"
#include "utils/AnimatedSpinner.h"
#include "widgets/OverlayWidget.h"
#include "ContextMenu.h"
#include "TomahawkSettings.h"
#include "ViewHeader.h"
#include "ColumnItemDelegate.h"
#include "ColumnViewPreviewWidget.h"
#include "TreeModel.h"
#include "PlayableItem.h"
#include "Source.h"
#include "ViewManager.h"
#include "utils/TomahawkUtilsGui.h"
#include "utils/Logger.h"
#include <QHeaderView>
#include <QKeyEvent>
#include <QPainter>
#include <QScrollBar>
#include <QDrag>
#include <QMimeData>
#include <boost/concept_check.hpp>
#define SCROLL_TIMEOUT 280
using namespace Tomahawk;
ColumnView::ColumnView( QWidget* parent )
: QColumnView( parent )
, m_header( new ViewHeader( this ) )
, m_overlay( new OverlayWidget( this ) )
, m_model( 0 )
, m_proxyModel( 0 )
, m_delegate( 0 )
, m_loadingSpinner( new LoadingSpinner( this ) )
, m_previewWidget( new ColumnViewPreviewWidget( this ) )
, m_updateContextView( true )
, m_contextMenu( new ContextMenu( this ) )
{
setFrameShape( QFrame::NoFrame );
setAttribute( Qt::WA_MacShowFocusRect, 0 );
setContentsMargins( 0, 0, 0, 0 );
setMouseTracking( true );
setAlternatingRowColors( true );
setDragEnabled( true );
setDropIndicatorShown( false );
setDragDropOverwriteMode( false );
setVerticalScrollMode( QAbstractItemView::ScrollPerPixel );
setSelectionMode( QAbstractItemView::SingleSelection );
setSelectionBehavior( QAbstractItemView::SelectRows );
setContextMenuPolicy( Qt::CustomContextMenu );
setProxyModel( new TreeProxyModel( this ) );
setPreviewWidget( m_previewWidget );
m_timer.setInterval( SCROLL_TIMEOUT );
connect( verticalScrollBar(), SIGNAL( rangeChanged( int, int ) ), SLOT( onViewChanged() ) );
connect( verticalScrollBar(), SIGNAL( valueChanged( int ) ), SLOT( onViewChanged() ) );
connect( &m_timer, SIGNAL( timeout() ), SLOT( onScrollTimeout() ) );
connect( this, SIGNAL( updatePreviewWidget( QModelIndex ) ), SLOT( onUpdatePreviewWidget( QModelIndex ) ) );
connect( this, SIGNAL( doubleClicked( QModelIndex ) ), SLOT( onItemActivated( QModelIndex ) ) );
connect( this, SIGNAL( customContextMenuRequested( QPoint ) ), SLOT( onCustomContextMenu( QPoint ) ) );
connect( m_contextMenu, SIGNAL( triggered( int ) ), SLOT( onMenuTriggered( int ) ) );
}
ColumnView::~ColumnView()
{
tDebug() << Q_FUNC_INFO;
}
void
ColumnView::setProxyModel( TreeProxyModel* model )
{
m_proxyModel = model;
m_delegate = new ColumnItemDelegate( this, m_proxyModel );
setItemDelegate( m_delegate );
QColumnView::setModel( m_proxyModel );
}
void
ColumnView::setModel( QAbstractItemModel* model )
{
Q_UNUSED( model );
tDebug() << "Explicitly use setPlaylistModel instead";
Q_ASSERT( false );
}
void
ColumnView::setTreeModel( TreeModel* model )
{
m_model = model;
if ( m_proxyModel )
{
m_proxyModel->setSourcePlayableModel( model );
m_proxyModel->sort( 0 );
}
connect( m_proxyModel, SIGNAL( filteringStarted() ), SLOT( onFilteringStarted() ) );
connect( m_proxyModel, SIGNAL( filteringFinished() ), m_loadingSpinner, SLOT( fadeOut() ) );
connect( m_proxyModel, SIGNAL( filteringFinished() ), SLOT( onFilterChangeFinished() ) );
connect( m_proxyModel, SIGNAL( rowsInserted( QModelIndex, int, int ) ), SLOT( onViewChanged() ) );
guid(); // this will set the guid on the header
m_header->setDefaultColumnWeights( m_proxyModel->columnWeights() );
if ( m_proxyModel->style() == PlayableProxyModel::Large )
{
setHorizontalScrollBarPolicy( Qt::ScrollBarAlwaysOff );
}
else
{
setHorizontalScrollBarPolicy( Qt::ScrollBarAsNeeded );
}
connect( model, SIGNAL( changed() ), this, SIGNAL( modelChanged() ) );
emit modelChanged();
/* setColumnHidden( PlayableModel::Score, true ); // Hide score column per default
setColumnHidden( PlayableModel::Origin, true ); // Hide origin column per default
setColumnHidden( PlayableModel::Composer, true ); //Hide composer column per default
setGuid( QString( "columnview/%1" ).arg( model->columnCount() ) );
sortByColumn( PlayableModel::Artist, Qt::AscendingOrder );*/
QList< int > widths;
widths << 320;
setColumnWidths( widths );
}
void
ColumnView::setEmptyTip( const QString& tip )
{
m_emptyTip = tip;
m_overlay->setText( tip );
}
bool
ColumnView::setFilter( const QString& filter )
{
proxyModel()->setFilter( filter );
return true;
}
void
ColumnView::onViewChanged()
{
if ( m_timer.isActive() )
m_timer.stop();
m_timer.start();
}
void
ColumnView::onScrollTimeout()
{
if ( m_timer.isActive() )
m_timer.stop();
QModelIndex left = indexAt( viewport()->rect().topLeft() );
while ( left.isValid() && left.parent().isValid() )
left = left.parent();
QModelIndex right = indexAt( viewport()->rect().bottomLeft() );
while ( right.isValid() && right.parent().isValid() )
right = right.parent();
int max = m_proxyModel->playlistInterface()->trackCount();
if ( right.isValid() )
max = right.row() + 1;
if ( !max )
return;
for ( int i = left.row(); i < max; i++ )
{
m_model->getCover( m_proxyModel->mapToSource( m_proxyModel->index( i, 0 ) ) );
}
}
void
ColumnView::currentChanged( const QModelIndex& current, const QModelIndex& previous )
{
QColumnView::currentChanged( current, previous );
if ( !m_updateContextView )
return;
PlayableItem* item = m_model->itemFromIndex( m_proxyModel->mapToSource( current ) );
if ( item )
{
if ( !item->result().isNull() )
ViewManager::instance()->context()->setQuery( item->result()->toQuery() );
else if ( !item->artist().isNull() )
ViewManager::instance()->context()->setArtist( item->artist() );
else if ( !item->album().isNull() )
ViewManager::instance()->context()->setAlbum( item->album() );
else if ( !item->query().isNull() )
ViewManager::instance()->context()->setQuery( item->query() );
}
}
void
ColumnView::onItemActivated( const QModelIndex& index )
{
PlayableItem* item = m_model->itemFromIndex( m_proxyModel->mapToSource( index ) );
if ( item )
{
if ( !item->result().isNull() && item->result()->isOnline() )
{
AudioEngine::instance()->playItem( m_proxyModel->playlistInterface(), item->result() );
}
else if ( !item->query().isNull() )
{
AudioEngine::instance()->playItem( m_proxyModel->playlistInterface(), item->query() );
}
}
}
void
ColumnView::keyPressEvent( QKeyEvent* event )
{
QColumnView::keyPressEvent( event );
if ( !model() )
return;
if ( event->key() == Qt::Key_Enter || event->key() == Qt::Key_Return )
{
onItemActivated( currentIndex() );
}
}
void
ColumnView::resizeEvent( QResizeEvent* event )
{
QColumnView::resizeEvent( event );
m_header->checkState();
if ( !model() )
return;
if ( model()->columnCount( QModelIndex() ) == 1 )
{
m_header->resizeSection( 0, event->size().width() );
}
}
void
ColumnView::wheelEvent( QWheelEvent* event )
{
QColumnView::wheelEvent( event );
m_delegate->resetHoverIndex();
repaint();
}
void
ColumnView::onFilterChangeFinished()
{
if ( selectedIndexes().count() )
scrollTo( selectedIndexes().at( 0 ), QAbstractItemView::PositionAtCenter );
if ( !proxyModel()->filter().isEmpty() && !proxyModel()->playlistInterface()->trackCount() && model()->trackCount() )
{
m_overlay->setText( tr( "Sorry, your filter '%1' did not match any results." ).arg( proxyModel()->filter() ) );
m_overlay->show();
}
else
{
if ( model()->trackCount() )
{
m_overlay->hide();
}
else
{
m_overlay->setText( m_emptyTip );
m_overlay->show();
}
}
}
void
ColumnView::onFilteringStarted()
{
m_overlay->hide();
m_loadingSpinner->fadeIn();
}
void
ColumnView::startDrag( Qt::DropActions supportedActions )
{
QList<QPersistentModelIndex> pindexes;
QModelIndexList indexes;
foreach( const QModelIndex& idx, selectedIndexes() )
{
if ( ( m_proxyModel->flags( idx ) & Qt::ItemIsDragEnabled ) )
{
indexes << idx;
pindexes << idx;
}
}
if ( indexes.count() == 0 )
return;
tDebug( LOGVERBOSE ) << "Dragging" << indexes.count() << "indexes";
QMimeData* data = m_proxyModel->mimeData( indexes );
if ( !data )
return;
QDrag* drag = new QDrag( this );
drag->setMimeData( data );
QPixmap p;
if ( data->hasFormat( "application/tomahawk.metadata.artist" ) )
p = TomahawkUtils::createDragPixmap( TomahawkUtils::MediaTypeArtist, indexes.count() );
else if ( data->hasFormat( "application/tomahawk.metadata.album" ) )
p = TomahawkUtils::createDragPixmap( TomahawkUtils::MediaTypeAlbum, indexes.count() );
else
p = TomahawkUtils::createDragPixmap( TomahawkUtils::MediaTypeTrack, indexes.count() );
drag->setPixmap( p );
drag->setHotSpot( QPoint( -20, -20 ) );
drag->exec( supportedActions, Qt::CopyAction );
}
void
ColumnView::onCustomContextMenu( const QPoint& pos )
{
m_contextMenu->clear();
QModelIndex idx = indexAt( pos );
idx = idx.sibling( idx.row(), 0 );
m_contextMenuIndex = idx;
if ( !idx.isValid() )
return;
QList<query_ptr> queries;
QList<artist_ptr> artists;
QList<album_ptr> albums;
QModelIndexList indexes = selectedIndexes();
if ( !indexes.contains( idx ) )
{
indexes.clear();
indexes << idx;
}
foreach ( const QModelIndex& index, indexes )
{
if ( index.column() || indexes.contains( index.parent() ) )
continue;
PlayableItem* item = m_proxyModel->itemFromIndex( m_proxyModel->mapToSource( index ) );
if ( item && !item->result().isNull() )
queries << item->result()->toQuery();
else if ( item && !item->query().isNull() )
queries << item->query();
if ( item && !item->artist().isNull() )
artists << item->artist();
if ( item && !item->album().isNull() )
albums << item->album();
}
m_contextMenu->setQueries( queries );
m_contextMenu->setArtists( artists );
m_contextMenu->setAlbums( albums );
m_contextMenu->setPlaylistInterface( proxyModel()->playlistInterface() );
m_contextMenu->exec( viewport()->mapToGlobal( pos ) );
}
void
ColumnView::onMenuTriggered( int action )
{
switch ( action )
{
case ContextMenu::ActionPlay:
onItemActivated( m_contextMenuIndex );
break;
default:
break;
}
}
bool
ColumnView::jumpToCurrentTrack()
{
if ( !m_proxyModel || !m_proxyModel->sourceModel() )
return false;
scrollTo( m_proxyModel->currentIndex(), QAbstractItemView::PositionAtCenter );
return true;
}
QString
ColumnView::guid() const
{
if ( m_guid.isEmpty() )
{
m_guid = QString( "columnview/%1" ).arg( m_model->columnCount( QModelIndex() ) );
m_header->setGuid( m_guid );
}
return m_guid;
}
void
ColumnView::onUpdatePreviewWidget( const QModelIndex& index )
{
PlayableItem* item = m_proxyModel->itemFromIndex( m_proxyModel->mapToSource( index ) );
if ( !item || !item->result() )
{
QList< int > widths = columnWidths();
QList< int > finalWidths;
foreach ( int w, widths )
{
finalWidths << qMax( 320, w );
}
setColumnWidths( finalWidths );
return;
}
m_previewWidget->setQuery( item->result()->toQuery() );
QList< int > widths = columnWidths();
const int previewWidth = viewport()->width() - widths.at( 0 ) - widths.at( 1 ) - widths.at( 2 );
widths.removeLast();
widths << qMax( previewWidth, m_previewWidget->minimumSize().width() + 32 );
setColumnWidths( widths );
}

View File

@ -0,0 +1,119 @@
/* === This file is part of Tomahawk Player - <http://tomahawk-player.org> ===
*
* Copyright 2010-2011, Christian Muehlhaeuser <muesli@tomahawk-player.org>
* Copyright 2010-2012, Jeff Mitchell <jeff@tomahawk-player.org>
*
* Tomahawk 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 3 of the License, or
* (at your option) any later version.
*
* Tomahawk 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 Tomahawk. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef COLUMNVIEW_H
#define COLUMNVIEW_H
#include <QSortFilterProxyModel>
#include <QColumnView>
#include <QTimer>
#include "TreeProxyModel.h"
#include "ViewPage.h"
#include "PlaylistInterface.h"
#include "DllMacro.h"
namespace Tomahawk
{
class ContextMenu;
};
class ViewHeader;
class AnimatedSpinner;
class OverlayWidget;
class TreeModel;
class ColumnItemDelegate;
class ColumnViewPreviewWidget;
class DLLEXPORT ColumnView : public QColumnView
{
Q_OBJECT
public:
explicit ColumnView( QWidget* parent = 0 );
~ColumnView();
virtual QString guid() const;
virtual void setGuid( const QString& guid ) { m_guid = guid; }
void setProxyModel( TreeProxyModel* model );
TreeModel* model() const { return m_model; }
TreeProxyModel* proxyModel() const { return m_proxyModel; }
OverlayWidget* overlay() const { return m_overlay; }
void setModel( QAbstractItemModel* model );
void setTreeModel( TreeModel* model );
virtual bool setFilter( const QString& filter );
void setEmptyTip( const QString& tip );
virtual bool jumpToCurrentTrack();
bool updatesContextView() const { return m_updateContextView; }
void setUpdatesContextView( bool b ) { m_updateContextView = b; }
public slots:
void onItemActivated( const QModelIndex& index );
signals:
void modelChanged();
protected:
virtual void startDrag( Qt::DropActions supportedActions );
virtual void resizeEvent( QResizeEvent* event );
virtual void keyPressEvent( QKeyEvent* event );
virtual void wheelEvent( QWheelEvent* event );
protected slots:
virtual void currentChanged( const QModelIndex& current, const QModelIndex& previous );
virtual void onUpdatePreviewWidget( const QModelIndex& index );
private slots:
void onFilterChangeFinished();
void onFilteringStarted();
void onViewChanged();
void onScrollTimeout();
void onCustomContextMenu( const QPoint& pos );
void onMenuTriggered( int action );
private:
ViewHeader* m_header;
OverlayWidget* m_overlay;
TreeModel* m_model;
TreeProxyModel* m_proxyModel;
ColumnItemDelegate* m_delegate;
AnimatedSpinner* m_loadingSpinner;
ColumnViewPreviewWidget* m_previewWidget;
bool m_updateContextView;
QModelIndex m_contextMenuIndex;
Tomahawk::ContextMenu* m_contextMenu;
QString m_emptyTip;
QTimer m_timer;
mutable QString m_guid;
};
#endif // COLUMNVIEW_H