Commit bd396077 authored by Francois Cartegnie's avatar Francois Cartegnie

Qt: add addons management UI

parent aea9ebb9
...@@ -40,6 +40,7 @@ libqt4_plugin_la_SOURCES = \ ...@@ -40,6 +40,7 @@ libqt4_plugin_la_SOURCES = \
input_manager.cpp input_manager.hpp \ input_manager.cpp input_manager.hpp \
actions_manager.cpp actions_manager.hpp \ actions_manager.cpp actions_manager.hpp \
extensions_manager.cpp extensions_manager.hpp \ extensions_manager.cpp extensions_manager.hpp \
managers/addons_manager.cpp managers/addons_manager.hpp \
recents.cpp recents.hpp \ recents.cpp recents.hpp \
adapters/seekpoints.cpp adapters/seekpoints.hpp \ adapters/seekpoints.cpp adapters/seekpoints.hpp \
adapters/chromaprint.cpp adapters/chromaprint.hpp \ adapters/chromaprint.cpp adapters/chromaprint.hpp \
...@@ -141,6 +142,7 @@ nodist_libqt4_plugin_la_SOURCES = \ ...@@ -141,6 +142,7 @@ nodist_libqt4_plugin_la_SOURCES = \
input_manager.moc.cpp \ input_manager.moc.cpp \
actions_manager.moc.cpp \ actions_manager.moc.cpp \
extensions_manager.moc.cpp \ extensions_manager.moc.cpp \
managers/addons_manager.moc.cpp \
recents.moc.cpp \ recents.moc.cpp \
adapters/seekpoints.moc.cpp \ adapters/seekpoints.moc.cpp \
adapters/chromaprint.moc.cpp \ adapters/chromaprint.moc.cpp \
......
...@@ -30,6 +30,8 @@ ...@@ -30,6 +30,8 @@
#include "util/searchlineedit.hpp" #include "util/searchlineedit.hpp"
#include "extensions_manager.hpp" #include "extensions_manager.hpp"
#include "managers/addons_manager.hpp"
#include "util/animators.hpp"
#include <assert.h> #include <assert.h>
...@@ -52,7 +54,12 @@ ...@@ -52,7 +54,12 @@
#include <QStyleOptionViewItem> #include <QStyleOptionViewItem>
#include <QKeyEvent> #include <QKeyEvent>
#include <QPushButton> #include <QPushButton>
#include <QCheckBox>
#include <QPixmap> #include <QPixmap>
#include <QStylePainter>
#include <QGraphicsColorizeEffect>
#include <QProgressBar>
#include <QTextEdit>
static QPixmap *loadPixmapFromData( char *, int size ); static QPixmap *loadPixmapFromData( char *, int size );
...@@ -65,9 +72,11 @@ PluginDialog::PluginDialog( intf_thread_t *_p_intf ) : QVLCFrame( _p_intf ) ...@@ -65,9 +72,11 @@ PluginDialog::PluginDialog( intf_thread_t *_p_intf ) : QVLCFrame( _p_intf )
QVBoxLayout *layout = new QVBoxLayout( this ); QVBoxLayout *layout = new QVBoxLayout( this );
tabs = new QTabWidget( this ); tabs = new QTabWidget( this );
tabs->addTab( extensionTab = new ExtensionTab( p_intf ), tabs->addTab( extensionTab = new ExtensionTab( p_intf ),
qtr( "Extensions" ) ); qtr( "Active Extensions" ) );
tabs->addTab( pluginTab = new PluginTab( p_intf ), tabs->addTab( pluginTab = new PluginTab( p_intf ),
qtr( "Plugins" ) ); qtr( "Plugins" ) );
tabs->addTab( addonsTab = new AddonsTab( p_intf ),
qtr( "Addons Manager" ) );
layout->addWidget( tabs ); layout->addWidget( tabs );
QDialogButtonBox *box = new QDialogButtonBox; QDialogButtonBox *box = new QDialogButtonBox;
...@@ -289,6 +298,182 @@ void ExtensionTab::moreInformation() ...@@ -289,6 +298,182 @@ void ExtensionTab::moreInformation()
dlg.exec(); dlg.exec();
} }
/* Add-ons tab */
AddonsTab::AddonsTab( intf_thread_t *p_intf_ ) : QVLCFrame( p_intf_ )
{
// Layout
QVBoxLayout *layout = new QVBoxLayout( this );
// Filters
QHBoxLayout *filtersLayout = new QHBoxLayout();
QLabel *addonsLabel = new QLabel( qtr("Addon type:") );
addonsLabel->setSizePolicy( QSizePolicy::Maximum, QSizePolicy::Preferred );
filtersLayout->addWidget( addonsLabel );
QComboBox *typeCombo = new QComboBox();
typeCombo->addItem( qtr("All"), -1 );
typeCombo->addItem( qtr("Skins"), ADDON_SKIN2 );
typeCombo->addItem( qtr("Playlist parsers"), ADDON_PLAYLIST_PARSER );
typeCombo->addItem( qtr("Service Discovery"), ADDON_SERVICE_DISCOVERY );
typeCombo->addItem( qtr("Extensions"), ADDON_EXTENSION );
CONNECT( typeCombo, currentIndexChanged(int), this, typeChanged( int ) );
filtersLayout->addWidget( typeCombo );
QCheckBox *installedOnlyBox = new QCheckBox( qtr("Show Installed Only") );
filtersLayout->addWidget( installedOnlyBox );
CONNECT( installedOnlyBox, stateChanged(int), this, installChecked(int) );
layout->addLayout( filtersLayout );
// Help Tab
helpLabel = new QLabel();
layout->addWidget( helpLabel );
// Main View
AddonsManager *AM = AddonsManager::getInstance( p_intf );
// ListView
addonsView = new QListView( this );
CONNECT( addonsView, activated( const QModelIndex& ), this, moreInformation() );
layout->addWidget( addonsView );
// List item delegate
AddonItemDelegate *addonsDelegate = new AddonItemDelegate( addonsView );
addonsView->setItemDelegate( addonsDelegate );
addonsDelegate->setAnimator( new DelegateAnimationHelper( addonsView ) );
CONNECT( addonsDelegate, showInfo(), this, moreInformation() );
// Extension list look & feeling
addonsView->setAlternatingRowColors( true );
addonsView->setSelectionMode( QAbstractItemView::SingleSelection );
// Model
AddonsListModel *model = new AddonsListModel( AM, addonsView );
addonsModel = new AddonsSortFilterProxyModel();
addonsModel->setDynamicSortFilter( true );
addonsModel->setSourceModel( model );
addonsModel->setFilterRole( Qt::DisplayRole );
addonsView->setModel( addonsModel );
CONNECT( addonsView->selectionModel(), currentChanged(QModelIndex,QModelIndex),
addonsView, edit(QModelIndex) );
CONNECT( AM, addonAdded( addon_entry_t * ),
model, addonAdded( addon_entry_t * ) );
CONNECT( AM, addonChanged( const addon_entry_t * ),
model, addonChanged( const addon_entry_t * ) );
QList<QString> frames;
frames << ":/util/wait1";
frames << ":/util/wait2";
frames << ":/util/wait3";
frames << ":/util/wait4";
spinnerAnimation = new PixmapAnimator( this, frames );
CONNECT( spinnerAnimation, pixmapReady( const QPixmap & ),
addonsView->viewport(), update() );
addonsView->viewport()->installEventFilter( this );
}
AddonsTab::~AddonsTab()
{
delete spinnerAnimation;
}
bool AddonsTab::eventFilter( QObject *obj, QEvent *event )
{
switch( event->type() )
{
case QEvent::Paint:
if ( spinnerAnimation->state() == PixmapAnimator::Running )
{
QWidget *viewport = qobject_cast<QWidget *>( obj );
QStylePainter painter( viewport );
QPixmap *spinner = spinnerAnimation->getPixmap();
QPoint point = viewport->geometry().center();
point -= QPoint( spinner->size().width() / 2, spinner->size().height() / 2 );
painter.drawPixmap( point, *spinner );
QString text = qtr("Retrieving addons...");
QSize textsize = fontMetrics().size( 0, text );
point = viewport->geometry().center();
point -= QPoint( textsize.width() / 2, -spinner->size().height() );
painter.drawText( point, text );
}
else if ( addonsModel->rowCount() == 0 )
{
QWidget *viewport = qobject_cast<QWidget *>( obj );
QStylePainter painter( viewport );
QString text = qtr("No addons found");
QSize size = fontMetrics().size( 0, text );
QPoint point = viewport->geometry().center();
point -= QPoint( size.width() / 2, size.height() / 2 );
painter.drawText( point, text );
}
break;
case QEvent::Show:
if ( addonsView->model()->rowCount() < 1 )
{
AddonsManager *AM = AddonsManager::getInstance( p_intf );
CONNECT( AM, discoveryEnded(), spinnerAnimation, stop() );
spinnerAnimation->start();
AM->findInstalled();
AM->findNewAddons();
}
break;
default:
break;
}
return false;
}
void AddonsTab::moreInformation()
{
QModelIndex index = addonsView->selectionModel()->selectedIndexes().first();
if( !index.isValid() ) return;
AddonInfoDialog dlg( index, p_intf, this );
dlg.exec();
}
void AddonsTab::typeChanged( int i )
{
QComboBox *combo = qobject_cast<QComboBox *>( sender() );
int i_type = combo->itemData( i, Qt::UserRole ).toInt();
addonsModel->setTypeFilter( i_type );
QString help;
switch( i_type )
{
case ADDON_SKIN2:
help = qtr( "Skins customize player's appearance."
" You can activate them through preferences." );
break;
case ADDON_PLAYLIST_PARSER:
help = qtr( "Playlist parsers add new capabilities to read"
" internet streams or extract meta data." );
break;
case ADDON_SERVICE_DISCOVERY:
help = qtr( "Service discoveries adds new sources to your playlist"
" such as web radios, video websites, ..." );
break;
case ADDON_EXTENSION:
help = qtr( "Extensions brings various enhancements."
" Check descriptions for more details" );
break;
default:
helpLabel->setText("");
return;
}
helpLabel->setTextFormat( Qt::RichText );
helpLabel->setText( QString( "<img src=\":/menu/info\"/> %1" ).arg( help ) );
}
void AddonsTab::installChecked( int i )
{
if ( i == Qt::Checked )
addonsModel->setStatusFilter( ADDON_INSTALLED );
else
addonsModel->setStatusFilter( 0 );
}
/* Safe copy of the extension_t struct */ /* Safe copy of the extension_t struct */
ExtensionListModel::ExtensionCopy::ExtensionCopy( extension_t *p_ext ) ExtensionListModel::ExtensionCopy::ExtensionCopy( extension_t *p_ext )
{ {
...@@ -320,7 +505,7 @@ QVariant ExtensionListModel::ExtensionCopy::data( int role ) const ...@@ -320,7 +505,7 @@ QVariant ExtensionListModel::ExtensionCopy::data( int role ) const
case Qt::DecorationRole: case Qt::DecorationRole:
if ( !icon ) return QPixmap( ":/logo/vlc48.png" ); if ( !icon ) return QPixmap( ":/logo/vlc48.png" );
return *icon; return *icon;
case DescriptionRole: case SummaryRole:
return shortdesc; return shortdesc;
case VersionRole: case VersionRole:
return version; return version;
...@@ -328,7 +513,7 @@ QVariant ExtensionListModel::ExtensionCopy::data( int role ) const ...@@ -328,7 +513,7 @@ QVariant ExtensionListModel::ExtensionCopy::data( int role ) const
return author; return author;
case LinkRole: case LinkRole:
return url; return url;
case NameRole: case FilenameRole:
return name; return name;
default: default:
return QVariant(); return QVariant();
...@@ -336,6 +521,11 @@ QVariant ExtensionListModel::ExtensionCopy::data( int role ) const ...@@ -336,6 +521,11 @@ QVariant ExtensionListModel::ExtensionCopy::data( int role ) const
} }
/* Extensions list model for the QListView */ /* Extensions list model for the QListView */
ExtensionListModel::ExtensionListModel( QObject *parent )
: QAbstractListModel( parent ), EM( NULL )
{
}
ExtensionListModel::ExtensionListModel( QObject *parent, ExtensionsManager* EM_ ) ExtensionListModel::ExtensionListModel( QObject *parent, ExtensionsManager* EM_ )
: QAbstractListModel( parent ), EM( EM_ ) : QAbstractListModel( parent ), EM( EM_ )
...@@ -421,6 +611,244 @@ QModelIndex ExtensionListModel::index( int row, int column, ...@@ -421,6 +611,244 @@ QModelIndex ExtensionListModel::index( int row, int column,
return createIndex( row, 0, extensions.at( row ) ); return createIndex( row, 0, extensions.at( row ) );
} }
AddonsListModel::Addon::Addon( addon_entry_t *p_entry_ )
{
p_entry = p_entry_;
addon_entry_Hold( p_entry );
}
AddonsListModel::Addon::~Addon()
{
addon_entry_Release( p_entry );
}
bool AddonsListModel::Addon::operator==( const Addon & other ) const
{
//return data( IDRole ) == other.data( IDRole );
return p_entry == other.p_entry;
}
bool AddonsListModel::Addon::operator==( const addon_entry_t * p_other ) const
{
return p_entry == p_other;
}
QVariant AddonsListModel::Addon::data( int role ) const
{
QVariant returnval;
vlc_mutex_lock( &p_entry->lock );
switch( role )
{
case Qt::DisplayRole:
{
QString name = qfu( p_entry->psz_name );
if ( p_entry->e_state == ADDON_INSTALLED )
name.append( QString(" (%1)").arg( qtr("installed") ) );
returnval = name;
break;
}
case Qt::DecorationRole:
if ( p_entry->psz_image_data )
{
QPixmap pixmap;
pixmap.loadFromData( QByteArray::fromBase64( QByteArray( p_entry->psz_image_data ) ),
0,
Qt::AutoColor
);
returnval = pixmap;
}
else if ( p_entry->e_flags & ADDON_BROKEN )
returnval = QPixmap( ":/addons/broken" );
else
returnval = QPixmap( ":/addons/default" );
break;
case Qt::ToolTipRole:
{
if ( !( p_entry->e_flags & ADDON_MANAGEABLE ) )
{
returnval = qtr("This addon has been installed manually. VLC can't manage it by itself.");
}
break;
}
case SummaryRole:
returnval = qfu( p_entry->psz_summary );
break;
case DescriptionRole:
returnval = qfu( p_entry->psz_description );
break;
case TypeRole:
returnval = QVariant( (int) p_entry->e_type );
break;
case UUIDRole:
returnval = QByteArray( (const char *) p_entry->uuid, (int) sizeof( addon_uuid_t ) );
break;
case FlagsRole:
returnval = QVariant( (int) p_entry->e_flags );
break;
case StateRole:
returnval = QVariant( (int) p_entry->e_state );
break;
case DownloadsCountRole:
returnval = QVariant( (double) p_entry->i_downloads );
break;
case ScoreRole:
returnval = QVariant( (double) p_entry->i_score );
break;
case VersionRole:
returnval = QVariant( p_entry->psz_version );
break;
case AuthorRole:
returnval = qfu( p_entry->psz_author );
break;
case LinkRole:
returnval = qfu( p_entry->psz_source_uri );
break;
case FilenameRole:
{
QList<QString> list;
FOREACH_ARRAY( addon_file_t *p_file, p_entry->files )
list << qfu( p_file->psz_filename );
FOREACH_END();
returnval = QVariant( list );
break;
}
default:
break;
}
vlc_mutex_unlock( &p_entry->lock );
return returnval;
}
AddonsListModel::AddonsListModel( AddonsManager *AM_, QObject *parent )
:ExtensionListModel( parent ), AM( AM_ )
{
}
void AddonsListModel::addonAdded( addon_entry_t *p_entry )
{
beginInsertRows( QModelIndex(), addons.count(), addons.count() );
addons << new Addon( p_entry );
insertRow( addons.count() - 1 );
endInsertRows();
}
void AddonsListModel::addonChanged( const addon_entry_t *p_entry )
{
int row = 0;
foreach ( const Addon *addon, addons )
{
if ( *addon == p_entry )
{
emit dataChanged( index( row, 0 ), index( row, 0 ) );
break;
}
row++;
}
}
int AddonsListModel::rowCount( const QModelIndex & ) const
{
return addons.count();
}
Qt::ItemFlags AddonsListModel::flags( const QModelIndex &index ) const
{
Qt::ItemFlags i_flags = ExtensionListModel::flags( index );
int i_state = data( index, StateRole ).toInt();
if ( i_state == ADDON_UNINSTALLING || i_state == ADDON_INSTALLING )
{
i_flags &= !Qt::ItemIsEnabled;
}
i_flags |= Qt::ItemIsEditable;
return i_flags;
}
bool AddonsListModel::setData( const QModelIndex &index, const QVariant &value, int role )
{
/* We NEVER set values directly */
if ( role == StateRole )
{
int i_value = value.toInt();
if ( i_value == ADDON_INSTALLING )
{
AM->install( data( index, UUIDRole ).toByteArray() );
}
else if ( i_value == ADDON_UNINSTALLING )
{
AM->remove( data( index, UUIDRole ).toByteArray() );
}
}
else if ( role == StateRole + 1 )
{
emit dataChanged( index, index );
}
return true;
}
QVariant AddonsListModel::data( const QModelIndex& index, int role ) const
{
if( !index.isValid() )
return QVariant();
return ((Addon *)index.internalPointer())->data( role );
}
QModelIndex AddonsListModel::index( int row, int column,
const QModelIndex& ) const
{
if( column != 0 )
return QModelIndex();
if( row < 0 || row >= addons.count() )
return QModelIndex();
return createIndex( row, 0, addons.at( row ) );
}
/* Sort Filter */
AddonsSortFilterProxyModel::AddonsSortFilterProxyModel( QObject *parent )
: QSortFilterProxyModel( parent )
{
i_type_filter = -1;
i_status_filter = 0;
}
void AddonsSortFilterProxyModel::setTypeFilter( int type )
{
i_type_filter = type;
invalidateFilter();
}
void AddonsSortFilterProxyModel::setStatusFilter( int flags )
{
i_status_filter = flags;
invalidateFilter();
}
bool AddonsSortFilterProxyModel::filterAcceptsRow( int source_row,
const QModelIndex &source_parent ) const
{
if ( !QSortFilterProxyModel::filterAcceptsRow( source_row, source_parent ) )
return false;
QModelIndex item = sourceModel()->index( source_row, 0, source_parent );
if ( i_type_filter > -1 &&
item.data( AddonsListModel::TypeRole ).toInt() != i_type_filter )
return false;
if ( i_status_filter > 0 &&
( item.data( AddonsListModel::StateRole ).toInt() & i_status_filter ) != i_status_filter )
return false;
return true;
}
/* Extension List Widget Item */ /* Extension List Widget Item */
ExtensionItemDelegate::ExtensionItemDelegate( QObject *parent ) ExtensionItemDelegate::ExtensionItemDelegate( QObject *parent )
...@@ -478,7 +906,7 @@ void ExtensionItemDelegate::paint( QPainter *painter, ...@@ -478,7 +906,7 @@ void ExtensionItemDelegate::paint( QPainter *painter,
painter->setFont( font ); painter->setFont( font );
painter->drawText( textrect.translated( 0, option.fontMetrics.height() ), painter->drawText( textrect.translated( 0, option.fontMetrics.height() ),
Qt::AlignLeft, Qt::AlignLeft,
index.data( ExtensionListModel::DescriptionRole ).toString() ); index.data( ExtensionListModel::SummaryRole ).toString() );
painter->restore(); painter->restore();
} }
...@@ -504,6 +932,207 @@ void ExtensionItemDelegate::initStyleOption( QStyleOptionViewItem *option, ...@@ -504,6 +932,207 @@ void ExtensionItemDelegate::initStyleOption( QStyleOptionViewItem *option,
margins.top() + margins.bottom() ); margins.top() + margins.bottom() );
} }
AddonItemDelegate::AddonItemDelegate( QObject *parent )
: ExtensionItemDelegate( parent )
{
animator = NULL;
progressbar = NULL;
}
AddonItemDelegate::~AddonItemDelegate()
{
delete progressbar;
}
void AddonItemDelegate::paint( QPainter *painter,
const QStyleOptionViewItem &option,
const QModelIndex &index ) const
{
QStyleOptionViewItemV4 newopt = option;
int i_state = index.data( AddonsListModel::StateRole ).toInt();
if ( option.state.testFlag( QStyle::State_Editing ) )
newopt.rect.setRight( option.rect.right() - 100 );
ExtensionItemDelegate::paint( painter, newopt, index );
initStyleOption( &newopt, index );
painter->save();
painter->setRenderHint( QPainter::TextAntialiasing );
if ( newopt.state & QStyle::State_Selected )
painter->setPen( newopt.palette.highlightedText().color() );
/* Start below text */
QRect textrect( newopt.rect );
textrect.adjust( 2 * margins.left() + margins.right() + newopt.decorationSize.width(),
margins.top(),
- margins.right(),
- margins.bottom() - newopt.fontMetrics.height() );
textrect.translate( 0, newopt.fontMetrics.height() * 2 );
/* Version */
QString version = index.data( AddonsListModel::VersionRole ).toString();
if ( !version.isEmpty() )
painter->drawText( textrect, Qt::AlignLeft, qtr("Version %1").arg( version ) );
textrect.translate( 0, newopt.fontMetrics.height() );
/* Score */
double i_score = index.data( AddonsListModel::ScoreRole ).toDouble();
QPixmap scoreicon;
if ( i_score )
{
scoreicon = QPixmap( ":/addons/score" ).scaledToHeight(
newopt.fontMetrics.height(), Qt::SmoothTransformation );
int i_width = ( i_score / 5.0 ) * scoreicon.width();
/* Erase the end (value) of our pixmap with a shadow */
QPainter erasepainter( &scoreicon );
erasepainter.setCompositionMode( QPainter::CompositionMode_SourceIn );
erasepainter.fillRect( QRect( i_width, 0,
scoreicon.width() - i_width, scoreicon.height() ),
newopt.palette.color( QPalette::Dark ) );
erasepainter.end();
painter->drawPixmap( textrect.topLeft(), scoreicon );
}
/* Downloads # */
int i_downloads = index.data( AddonsListModel::DownloadsCountRole ).toInt();
if ( i_downloads )
painter->drawText( textrect.translated( scoreicon.width() + margins.left(), 0 ),
Qt::AlignLeft, qtr("%1 downloads").arg( i_downloads ) );
painter->restore();
if ( animator )
{
if ( animator->isRunning() && animator->getIndex() == index )
{
if ( i_state != ADDON_INSTALLING && i_state != ADDON_UNINSTALLING )
animator->run( false );
}
/* Create our installation progress overlay */
if ( i_state == ADDON_INSTALLING || i_state == ADDON_UNINSTALLING )
{
painter->save();
painter->setCompositionMode( QPainter::CompositionMode_SourceOver );
painter->fillRect( newopt.rect, QColor( 255, 255, 255, 128 ) );
if ( animator && index.isValid() )
{
animator->setIndex( index );
animator->run( true );
QSize adjustment = newopt.rect.size() / 4;
progressbar->setGeometry(
newopt.rect.adjusted( adjustment.width(), adjustment.height(),
-adjustment.width(), -adjustment.height() ) );
painter->drawPixmap( newopt.rect.left() + adjustment.width(),
newopt.rect.top() + adjustment.height(),
QPixmap::grabWidget( progressbar ) );
}
painter->restore();
}
}
}
QSize AddonItemDelegate::sizeHint( const QStyleOptionViewItem &option,
const QModelIndex &index ) const
{
if ( index.isValid() )
{
return QSize( 200, 4 * option.fontMetrics.height()
+ margins.top() + margins.bottom() );
}
else
return QSize();
}
QWidget *AddonItemDelegate::createEditor( QWidget *parent,
const QStyleOptionViewItem &option,
const QModelIndex &index) const
{
Q_UNUSED( option );
QWidget *editorWidget = new QWidget( parent );
QPushButton *installButton;
QPushButton *infoButton;
editorWidget->setLayout( new QHBoxLayout() );
editorWidget->layout()->setMargin( 0 );
infoButton = new QPushButton( QIcon( ":/menu/info" ),
qtr( "More information..." ) );
connect( infoButton, SIGNAL(clicked()), this, SIGNAL(showInfo()) );
editorWidget->layout()->addWidget( infoButton );
if ( ADDON_MANAGEABLE &
index.data( AddonsListModel::FlagsRole ).toInt() )
{
if ( index.data( AddonsListModel::StateRole ).toInt() == ADDON_INSTALLED )
installButton = new QPushButton( QIcon( ":/buttons/playlist/playlist_remove" ),
qtr("&Uninstall"), parent );
else
installButton = new QPushButton( QIcon( ":/buttons/playlist/playlist_add" ),
qtr("&Install"), parent );
CONNECT( installButton, clicked(), this, editButtonClicked() );
editorWidget->layout()->addWidget( installButton );
}
editorWidget->setSizePolicy( QSizePolicy::Maximum, QSizePolicy::Preferred );
return editorWidget;
}
void AddonItemDelegate::updateEditorGeometry( QWidget *editor,
const QStyleOptionViewItem &option,
const QModelIndex &index) const
{
Q_UNUSED( index );
QSize size = editor->sizeHint();
editor->setGeometry( option.rect.right() - size.width(),
option.rect.top() + ( option.rect.height() - size.height()),
size.width(),
size.height() );
}
void AddonItemDelegate::setModelData( QWidget *editor, QAbstractItemModel *model,
const QModelIndex &index ) const
{
model->setData( index, editor->property("Addon::state"), AddonsListModel::StateRole );
}
void AddonItemDelegate::setEditorData( QWidget *editor, const QModelIndex &index ) const
{
editor->setProperty("Addon::state", index.data( AddonsListModel::StateRole ) );
}
void AddonItemDelegate::setAnimator( DelegateAnimationHelper *animator_ )
{
if ( !progressbar )
{
QProgressBar *progress = new QProgressBar( );
progress->setMinimum( 0 );
progress->setMaximum( 0 );
progress->setTextVisible( false );
progressbar = progress;
}
animator = animator_;
}
void AddonItemDelegate::editButtonClicked()
{
QWidget *editor = qobject_cast<QWidget *>(sender()->parent());
int value = editor->property("Addon::state").toInt();
if ( ( value == ADDON_INSTALLED ) )
/* uninstall */
editor->setProperty("Addon::state", ADDON_UNINSTALLING );
else
/* install */
editor->setProperty("Addon::state", ADDON_INSTALLING );
emit commitData( editor );
emit closeEditor( editor );
}
/* "More information" dialog */ /* "More information" dialog */
ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index, ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index,
...@@ -552,7 +1181,7 @@ ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index, ...@@ -552,7 +1181,7 @@ ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index,
// Description // Description
label = new QLabel( this ); label = new QLabel( this );
label->setText( index.data(ExtensionListModel::DescriptionRole).toString() ); label->setText( index.data(ExtensionListModel::SummaryRole).toString() );
label->setWordWrap( true ); label->setWordWrap( true );
label->setOpenExternalLinks( true ); label->setOpenExternalLinks( true );
layout->addWidget( label, 4, 0, 1, -1 ); layout->addWidget( label, 4, 0, 1, -1 );
...@@ -571,7 +1200,7 @@ ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index, ...@@ -571,7 +1200,7 @@ ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index,
label = new QLabel( "<b>" + qtr( "File" ) + ":</b>", this ); label = new QLabel( "<b>" + qtr( "File" ) + ":</b>", this );
layout->addWidget( label, 6, 0, 1, 2 ); layout->addWidget( label, 6, 0, 1, 2 );
QLineEdit *line = QLineEdit *line =
new QLineEdit( index.data(ExtensionListModel::NameRole).toString(), this ); new QLineEdit( index.data(ExtensionListModel::FilenameRole).toString(), this );
line->setReadOnly( true ); line->setReadOnly( true );
layout->addWidget( line, 6, 2, 1, -1 ); layout->addWidget( line, 6, 2, 1, -1 );
...@@ -589,6 +1218,115 @@ ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index, ...@@ -589,6 +1218,115 @@ ExtensionInfoDialog::ExtensionInfoDialog( const QModelIndex &index,
setMinimumSize( 450, 350 ); setMinimumSize( 450, 350 );
} }
AddonInfoDialog::AddonInfoDialog( const QModelIndex &index,
intf_thread_t *p_intf, QWidget *parent )
: QVLCDialog( parent, p_intf )
{
// Let's be a modal dialog
setWindowModality( Qt::WindowModal );
// Window title
setWindowTitle( qtr( "About" ) + " " + index.data(Qt::DisplayRole).toString() );
// Layout
QGridLayout *layout = new QGridLayout( this );
QLabel *label;
// Icon
QLabel *iconLabel = new QLabel( this );
iconLabel->setFixedSize( 100, 100 );
QPixmap icon = index.data( Qt::DecorationRole ).value<QPixmap>();
icon.scaled( iconLabel->size(), Qt::KeepAspectRatio, Qt::SmoothTransformation );
iconLabel->setPixmap( icon );
iconLabel->setAlignment( Qt::AlignCenter | Qt::AlignTop );
layout->addWidget( iconLabel, 1, 0, 2, 1 );
// Title
label = new QLabel( index.data(Qt::DisplayRole).toString(), this );
QFont font = label->font();
font.setBold( true );
font.setPointSizeF( font.pointSizeF() * 1.3f );
label->setFont( font );
layout->addWidget( label, 0, 0, 1, -1 );
// HTML Content on right side
QTextEdit *textContent = new QTextEdit();
textContent->viewport()->setAutoFillBackground( false );
textContent->setAcceptRichText( true );
textContent->setBackgroundRole( QPalette::Window );
textContent->setFrameStyle( QFrame::NoFrame );
textContent->setAutoFillBackground( false );
textContent->setReadOnly( true );
layout->addWidget( textContent, 1, 1, 4, -1 );
// Type
QString type = AddonsManager::getAddonType( index.data(AddonsListModel::TypeRole).toInt() );
textContent->append( QString("<b>%1:</b> %2<br/>")
.arg( qtr("Type") ).arg( type ) );
// Version
QString version = index.data(ExtensionListModel::VersionRole).toString();
if ( !version.isEmpty() )
{
textContent->append( QString("<b>%1:</b> %2<br/>")
.arg( qtr("Version") ).arg( version ) );
}
// Author
QString author = index.data(ExtensionListModel::AuthorRole).toString();
if ( !author.isEmpty() )
{
textContent->append( QString("<b>%1:</b> %2<br/>")
.arg( qtr("Author") ).arg( author ) );
}
// Description
QString description = index.data(AddonsListModel::DescriptionRole).toString();
if ( !description.isEmpty() )
{
textContent->append( QString("<hr/>\n%1")
.arg( description.replace("\n", "<br/>") ) );
}
// URL
QString sourceUrl = index.data(ExtensionListModel::LinkRole).toString();
if ( !sourceUrl.isEmpty() )
{
label = new QLabel( "<b>" + qtr( "Website" ) + ":</b>", this );
layout->addWidget( label, 5, 0, 1, 2 );
label = new QLabel( QString("<a href=\"%1\">%2</a>")
.arg( sourceUrl ).arg( sourceUrl ), this );
label->setOpenExternalLinks( true );
layout->addWidget( label, 5, 2, 1, -1 );
}
// Script files
QList<QVariant> list = index.data(ExtensionListModel::FilenameRole).toList();
if ( ! list.empty() )
{
label = new QLabel( "<b>" + qtr( "Files" ) + ":</b>", this );
layout->addWidget( label, 6, 0, 1, 2 );
QComboBox *filesCombo = new QComboBox();
Q_FOREACH( const QVariant & file, list )
filesCombo->addItem( file.toString() );
layout->addWidget( filesCombo, 6, 2, 1, -1 );
}
// Close button
QDialogButtonBox *group = new QDialogButtonBox( this );
QPushButton *closeButton = new QPushButton( qtr( "&Close" ) );
group->addButton( closeButton, QDialogButtonBox::RejectRole );
BUTTONACT( closeButton, close() );
layout->addWidget( group, 7, 0, 1, -1 );
// Fix layout
layout->setColumnStretch( 2, 1 );
layout->setRowStretch( 4, 1 );
setMinimumSize( 640, 480 );
}
static QPixmap *loadPixmapFromData( char *data, int size ) static QPixmap *loadPixmapFromData( char *data, int size )
{ {
if( !data || size <= 0 ) if( !data || size <= 0 )
......
...@@ -28,10 +28,13 @@ ...@@ -28,10 +28,13 @@
#include "util/singleton.hpp" #include "util/singleton.hpp"
#include <vlc_extensions.h> #include <vlc_extensions.h>
#include <vlc_addons.h>
#include <QStringList> #include <QStringList>
#include <QTreeWidgetItem> #include <QTreeWidgetItem>
#include <QPushButton>
#include <QAbstractListModel> #include <QAbstractListModel>
#include <QSortFilterProxyModel>
#include <QStyledItemDelegate> #include <QStyledItemDelegate>
class QLabel; class QLabel;
...@@ -39,17 +42,22 @@ class QTabWidget; ...@@ -39,17 +42,22 @@ class QTabWidget;
class QComboBox; class QComboBox;
class QTreeWidget; class QTreeWidget;
class QLineEdit; class QLineEdit;
class QTextBrowser; //class QTextBrowser;
class QListView; class QListView;
class QStyleOptionViewItem; class QStyleOptionViewItem;
class QPainter; class QPainter;
class QKeyEvent; class QKeyEvent;
class PluginTab; class PluginTab;
class ExtensionTab; class ExtensionTab;
class AddonsTab;
class ExtensionListItem; class ExtensionListItem;
class SearchLineEdit; class SearchLineEdit;
class ExtensionCopy; class ExtensionCopy;
class ExtensionsManager; class ExtensionsManager;
class AddonsManager;
class PixmapAnimator;
class DelegateAnimationHelper;
class AddonsSortFilterProxyModel;
class PluginDialog : public QVLCFrame, public Singleton<PluginDialog> class PluginDialog : public QVLCFrame, public Singleton<PluginDialog>
{ {
...@@ -62,6 +70,7 @@ private: ...@@ -62,6 +70,7 @@ private:
QTabWidget *tabs; QTabWidget *tabs;
PluginTab *pluginTab; PluginTab *pluginTab;
ExtensionTab *extensionTab; ExtensionTab *extensionTab;
AddonsTab *addonsTab;
friend class Singleton<PluginDialog>; friend class Singleton<PluginDialog>;
}; };
...@@ -116,6 +125,28 @@ private: ...@@ -116,6 +125,28 @@ private:
friend class PluginDialog; friend class PluginDialog;
}; };
class AddonsTab : public QVLCFrame
{
Q_OBJECT
friend class PluginDialog;
private slots:
void moreInformation();
void typeChanged( int );
void installChecked( int );
private:
AddonsTab( intf_thread_t *p_intf );
virtual ~AddonsTab();
bool eventFilter ( QObject * watched, QEvent * event );
QLabel *helpLabel;
QListView *addonsView;
AddonsSortFilterProxyModel *addonsModel;
/* Wait spinner */
PixmapAnimator *spinnerAnimation;
};
class PluginTreeItem : public QTreeWidgetItem class PluginTreeItem : public QTreeWidgetItem
{ {
public: public:
...@@ -147,15 +178,16 @@ public: ...@@ -147,15 +178,16 @@ public:
}; };
ExtensionListModel( QObject *parent, ExtensionsManager *EM ); ExtensionListModel( QObject *parent, ExtensionsManager *EM );
ExtensionListModel( QObject *parent = 0 );
virtual ~ExtensionListModel(); virtual ~ExtensionListModel();
enum enum
{ {
DescriptionRole = Qt::UserRole, SummaryRole = Qt::UserRole,
VersionRole, VersionRole,
AuthorRole, AuthorRole,
LinkRole, LinkRole,
NameRole FilenameRole
}; };
virtual QVariant data( const QModelIndex& index, int role ) const; virtual QVariant data( const QModelIndex& index, int role ) const;
...@@ -163,17 +195,83 @@ public: ...@@ -163,17 +195,83 @@ public:
const QModelIndex& = QModelIndex() ) const; const QModelIndex& = QModelIndex() ) const;
virtual int rowCount( const QModelIndex& = QModelIndex() ) const; virtual int rowCount( const QModelIndex& = QModelIndex() ) const;
private slots: protected slots:
void updateList(); void updateList();
private: private:
ExtensionsManager *EM; ExtensionsManager *EM;
QList<ExtensionCopy*> extensions; QList<ExtensionCopy*> extensions;
}; };
class AddonsListModel: public ExtensionListModel
{
Q_OBJECT
public:
AddonsListModel( AddonsManager *AM, QObject *parent = 0 );
virtual QVariant data( const QModelIndex& index, int role ) const;
virtual QModelIndex index( int row, int column = 0,
const QModelIndex& = QModelIndex() ) const;
virtual int rowCount( const QModelIndex& = QModelIndex() ) const;
virtual Qt::ItemFlags flags( const QModelIndex &index ) const;
virtual bool setData( const QModelIndex &index, const QVariant &value, int role );
enum
{
TypeRole = FilenameRole + 1,
DescriptionRole,
UUIDRole,
FlagsRole,
StateRole,
DownloadsCountRole,
ScoreRole
};
protected slots:
void addonAdded( addon_entry_t * );
void addonChanged( const addon_entry_t * );
protected:
class Addon
{
public:
Addon( addon_entry_t * );
~Addon();
bool operator==( const Addon & other ) const;
bool operator==( const addon_entry_t * p_other ) const;
QVariant data( int ) const;
private:
addon_entry_t * p_entry;
};
QList<Addon*> addons;
AddonsManager *AM;
};
class AddonsSortFilterProxyModel : public QSortFilterProxyModel
{
Q_OBJECT
public:
AddonsSortFilterProxyModel( QObject *parent = 0 );
public slots:
virtual void setTypeFilter( int );
virtual void setStatusFilter( int );
protected:
virtual bool filterAcceptsRow( int, const QModelIndex & ) const;
private:
int i_type_filter;
int i_status_filter;
};
class ExtensionItemDelegate : public QStyledItemDelegate class ExtensionItemDelegate : public QStyledItemDelegate
{ {
Q_OBJECT
public: public:
ExtensionItemDelegate( QObject *parent ); ExtensionItemDelegate( QObject *parent );
virtual ~ExtensionItemDelegate(); virtual ~ExtensionItemDelegate();
...@@ -186,10 +284,42 @@ public: ...@@ -186,10 +284,42 @@ public:
virtual void initStyleOption( QStyleOptionViewItem *option, virtual void initStyleOption( QStyleOptionViewItem *option,
const QModelIndex &index ) const; const QModelIndex &index ) const;
private: protected:
QMargins margins; QMargins margins;
}; };
class AddonItemDelegate : public ExtensionItemDelegate
{
Q_OBJECT
public:
AddonItemDelegate( QObject *parent );
~AddonItemDelegate();
virtual void paint( QPainter *painter,
const QStyleOptionViewItem &option,
const QModelIndex &index ) const;
virtual QSize sizeHint( const QStyleOptionViewItem &option,
const QModelIndex &index ) const;
virtual QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const;
virtual void updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const;
virtual void setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const;
virtual void setEditorData(QWidget *editor, const QModelIndex &index) const;
void setAnimator( DelegateAnimationHelper *animator );
public slots:
void editButtonClicked();
signals:
void showInfo();
protected:
DelegateAnimationHelper *animator;
QWidget *progressbar;
};
class ExtensionInfoDialog : public QVLCDialog class ExtensionInfoDialog : public QVLCDialog
{ {
public: public:
...@@ -197,5 +327,12 @@ public: ...@@ -197,5 +327,12 @@ public:
intf_thread_t *p_intf, QWidget *parent ); intf_thread_t *p_intf, QWidget *parent );
}; };
class AddonInfoDialog : public QVLCDialog
{
public:
AddonInfoDialog( const QModelIndex &index,
intf_thread_t *p_intf, QWidget *parent );
};
#endif #endif
/*****************************************************************************
* addons_manager.cpp: Addons manager for Qt
****************************************************************************
* Copyright (C) 2013 VideoLAN and authors
*
* 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
* (at your option) any later version.
*
* 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.
*****************************************************************************/
#include "addons_manager.hpp"
#include <QApplication>
const QEvent::Type AddonManagerEvent::AddedEvent =
(QEvent::Type)QEvent::registerEventType();
const QEvent::Type AddonManagerEvent::ChangedEvent =
(QEvent::Type)QEvent::registerEventType();
const QEvent::Type AddonManagerEvent::DiscoveryEndedEvent =
(QEvent::Type)QEvent::registerEventType();
AddonsManager::AddonsManager( intf_thread_t *p_intf )
{
p_manager = addons_manager_New( VLC_OBJECT(p_intf) );
if ( !p_manager ) return;
vlc_event_manager_t *p_em = p_manager->p_event_manager;
vlc_event_attach( p_em, vlc_AddonFound, addonsEventsCallback, this );
vlc_event_attach( p_em, vlc_AddonsDiscoveryEnded, addonsEventsCallback, this );
vlc_event_attach( p_em, vlc_AddonChanged, addonsEventsCallback, this );
}
AddonsManager::~AddonsManager()
{
if ( p_manager )
addons_manager_Delete( p_manager );
}
void AddonsManager::findNewAddons()
{
addons_manager_Gather( p_manager, NULL );
}
void AddonsManager::findInstalled()
{
addons_manager_LoadCatalog( p_manager );
}
void AddonsManager::install( QByteArray id )
{
Q_ASSERT( id.size() == sizeof(addon_uuid_t) );
addon_uuid_t addonid;
memcpy( &addonid, id.constData(), sizeof(addon_uuid_t) );
addons_manager_Install( p_manager, addonid );
}
void AddonsManager::remove( QByteArray id )
{
Q_ASSERT( id.size() == sizeof(addon_uuid_t) );
addon_uuid_t addonid;
memcpy( &addonid, id.constData(), sizeof(addon_uuid_t) );
addons_manager_Remove( p_manager, addonid );
}
QString AddonsManager::getAddonType( int i_type )
{
switch ( i_type )
{
case ADDON_SKIN2:
return qtr( "Skins" );
case ADDON_PLAYLIST_PARSER:
return qtr("Playlist parsers");
case ADDON_SERVICE_DISCOVERY:
return qtr("Service Discovery");
case ADDON_EXTENSION:
return qtr("Extensions");
default:
return qtr("Unknown");
}
}
void AddonsManager::addonsEventsCallback( const vlc_event_t *event, void *data )
{
AddonsManager *me = ( AddonsManager * ) data;
QEvent *ev = NULL;
if ( event->type == vlc_AddonFound )
{
ev = new AddonManagerEvent( AddonManagerEvent::AddedEvent,
event->u.addon_generic_event.p_entry );
}
else if ( event->type == vlc_AddonsDiscoveryEnded )
{
ev = new QEvent( AddonManagerEvent::DiscoveryEndedEvent );
}
else if ( event->type == vlc_AddonChanged )
{
ev = new AddonManagerEvent( AddonManagerEvent::ChangedEvent,
event->u.addon_generic_event.p_entry );
}
if ( ev ) QApplication::postEvent( me, ev );
}
void AddonsManager::customEvent( QEvent *event )
{
if ( event->type() == AddonManagerEvent::AddedEvent )
{
AddonManagerEvent *ev = static_cast<AddonManagerEvent *>(event);
emit addonAdded( ev->entry() );
}
else if ( event->type() == AddonManagerEvent::ChangedEvent )
{
AddonManagerEvent *ev = static_cast<AddonManagerEvent *>(event);
emit addonChanged( ev->entry() );
}
else if ( event->type() == AddonManagerEvent::DiscoveryEndedEvent )
{
emit discoveryEnded();
}
}
/*****************************************************************************
* addons_manager.hpp: Addons manager for Qt
****************************************************************************
* Copyright (C) 2013 VideoLAN and authors
*
* 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
* (at your option) any later version.
*
* 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.
*****************************************************************************/
#ifndef ADDONS_MANAGER_HPP
#define ADDONS_MANAGER_HPP
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include "qt4.hpp"
#include "util/singleton.hpp"
#include <vlc_events.h>
#include <vlc_addons.h>
#include <QObject>
#include <QEvent>
class AddonManagerEvent : public QEvent
{
public:
static const QEvent::Type AddedEvent;
static const QEvent::Type ChangedEvent;
static const QEvent::Type DiscoveryEndedEvent;
AddonManagerEvent( QEvent::Type type, addon_entry_t *_p_entry )
: QEvent( type ), p_entry( _p_entry )
{
addon_entry_Hold( p_entry );
}
~AddonManagerEvent()
{
addon_entry_Release( p_entry );
}
addon_entry_t *entry() const { return p_entry; }
private:
addon_entry_t *p_entry;
};
class AddonsManager : public QObject, public Singleton<AddonsManager>
{
Q_OBJECT
friend class Singleton<AddonsManager>;
public:
AddonsManager( intf_thread_t * );
~AddonsManager();
static void addonsEventsCallback( const vlc_event_t *, void * );
void customEvent( QEvent * );
void install( QByteArray id );
void remove( QByteArray id );
static QString getAddonType( int );
signals:
void addonAdded( addon_entry_t * );
void addonChanged( const addon_entry_t * );
void discoveryEnded();
public slots:
void findNewAddons();
void findInstalled();
private:
addons_manager_t* p_manager;
};
#endif // ADDONS_MANAGER_HPP
...@@ -153,4 +153,9 @@ ...@@ -153,4 +153,9 @@
<file alias="katsomo">pixmaps/playlist/sidebar-icons/sd/katsomo.png</file> <file alias="katsomo">pixmaps/playlist/sidebar-icons/sd/katsomo.png</file>
<file alias="metachannels">pixmaps/playlist/sidebar-icons/sd/metachannels.png</file> <file alias="metachannels">pixmaps/playlist/sidebar-icons/sd/metachannels.png</file>
</qresource> </qresource>
<qresource prefix="/addons">
<file alias="default">pixmaps/addons/addon.png</file>
<file alias="broken">pixmaps/addons/addon_broken.png</file>
<file alias="score">pixmaps/addons/score.png</file>
</qresource>
</RCC> </RCC>
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment