You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3825 lines
112 KiB
3825 lines
112 KiB
// (c) 2004 Christian Muehlhaeuser <chris@chris.de>
|
|
// (c) 2005-2006 Martin Aumueller <aumuell@reserv.at>
|
|
// (c) 2005 Seb Ruiz <me@sebruiz.net>
|
|
// (c) 2006 T.R.Shashwath <trshash84@gmail.com>
|
|
// See COPYING file for licensing information
|
|
|
|
|
|
#define DEBUG_PREFIX "MediaBrowser"
|
|
|
|
#include <config.h>
|
|
|
|
#include "amarok.h"
|
|
#include "amarokconfig.h"
|
|
#include "app.h"
|
|
#include "browserToolBar.h"
|
|
#include "clicklineedit.h"
|
|
#include "collectiondb.h"
|
|
#include "colorgenerator.h"
|
|
#include "contextbrowser.h"
|
|
#include "debug.h"
|
|
#include "editfilterdialog.h"
|
|
#include "deviceconfiguredialog.h"
|
|
#include "mediadevicemanager.h"
|
|
#include "expression.h"
|
|
#include "hintlineedit.h"
|
|
#include "mediabrowser.h"
|
|
#include "medium.h"
|
|
#include "mediumpluginmanager.h"
|
|
#include "metabundle.h"
|
|
#include "mountpointmanager.h"
|
|
#include "playlist.h"
|
|
#include "playlistbrowser.h"
|
|
#include "playlistbrowseritem.h"
|
|
#include "playlistloader.h"
|
|
#include "pluginmanager.h"
|
|
#include "podcastbundle.h"
|
|
#include "scriptmanager.h"
|
|
#include "scrobbler.h"
|
|
#include "statusbar.h"
|
|
#include "transferdialog.h"
|
|
#include "browserToolBar.h"
|
|
|
|
#include <tqvbuttongroup.h>
|
|
#include <tqcheckbox.h>
|
|
#include <tqdatetime.h>
|
|
#include <tqdir.h>
|
|
#include <tqdom.h>
|
|
#include <tqfileinfo.h>
|
|
#include <tqgroupbox.h>
|
|
#include <tqheader.h>
|
|
#include <tqimage.h>
|
|
#include <tqlabel.h>
|
|
#include <tqobjectlist.h>
|
|
#include <tqpainter.h>
|
|
#include <tqradiobutton.h>
|
|
#include <tqsimplerichtext.h>
|
|
#include <tqtimer.h>
|
|
#include <tqtooltip.h> //TQToolTip::add()
|
|
|
|
#include <kapplication.h> //kapp
|
|
#include <kcombobox.h>
|
|
#include <kdirlister.h>
|
|
#include <kfiledialog.h>
|
|
#include <kglobal.h>
|
|
#include <kiconloader.h>
|
|
#include <kinputdialog.h>
|
|
#include <kio/job.h>
|
|
#include <klocale.h>
|
|
#include <kmessagebox.h>
|
|
#include <kmultipledrag.h>
|
|
#include <kpopupmenu.h>
|
|
#include <kprocess.h>
|
|
#include <kprogress.h>
|
|
#include <kpushbutton.h>
|
|
#include <krun.h>
|
|
#include <kstandarddirs.h> //locate file
|
|
#include <ktabbar.h>
|
|
#include <ktempfile.h>
|
|
#include <ktoolbarbutton.h> //ctor
|
|
#include <kurldrag.h> //dragObject()
|
|
#include <kactioncollection.h>
|
|
|
|
|
|
MediaBrowser *MediaBrowser::s_instance = 0;
|
|
|
|
TQPixmap *MediaItem::s_pixUnknown = 0;
|
|
TQPixmap *MediaItem::s_pixArtist = 0;
|
|
TQPixmap *MediaItem::s_pixAlbum = 0;
|
|
TQPixmap *MediaItem::s_pixFile = 0;
|
|
TQPixmap *MediaItem::s_pixTrack = 0;
|
|
TQPixmap *MediaItem::s_pixPodcast = 0;
|
|
TQPixmap *MediaItem::s_pixPlaylist = 0;
|
|
TQPixmap *MediaItem::s_pixInvisible = 0;
|
|
TQPixmap *MediaItem::s_pixStale = 0;
|
|
TQPixmap *MediaItem::s_pixOrphaned = 0;
|
|
TQPixmap *MediaItem::s_pixDirectory = 0;
|
|
TQPixmap *MediaItem::s_pixRootItem = 0;
|
|
TQPixmap *MediaItem::s_pixTransferFailed = 0;
|
|
TQPixmap *MediaItem::s_pixTransferBegin = 0;
|
|
TQPixmap *MediaItem::s_pixTransferEnd = 0;
|
|
|
|
bool MediaBrowser::isAvailable() //static
|
|
{
|
|
if( !MediaBrowser::instance() )
|
|
return false;
|
|
|
|
return true;
|
|
|
|
//to re-enable hiding, uncomment this and get rid of the return true above:
|
|
//return MediaBrowser::instance()->m_haveDevices;
|
|
}
|
|
|
|
class SpaceLabel : public TQLabel {
|
|
public:
|
|
SpaceLabel(TQWidget *parent)
|
|
: TQLabel(parent)
|
|
{
|
|
m_total = m_used = m_scheduled = 0;
|
|
setBackgroundMode(TQt::NoBackground);
|
|
}
|
|
|
|
void paintEvent(TQPaintEvent *e)
|
|
{
|
|
TQPainter p(this);
|
|
p.fillRect(e->rect(), colorGroup().brush(TQColorGroup::Background));
|
|
|
|
if(m_total > 0)
|
|
{
|
|
int used = int(float(m_used)/float(m_total)*width());
|
|
int scheduled = int(float(m_used + m_scheduled)/float(m_total)*width());
|
|
|
|
if(m_used > 0)
|
|
{
|
|
TQColor blueish(70,120,255);
|
|
if(e->rect().left() < used)
|
|
{
|
|
int right = used;
|
|
if(e->rect().right() < right)
|
|
right = e->rect().right();
|
|
p.fillRect(e->rect().left(), e->rect().top(),
|
|
used, e->rect().bottom()+1, TQBrush(blueish, TQt::SolidPattern));
|
|
}
|
|
}
|
|
|
|
if(m_scheduled > 0)
|
|
{
|
|
TQColor sched(70, 230, 120);
|
|
if(m_used + m_scheduled > m_total - m_total/200)
|
|
{
|
|
sched.setRgb( 255, 120, 120 );
|
|
}
|
|
int left = e->rect().left();
|
|
if(used > left)
|
|
left = used;
|
|
int right = e->rect().right();
|
|
if(scheduled < right)
|
|
right = scheduled;
|
|
p.fillRect(left, e->rect().top(), right, e->rect().bottom()+1, TQBrush(sched, TQt::SolidPattern));
|
|
}
|
|
|
|
if(m_used + m_scheduled < m_total)
|
|
{
|
|
TQColor grey(180, 180, 180);
|
|
int left = e->rect().left();
|
|
if(scheduled > left)
|
|
left = scheduled;
|
|
int right = e->rect().right();
|
|
p.fillRect(left, e->rect().top(), right, e->rect().bottom()+1, colorGroup().brush(TQColorGroup::Background));
|
|
}
|
|
}
|
|
TQLabel::paintEvent(e);
|
|
}
|
|
|
|
KIO::filesize_t m_total;
|
|
KIO::filesize_t m_used;
|
|
KIO::filesize_t m_scheduled;
|
|
};
|
|
|
|
class DummyMediaDevice : public MediaDevice
|
|
{
|
|
public:
|
|
DummyMediaDevice() : MediaDevice()
|
|
{
|
|
m_name = i18n( "No Device Available" );
|
|
m_type = "dummy-mediadevice";
|
|
m_medium = Medium( "DummyDevice", "DummyDevice" );
|
|
}
|
|
void init( MediaBrowser *browser ) { MediaDevice::init( browser ); }
|
|
virtual ~DummyMediaDevice() {}
|
|
virtual bool isConnected() { return false; }
|
|
virtual MediaItem* trackExists(const MetaBundle&) { return 0; }
|
|
virtual bool lockDevice(bool) { return true; }
|
|
virtual void unlockDevice() {}
|
|
virtual bool openDevice( bool silent )
|
|
{
|
|
if( !silent )
|
|
{
|
|
//TQString msg = i18n( "Sorry, you do not have a supported portable music player." );
|
|
//Amarok::StatusBar::instance()->longMessage( msg, KDE::StatusBar::Sorry );
|
|
}
|
|
return false;
|
|
}
|
|
virtual bool closeDevice() { return false; }
|
|
virtual void synchronizeDevice() {}
|
|
virtual MediaItem* copyTrackToDevice(const MetaBundle&) { return 0; }
|
|
virtual int deleteItemFromDevice(MediaItem*, int) { return -1; }
|
|
};
|
|
|
|
|
|
MediaBrowser::MediaBrowser( const char *name )
|
|
: TQVBox( 0, name )
|
|
, m_timer( new TQTimer( this ) )
|
|
, m_currentDevice( m_devices.end() )
|
|
, m_waitForTranscode( false )
|
|
, m_quitting( false )
|
|
{
|
|
s_instance = this;
|
|
|
|
// preload pixmaps used in browser
|
|
KIconLoader iconLoader;
|
|
MediaItem::s_pixUnknown = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "unknown" ), KIcon::Toolbar, KIcon::SizeSmall ));
|
|
MediaItem::s_pixTrack = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "playlist" ), KIcon::Toolbar, KIcon::SizeSmall ));
|
|
MediaItem::s_pixFile = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "sound" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixPodcast = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "podcast" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixPlaylist = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "playlist" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixRootItem = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "files2" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
// history
|
|
// favorites
|
|
// collection
|
|
// folder
|
|
// folder_red
|
|
// player_playlist_2
|
|
// cancel
|
|
// sound
|
|
MediaItem::s_pixArtist = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "personal" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixAlbum = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "cdrom_unmount" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixInvisible = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "cancel" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixStale = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "cancel" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixOrphaned = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "cancel" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixDirectory = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "folder" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixTransferBegin = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "play" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixTransferEnd = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "stop" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
MediaItem::s_pixTransferFailed = new TQPixmap(iconLoader.loadIcon( Amarok::icon( "cancel" ), KIcon::Toolbar, KIcon::SizeSmall ) );
|
|
|
|
setSpacing( 4 );
|
|
|
|
m_toolbar = new Browser::ToolBar( this );
|
|
m_toolbar->setIconText( KToolBar::IconTextRight, false );
|
|
|
|
m_toolbar->insertButton( "connect_creating", CONNECT, true, i18n("Connect") );
|
|
TQToolTip::add( m_toolbar->getButton(CONNECT), i18n( "Connect media device" ) );
|
|
|
|
m_toolbar->insertButton( "player_eject", DISCONNECT, true, i18n("Disconnect") );
|
|
TQToolTip::add( m_toolbar->getButton(DISCONNECT), i18n( "Disconnect media device" ) );
|
|
|
|
m_toolbar->insertButton( "rebuild", TRANSFER, true, i18n("Transfer") );
|
|
TQToolTip::add( m_toolbar->getButton(TRANSFER), i18n( "Transfer tracks to media device" ) );
|
|
|
|
m_toolbar->insertLineSeparator();
|
|
|
|
// m_toolbar->setIconText( KToolBar::IconTextRight, true );
|
|
m_toolbar->insertButton( Amarok::icon( "add_playlist" ), CUSTOM, TQT_SIGNAL( clicked() ), TQT_TQOBJECT(this), TQT_SLOT( customClicked() ), true, "custom" );
|
|
TQToolTip::add( m_toolbar->getButton(TRANSFER), i18n( "Transfer tracks to media device" ) );
|
|
|
|
m_toolbar->setIconText( KToolBar::IconOnly, false );
|
|
|
|
m_toolbar->insertButton( Amarok::icon( "configure" ), CONFIGURE, true, i18n("Configure") );
|
|
TQToolTip::add( m_toolbar->getButton(CONFIGURE), i18n( "Configure device" ) );
|
|
|
|
|
|
m_deviceCombo = new KComboBox( this );
|
|
|
|
// searching/filtering
|
|
{ //<Search LineEdit>
|
|
KToolBar* searchToolBar = new Browser::ToolBar( this );
|
|
KToolBarButton *button = new KToolBarButton( "locationbar_erase", 0, searchToolBar );
|
|
m_searchEdit = new ClickLineEdit( i18n( "Enter search terms here" ), searchToolBar );
|
|
KPushButton *filterButton = new KPushButton("...", searchToolBar, "filter");
|
|
searchToolBar->setStretchableWidget( m_searchEdit );
|
|
m_searchEdit->setFrame( TQFrame::Sunken );
|
|
|
|
connect( button, TQT_SIGNAL( clicked() ), m_searchEdit, TQT_SLOT( clear() ) );
|
|
connect( filterButton, TQT_SIGNAL( clicked() ), TQT_SLOT( slotEditFilter() ) );
|
|
|
|
TQToolTip::add( button, i18n( "Clear filter" ) );
|
|
TQToolTip::add( m_searchEdit, i18n( "Enter space-separated terms to search" ) );
|
|
TQToolTip::add( filterButton, i18n( "Click to edit filter" ) );
|
|
} //</Search LineEdit>
|
|
|
|
connect( m_timer, TQT_SIGNAL( timeout() ), TQT_SLOT( slotSetFilter() ) );
|
|
connect( m_searchEdit, TQT_SIGNAL( textChanged( const TQString& ) ), TQT_SLOT( slotSetFilterTimeout() ) );
|
|
connect( m_searchEdit, TQT_SIGNAL( returnPressed() ), TQT_SLOT( slotSetFilter() ) );
|
|
|
|
// connect to device manager
|
|
connect( MediaDeviceManager::instance(), TQT_SIGNAL( mediumAdded(const Medium *, TQString) ),
|
|
TQT_SLOT( mediumAdded(const Medium *, TQString) ) );
|
|
connect( MediaDeviceManager::instance(), TQT_SIGNAL( mediumChanged(const Medium *, TQString) ),
|
|
TQT_SLOT( mediumChanged(const Medium *, TQString) ) );
|
|
connect( MediaDeviceManager::instance(), TQT_SIGNAL( mediumRemoved(const Medium *, TQString) ),
|
|
TQT_SLOT( mediumRemoved(const Medium *, TQString) ) );
|
|
|
|
|
|
// we always have a dummy device
|
|
m_pluginName[ i18n( "Disable" ) ] = "dummy-mediadevice";
|
|
m_pluginAmarokName["dummy-mediadevice"] = i18n( "Disable" );
|
|
m_pluginName[ i18n( "Do not handle" ) ] = "ignore";
|
|
m_pluginAmarokName["ignore"] = i18n( "Do not handle" );
|
|
// query available device plugins
|
|
m_plugins = PluginManager::query( "[X-KDE-Amarok-plugintype] == 'mediadevice'" );
|
|
for( KTrader::OfferList::ConstIterator it = m_plugins.begin(); it != m_plugins.end(); ++it ) {
|
|
// Save name properties in TQMap for lookup
|
|
m_pluginName[(*it)->name()] = (*it)->property( "X-KDE-Amarok-name" ).toString();
|
|
m_pluginAmarokName[(*it)->property( "X-KDE-Amarok-name" ).toString()] = (*it)->name();
|
|
}
|
|
|
|
m_views = new TQVBox( this );
|
|
m_queue = new MediaQueue( this );
|
|
m_progressBox = new TQHBox( this );
|
|
m_progress = new KProgress( m_progressBox );
|
|
m_cancelButton = new KPushButton( SmallIconSet( Amarok::icon( "cancel" ) ), i18n("Cancel"), m_progressBox );
|
|
|
|
|
|
m_stats = new SpaceLabel(this);
|
|
|
|
m_progressBox->hide();
|
|
|
|
MediaDevice *dev = new DummyMediaDevice();
|
|
dev->init( this );
|
|
addDevice( dev );
|
|
activateDevice( 0, false );
|
|
queue()->load( Amarok::saveLocation() + "transferlist.xml" );
|
|
queue()->computeSize();
|
|
|
|
setFocusProxy( m_queue );
|
|
|
|
updateStats();
|
|
|
|
TQMap<TQString, Medium*> mmap = MediaDeviceManager::instance()->getMediumMap();
|
|
|
|
bool newflag = false;
|
|
//This deals with <strike>auto-detectable</strike> ALL devices!
|
|
for( TQMap<TQString, Medium*>::Iterator it = mmap.begin();
|
|
it != mmap.end();
|
|
it++ )
|
|
{
|
|
TQString handler = Amarok::config( "MediaBrowser" )->readEntry( (*it)->id() );
|
|
//debug() << "[MediaBrowser] (*it)->id() = " << (*it)->id() << ", handler = " << handler << endl;
|
|
if( handler.isEmpty() )
|
|
{
|
|
//this should probably never be the case with a manually added device, unless amarokrc's been messed with
|
|
Amarok::config( "MediaBrowser" )->writeEntry( (*it)->id(), "ignore" );
|
|
newflag = true;
|
|
mediumAdded( *it, (*it)->name(), true );
|
|
}
|
|
//and this definitely shouldn't!
|
|
else if( handler != "deleted" )
|
|
mediumAdded( *it, (*it)->name(), true );
|
|
}
|
|
|
|
if ( newflag )
|
|
Amarok::StatusBar::instance()->longMessageThreadSafe(
|
|
i18n("Amarok has detected new portable media devices.\n"
|
|
"Go to the \"Media Devices\" pane of the configuration\n"
|
|
"dialog to choose a plugin for these devices.") );
|
|
|
|
connect( m_toolbar->getButton(CONNECT), TQT_SIGNAL( clicked() ), TQT_SLOT( connectClicked() ) );
|
|
connect( m_toolbar->getButton(DISCONNECT), TQT_SIGNAL( clicked() ), TQT_SLOT( disconnectClicked() ) );
|
|
connect( m_toolbar->getButton(TRANSFER), TQT_SIGNAL( clicked() ), TQT_SLOT( transferClicked() ) );
|
|
connect( m_toolbar->getButton(CONFIGURE), TQT_SIGNAL( clicked() ), TQT_SLOT( config() ) );
|
|
|
|
connect( m_deviceCombo, TQT_SIGNAL( activated( int ) ), TQT_SLOT( activateDevice( int ) ) );
|
|
|
|
connect( m_cancelButton, TQT_SIGNAL( clicked() ), TQT_SLOT( cancelClicked() ) );
|
|
connect( pApp, TQT_SIGNAL( prepareToQuit() ), TQT_SLOT( prepareToQuit() ) );
|
|
connect( CollectionDB::instance(), TQT_SIGNAL( tagsChanged( const MetaBundle& ) ),
|
|
TQT_SLOT( tagsChanged( const MetaBundle& ) ) );
|
|
|
|
m_haveDevices = false;
|
|
TQMap<TQString,TQString> savedDevices = Amarok::config( "MediaBrowser" )->entryMap( "MediaBrowser" );
|
|
for( TQMap<TQString,TQString>::Iterator it = savedDevices.begin();
|
|
it != savedDevices.end();
|
|
++it )
|
|
{
|
|
if( it.data() != "deleted" && it.data() != "ignore" )
|
|
{
|
|
m_haveDevices = true;
|
|
break;
|
|
}
|
|
}
|
|
emit availabilityChanged( m_haveDevices );
|
|
}
|
|
|
|
bool
|
|
MediaBrowser::blockQuit() const
|
|
{
|
|
for( TQValueList<MediaDevice *>::const_iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
++it )
|
|
{
|
|
if( *it && (*it)->isConnected() )
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::tagsChanged( const MetaBundle &bundle )
|
|
{
|
|
m_itemMapMutex.lock();
|
|
debug() << "tags changed for " << bundle.url().url() << endl;
|
|
ItemMap::iterator it = m_itemMap.find( bundle.url().url() );
|
|
if( it != m_itemMap.end() )
|
|
{
|
|
MediaItem *item = *it;
|
|
m_itemMapMutex.unlock();
|
|
if( item->device() )
|
|
{
|
|
item->device()->tagsChanged( item, bundle );
|
|
}
|
|
else
|
|
{
|
|
// it's an item on the transfer queue
|
|
item->setBundle( new MetaBundle( bundle ) );
|
|
|
|
TQString text = item->bundle()->prettyTitle();
|
|
if( text.isEmpty() || (!item->bundle()->isValidMedia() && !item->bundle()->podcastBundle()) )
|
|
text = item->bundle()->url().prettyURL();
|
|
if( !item->m_playlistName.isNull() )
|
|
{
|
|
text += " (" + item->m_playlistName + ')';
|
|
}
|
|
item->setText( 0, text);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_itemMapMutex.unlock();
|
|
}
|
|
}
|
|
|
|
bool
|
|
MediaBrowser::getBundle( const KURL &url, MetaBundle *bundle ) const
|
|
{
|
|
TQMutexLocker locker( &m_itemMapMutex );
|
|
ItemMap::const_iterator it = m_itemMap.find( url.url() );
|
|
if( it == m_itemMap.end() )
|
|
return false;
|
|
|
|
if( bundle )
|
|
*bundle = TQDeepCopy<MetaBundle>( *(*it)->bundle() );
|
|
|
|
return true;
|
|
}
|
|
|
|
KURL
|
|
MediaBrowser::getProxyUrl( const KURL& daapUrl ) const
|
|
{
|
|
DEBUG_BLOCK
|
|
KURL url;
|
|
MediaDevice* dc = dynamic_cast<MediaDevice*>( queryList( "DaapClient" )->getFirst() );
|
|
if( dc )
|
|
url = dc->getProxyUrl( daapUrl );
|
|
return url;
|
|
}
|
|
|
|
MediaDevice *
|
|
MediaBrowser::currentDevice() const
|
|
{
|
|
TQValueList<MediaDevice *>::const_iterator current = m_currentDevice;
|
|
if( current != m_devices.constEnd() )
|
|
{
|
|
return *m_currentDevice;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
MediaDevice *
|
|
MediaBrowser::deviceFromId( const TQString &id ) const
|
|
{
|
|
for( TQValueList<MediaDevice *>::const_iterator it = m_devices.constBegin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( (*it)->uniqueId() == id )
|
|
return (*it);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::activateDevice( const MediaDevice *dev )
|
|
{
|
|
int index = 0;
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( *it == dev )
|
|
{
|
|
activateDevice( index );
|
|
break;
|
|
}
|
|
index++;
|
|
}
|
|
}
|
|
|
|
void
|
|
MediaBrowser::activateDevice( int index, bool skipDummy )
|
|
{
|
|
if( currentDevice() && currentDevice()->customAction() )
|
|
{
|
|
currentDevice()->customAction()->unplug( m_toolbar );
|
|
m_toolbar->hide();
|
|
m_toolbar->show();
|
|
}
|
|
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
(*it)->view()->hide();
|
|
}
|
|
|
|
if( index < 0 )
|
|
{
|
|
m_currentDevice = m_devices.end();
|
|
return;
|
|
}
|
|
|
|
if( skipDummy )
|
|
index++;
|
|
|
|
if( (uint)index >= m_devices.count() )
|
|
{
|
|
m_currentDevice = m_devices.end();
|
|
updateButtons();
|
|
queue()->computeSize();
|
|
updateStats();
|
|
return;
|
|
}
|
|
|
|
m_currentDevice = m_devices.at( index );
|
|
if( currentDevice() )
|
|
{
|
|
currentDevice()->view()->show();
|
|
if( currentDevice()->customAction() )
|
|
{
|
|
m_toolbar->setIconText( KToolBar::IconTextRight, false );
|
|
currentDevice()->customAction()->plug( m_toolbar );
|
|
m_toolbar->hide();
|
|
m_toolbar->show();
|
|
}
|
|
}
|
|
m_deviceCombo->setCurrentItem( index-1 );
|
|
|
|
updateButtons();
|
|
queue()->computeSize();
|
|
updateStats();
|
|
}
|
|
|
|
void
|
|
MediaBrowser::addDevice( MediaDevice *device )
|
|
{
|
|
m_devices.append( device );
|
|
|
|
device->loadConfig();
|
|
|
|
if( device->autoConnect() )
|
|
{
|
|
device->connectDevice( true );
|
|
updateButtons();
|
|
}
|
|
|
|
updateDevices();
|
|
}
|
|
|
|
void
|
|
MediaBrowser::removeDevice( MediaDevice *device )
|
|
{
|
|
DEBUG_BLOCK
|
|
|
|
debug() << "remove device: type=" << device->deviceType() << endl;
|
|
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( *it == device )
|
|
{
|
|
bool current = (it == m_currentDevice);
|
|
m_devices.remove( device );
|
|
if( current )
|
|
activateDevice( 0, false );
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( device->isConnected() )
|
|
{
|
|
if( device->disconnectDevice( false /* don't run post-disconnect command */ ) )
|
|
unloadDevicePlugin( device );
|
|
else
|
|
{
|
|
debug() << "Cannot remove device because disconnect failed" << endl;
|
|
Amarok::StatusBar::instance()->longMessage(
|
|
i18n( "Cannot remove device because disconnect failed" ),
|
|
KDE::StatusBar::Warning );
|
|
}
|
|
}
|
|
else
|
|
unloadDevicePlugin( device );
|
|
|
|
updateDevices();
|
|
}
|
|
|
|
void
|
|
MediaBrowser::updateDevices()
|
|
{
|
|
m_deviceCombo->clear();
|
|
uint i = 0;
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( m_devices.count() > 1 && dynamic_cast<DummyMediaDevice *>(*it) )
|
|
continue;
|
|
TQString name = (*it)->name();
|
|
if( !(*it)->deviceNode().isEmpty() )
|
|
{
|
|
name = i18n( "%1 at %2" ).arg( name, (*it)->deviceNode() );
|
|
}
|
|
if( (*it)->hasMountPoint() && !(*it)->mountPoint().isEmpty() )
|
|
{
|
|
name += i18n( " (mounted at %1)" ).arg( (*it)->mountPoint() );
|
|
}
|
|
m_deviceCombo->insertItem( name, i );
|
|
if( it == m_currentDevice )
|
|
{
|
|
m_deviceCombo->setCurrentItem( i );
|
|
}
|
|
i++;
|
|
}
|
|
m_deviceCombo->setEnabled( m_devices.count() > 1 );
|
|
m_haveDevices = m_devices.count() > 1;
|
|
emit availabilityChanged( m_haveDevices );
|
|
}
|
|
|
|
TQStringList
|
|
MediaBrowser::deviceNames() const
|
|
{
|
|
TQStringList list;
|
|
|
|
for( TQValueList<MediaDevice *>::const_iterator it = m_devices.constBegin();
|
|
it != m_devices.constEnd();
|
|
it++ )
|
|
{
|
|
TQString name = (*it)->name();
|
|
list << name;
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
bool
|
|
MediaBrowser::deviceSwitch( const TQString &name )
|
|
{
|
|
int index = 0;
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( (*it)->name() == name )
|
|
{
|
|
activateDevice( index, false );
|
|
return true;
|
|
}
|
|
index++;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::transcodingFinished( const TQString &src, const TQString &dst )
|
|
{
|
|
KURL srcJob = KURL::fromPathOrURL( m_transcodeSrc );
|
|
KURL srcResult = KURL::fromPathOrURL( src );
|
|
|
|
if( srcJob.path() == srcResult.path() )
|
|
{
|
|
m_transcodedUrl = KURL::fromPathOrURL( dst );
|
|
m_waitForTranscode = false;
|
|
}
|
|
else
|
|
{
|
|
debug() << "transcoding for " << src << " finished, "
|
|
<< "but we are waiting for " << m_transcodeSrc << " -- aborting" << endl;
|
|
m_waitForTranscode = false;
|
|
}
|
|
}
|
|
|
|
KURL
|
|
MediaBrowser::transcode( const KURL &src, const TQString &filetype )
|
|
{
|
|
const ScriptManager* const sm = ScriptManager::instance();
|
|
|
|
if( sm->transcodeScriptRunning().isEmpty() )
|
|
{
|
|
debug() << "cannot transcode with no transcoder registered" << endl;
|
|
return KURL();
|
|
}
|
|
|
|
m_waitForTranscode = true;
|
|
m_transcodeSrc = src.url();
|
|
m_transcodedUrl = KURL();
|
|
ScriptManager::instance()->notifyTranscode( src.url(), filetype );
|
|
|
|
while( m_waitForTranscode && sm->transcodeScriptRunning() != TQString() )
|
|
{
|
|
usleep( 10000 );
|
|
kapp->processEvents( 100 );
|
|
}
|
|
|
|
return m_transcodedUrl;
|
|
}
|
|
|
|
|
|
void
|
|
MediaBrowser::slotSetFilterTimeout() //SLOT
|
|
{
|
|
m_timer->start( 280, true ); //stops the timer for us first
|
|
}
|
|
|
|
void
|
|
MediaBrowser::slotSetFilter() //SLOT
|
|
{
|
|
m_timer->stop();
|
|
|
|
if( currentDevice() )
|
|
currentDevice()->view()->setFilter( m_searchEdit->text() );
|
|
}
|
|
|
|
void
|
|
MediaBrowser::slotSetFilter( const TQString &text )
|
|
{
|
|
m_searchEdit->setText( text );
|
|
slotSetFilter();
|
|
}
|
|
|
|
void
|
|
MediaBrowser::slotEditFilter()
|
|
{
|
|
EditFilterDialog *fd = new EditFilterDialog( this, true, m_searchEdit->text() );
|
|
connect( fd, TQT_SIGNAL(filterChanged(const TQString &)), TQT_SLOT(slotSetFilter(const TQString &)) );
|
|
if( fd->exec() )
|
|
m_searchEdit->setText( fd->filter() );
|
|
delete fd;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::prepareToQuit()
|
|
{
|
|
m_waitForTranscode = false;
|
|
m_quitting = true;
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
++it )
|
|
{
|
|
if( (*it)->isConnected() )
|
|
(*it)->disconnectDevice( false /* don't unmount */ );
|
|
}
|
|
}
|
|
|
|
MediaBrowser::~MediaBrowser()
|
|
{
|
|
debug() << "having to remove " << m_devices.count() << " devices" << endl;
|
|
while( !m_devices.isEmpty() )
|
|
{
|
|
removeDevice( m_devices.last() );
|
|
}
|
|
|
|
queue()->save( Amarok::saveLocation() + "transferlist.xml" );
|
|
|
|
delete m_deviceCombo;
|
|
delete m_queue;
|
|
}
|
|
|
|
|
|
MediaItem::MediaItem( TQListView* parent )
|
|
: KListViewItem( parent )
|
|
{
|
|
init();
|
|
}
|
|
|
|
MediaItem::MediaItem( TQListViewItem* parent )
|
|
: KListViewItem( parent )
|
|
{
|
|
init();
|
|
}
|
|
|
|
MediaItem::MediaItem( TQListView* parent, TQListViewItem* after )
|
|
: KListViewItem( parent, after )
|
|
{
|
|
init();
|
|
}
|
|
|
|
MediaItem::MediaItem( TQListViewItem* parent, TQListViewItem* after )
|
|
: KListViewItem( parent, after )
|
|
{
|
|
init();
|
|
}
|
|
|
|
MediaItem::~MediaItem()
|
|
{
|
|
setBundle( 0 );
|
|
}
|
|
|
|
void
|
|
MediaItem::init()
|
|
{
|
|
m_bundle=0;
|
|
m_order=0;
|
|
m_type=UNKNOWN;
|
|
m_playlistName = TQString();
|
|
m_device=0;
|
|
m_flags=0;
|
|
setExpandable( false );
|
|
setDragEnabled( true );
|
|
setDropEnabled( true );
|
|
}
|
|
|
|
void
|
|
MediaItem::setBundle( MetaBundle *bundle )
|
|
{
|
|
MediaBrowser::instance()->m_itemMapMutex.lock();
|
|
if( m_bundle )
|
|
{
|
|
TQString itemUrl = url().url();
|
|
MediaBrowser::ItemMap::iterator it = MediaBrowser::instance()->m_itemMap.find( itemUrl );
|
|
if( it != MediaBrowser::instance()->m_itemMap.end() && *it == this )
|
|
MediaBrowser::instance()->m_itemMap.remove( itemUrl );
|
|
}
|
|
delete m_bundle;
|
|
m_bundle = bundle;
|
|
|
|
if( m_bundle )
|
|
{
|
|
TQString itemUrl = url().url();
|
|
MediaBrowser::ItemMap::iterator it = MediaBrowser::instance()->m_itemMap.find( itemUrl );
|
|
if( it == MediaBrowser::instance()->m_itemMap.end() )
|
|
MediaBrowser::instance()->m_itemMap[itemUrl] = this;
|
|
}
|
|
MediaBrowser::instance()->m_itemMapMutex.unlock();
|
|
}
|
|
|
|
void MediaItem::paintCell( TQPainter *p, const TQColorGroup &cg, int column, int width, int align )
|
|
{
|
|
switch( type() )
|
|
{
|
|
case INVISIBLE:
|
|
case PODCASTSROOT:
|
|
case PLAYLISTSROOT:
|
|
case ORPHANEDROOT:
|
|
case STALEROOT:
|
|
{
|
|
TQFont font( p->font() );
|
|
font.setBold( true );
|
|
p->setFont( font );
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
KListViewItem::paintCell( p, cg, column, width, align );
|
|
}
|
|
|
|
const MetaBundle *
|
|
MediaItem::bundle() const
|
|
{
|
|
return m_bundle;
|
|
}
|
|
|
|
KURL
|
|
MediaItem::url() const
|
|
{
|
|
if( bundle() )
|
|
return bundle()->url();
|
|
else
|
|
return KURL();
|
|
}
|
|
|
|
bool
|
|
MediaItem::isFileBacked() const
|
|
{
|
|
switch( type() )
|
|
{
|
|
case ARTIST:
|
|
case ALBUM:
|
|
case PODCASTSROOT:
|
|
case PODCASTCHANNEL:
|
|
case PLAYLISTSROOT:
|
|
case PLAYLIST:
|
|
case PLAYLISTITEM:
|
|
case INVISIBLEROOT:
|
|
case STALEROOT:
|
|
case STALE:
|
|
case ORPHANEDROOT:
|
|
return false;
|
|
|
|
case UNKNOWN:
|
|
case TRACK:
|
|
case ORPHANED:
|
|
case INVISIBLE:
|
|
case PODCASTITEM:
|
|
case DIRECTORY:
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
long
|
|
MediaItem::size() const
|
|
{
|
|
if( !isFileBacked() )
|
|
return 0;
|
|
|
|
if( bundle() )
|
|
return bundle()->filesize();
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
MediaItem::setType( Type type )
|
|
{
|
|
m_type=type;
|
|
|
|
setDragEnabled(true);
|
|
setDropEnabled(false);
|
|
|
|
switch(m_type)
|
|
{
|
|
case UNKNOWN:
|
|
setPixmap(0, *s_pixUnknown);
|
|
break;
|
|
case INVISIBLE:
|
|
case TRACK:
|
|
setPixmap(0, *s_pixFile);
|
|
break;
|
|
case PLAYLISTITEM:
|
|
setPixmap(0, *s_pixTrack);
|
|
setDropEnabled(true);
|
|
break;
|
|
case ARTIST:
|
|
setPixmap(0, *s_pixArtist);
|
|
break;
|
|
case ALBUM:
|
|
setPixmap(0, *s_pixAlbum);
|
|
break;
|
|
case PODCASTSROOT:
|
|
setPixmap(0, *s_pixRootItem);
|
|
break;
|
|
case PODCASTITEM:
|
|
case PODCASTCHANNEL:
|
|
setPixmap(0, *s_pixPodcast);
|
|
break;
|
|
case PLAYLIST:
|
|
setPixmap(0, *s_pixPlaylist);
|
|
setDropEnabled(true);
|
|
break;
|
|
case PLAYLISTSROOT:
|
|
setPixmap(0, *s_pixRootItem);
|
|
setDropEnabled( true );
|
|
break;
|
|
case INVISIBLEROOT:
|
|
setPixmap(0, *s_pixInvisible);
|
|
break;
|
|
case STALEROOT:
|
|
case STALE:
|
|
setPixmap(0, *s_pixStale);
|
|
break;
|
|
case ORPHANEDROOT:
|
|
case ORPHANED:
|
|
setPixmap(0, *s_pixOrphaned);
|
|
break;
|
|
case DIRECTORY:
|
|
setExpandable( true );
|
|
setDropEnabled( true );
|
|
setPixmap(0, *s_pixDirectory);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
MediaItem::setFailed( bool failed )
|
|
{
|
|
if( failed )
|
|
{
|
|
m_flags &= ~MediaItem::Transferring;
|
|
m_flags |= MediaItem::Failed;
|
|
setPixmap(0, *MediaItem::s_pixTransferFailed);
|
|
}
|
|
else
|
|
{
|
|
m_flags &= ~MediaItem::Failed;
|
|
if( m_type == PODCASTITEM )
|
|
setPixmap(0, *s_pixPodcast);
|
|
else if( m_type == PLAYLIST )
|
|
setPixmap(0, *s_pixPlaylist);
|
|
else
|
|
setPixmap(0, TQPixmap() );
|
|
}
|
|
}
|
|
|
|
MediaItem *
|
|
MediaItem::lastChild() const
|
|
{
|
|
TQListViewItem *last = 0;
|
|
for( TQListViewItem *it = firstChild();
|
|
it;
|
|
it = it->nextSibling() )
|
|
{
|
|
last = it;
|
|
}
|
|
|
|
return dynamic_cast<MediaItem *>(last);
|
|
}
|
|
|
|
bool
|
|
MediaItem::isLeafItem() const
|
|
{
|
|
switch(type())
|
|
{
|
|
case UNKNOWN:
|
|
return false;
|
|
|
|
case INVISIBLE:
|
|
case TRACK:
|
|
case PODCASTITEM:
|
|
case PLAYLISTITEM:
|
|
case STALE:
|
|
case ORPHANED:
|
|
return true;
|
|
|
|
case ARTIST:
|
|
case ALBUM:
|
|
case PODCASTSROOT:
|
|
case PODCASTCHANNEL:
|
|
case PLAYLISTSROOT:
|
|
case PLAYLIST:
|
|
case INVISIBLEROOT:
|
|
case STALEROOT:
|
|
case ORPHANEDROOT:
|
|
case DIRECTORY:
|
|
return false;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
MediaItem *
|
|
MediaItem::findItem( const TQString &key, const MediaItem *after ) const
|
|
{
|
|
MediaItem *it = 0;
|
|
if( after )
|
|
it = dynamic_cast<MediaItem *>( after->nextSibling() );
|
|
else
|
|
it = dynamic_cast<MediaItem *>( firstChild() );
|
|
|
|
for( ; it; it = dynamic_cast<MediaItem *>(it->nextSibling()))
|
|
{
|
|
if(key == it->text(0))
|
|
return it;
|
|
if(key.isEmpty() && it->text(0).isEmpty())
|
|
return it;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
MediaItem::compare( TQListViewItem *i, int col, bool ascending ) const
|
|
{
|
|
MediaItem *item = dynamic_cast<MediaItem *>(i);
|
|
if(item && col==0 && item->m_order != m_order)
|
|
return m_order-item->m_order;
|
|
else if( item && item->type() == MediaItem::ARTIST )
|
|
{
|
|
TQString key1 = key( col, ascending );
|
|
if( key1.startsWith( "the ", false ) )
|
|
key1 = key1.mid( 4 );
|
|
TQString key2 = i->key( col, ascending );
|
|
if( key2.startsWith( "the ", false ) )
|
|
key2 = key2.mid( 4 );
|
|
|
|
return key1.localeAwareCompare( key2 );
|
|
}
|
|
|
|
return KListViewItem::compare(i, col, ascending);
|
|
}
|
|
|
|
class MediaItemTip : public TQToolTip
|
|
{
|
|
public:
|
|
MediaItemTip( TQListView *listview )
|
|
: TQToolTip( listview->viewport() )
|
|
, m_view( listview )
|
|
{}
|
|
virtual ~MediaItemTip() {}
|
|
protected:
|
|
virtual void maybeTip( const TQPoint &p )
|
|
{
|
|
MediaItem *i = dynamic_cast<MediaItem *>(m_view->itemAt( m_view->viewportToContents( p ) ) );
|
|
if( !i )
|
|
return;
|
|
|
|
TQString text;
|
|
switch( i->type() )
|
|
{
|
|
case MediaItem::TRACK:
|
|
{
|
|
const MetaBundle *b = i->bundle();
|
|
if( b )
|
|
{
|
|
if( b->track() )
|
|
text = TQString( "%1 - %2 (%3)" )
|
|
.arg( TQString::number(b->track()), b->title(), b->prettyLength() );
|
|
if( !b->genre().isEmpty() )
|
|
{
|
|
if( !text.isEmpty() )
|
|
text += "<br>";
|
|
text += TQString( "<i>Genre: %1</i>" )
|
|
.arg( b->genre() );
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case MediaItem::PLAYLISTSROOT:
|
|
text = i18n( "Drag items here to create new playlist" );
|
|
break;
|
|
case MediaItem::PLAYLIST:
|
|
text = i18n( "Drag items here to append to this playlist" );
|
|
break;
|
|
case MediaItem::PLAYLISTITEM:
|
|
text = i18n( "Drag items here to insert before this item" );
|
|
break;
|
|
case MediaItem::INVISIBLEROOT:
|
|
case MediaItem::INVISIBLE:
|
|
text = i18n( "Not visible on media device" );
|
|
break;
|
|
case MediaItem::STALEROOT:
|
|
case MediaItem::STALE:
|
|
text = i18n( "In device database, but file is missing" );
|
|
break;
|
|
case MediaItem::ORPHANEDROOT:
|
|
case MediaItem::ORPHANED:
|
|
text = i18n( "File on device, but not in device database" );
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if( !text.isEmpty() && !text.isNull() )
|
|
tip( m_view->itemRect( i ), text );
|
|
}
|
|
|
|
TQListView *m_view;
|
|
};
|
|
|
|
|
|
MediaView::MediaView( TQWidget* parent, MediaDevice *device )
|
|
: KListView( parent )
|
|
, m_parent( parent )
|
|
, m_device( device )
|
|
{
|
|
hide();
|
|
setSelectionMode( TQListView::Extended );
|
|
setItemsMovable( false );
|
|
setShowSortIndicator( true );
|
|
setFullWidth( true );
|
|
setRootIsDecorated( true );
|
|
setDragEnabled( true );
|
|
setDropVisualizer( true ); //the visualizer (a line marker) is drawn when dragging over tracks
|
|
setDropHighlighter( true ); //and the highligther (a focus rect) is drawn when dragging over playlists
|
|
setDropVisualizerWidth( 3 );
|
|
setAcceptDrops( true );
|
|
|
|
header()->hide();
|
|
addColumn( i18n( "Remote Media" ) );
|
|
|
|
KActionCollection* ac = new KActionCollection( this );
|
|
KStdAction::selectAll( TQT_TQOBJECT(this), TQT_SLOT( selectAll() ), ac, "mediabrowser_select_all" );
|
|
|
|
connect( this, TQT_SIGNAL( contextMenuRequested( TQListViewItem*, const TQPoint&, int ) ),
|
|
this, TQT_SLOT( rmbPressed( TQListViewItem*, const TQPoint&, int ) ) );
|
|
|
|
connect( this, TQT_SIGNAL( itemRenamed( TQListViewItem* ) ),
|
|
this, TQT_SLOT( renameItem( TQListViewItem* ) ) );
|
|
|
|
connect( this, TQT_SIGNAL( expanded( TQListViewItem* ) ),
|
|
this, TQT_SLOT( slotExpand( TQListViewItem* ) ) );
|
|
|
|
connect( this, TQT_SIGNAL( returnPressed( TQListViewItem* ) ),
|
|
this, TQT_SLOT( invokeItem( TQListViewItem* ) ) );
|
|
|
|
connect( this, TQT_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint&, int ) ),
|
|
this, TQT_SLOT( invokeItem( TQListViewItem*, const TQPoint &, int ) ) );
|
|
|
|
m_toolTip = new MediaItemTip( this );
|
|
}
|
|
|
|
void
|
|
MediaView::keyPressEvent( TQKeyEvent *e )
|
|
{
|
|
if( e->key() == Key_Delete )
|
|
m_device->deleteFromDevice();
|
|
else
|
|
KListView::keyPressEvent( e );
|
|
}
|
|
|
|
void
|
|
MediaView::invokeItem( TQListViewItem* i, const TQPoint& point, int column ) //SLOT
|
|
{
|
|
if( column == -1 )
|
|
return;
|
|
|
|
TQPoint p = mapFromGlobal( point );
|
|
if ( p.x() > header()->sectionPos( header()->mapToIndex( 0 ) ) + treeStepSize() * ( i->depth() + ( rootIsDecorated() ? 1 : 0) ) + itemMargin()
|
|
|| p.x() < header()->sectionPos( header()->mapToIndex( 0 ) ) )
|
|
invokeItem( i );
|
|
}
|
|
|
|
|
|
void
|
|
MediaView::invokeItem( TQListViewItem *i )
|
|
{
|
|
MediaItem *item = dynamic_cast<MediaItem *>( i );
|
|
if( !item )
|
|
return;
|
|
|
|
KURL::List urls = nodeBuildDragList( item );
|
|
Playlist::instance()->insertMedia( urls, Playlist::DefaultOptions );
|
|
}
|
|
|
|
void
|
|
MediaView::renameItem( TQListViewItem *item )
|
|
{
|
|
m_device->renameItem( item );
|
|
}
|
|
|
|
void
|
|
MediaView::slotExpand( TQListViewItem *item )
|
|
{
|
|
m_device->expandItem( item );
|
|
}
|
|
|
|
|
|
MediaView::~MediaView()
|
|
{
|
|
delete m_toolTip;
|
|
}
|
|
|
|
|
|
TQDragObject *
|
|
MediaView::dragObject()
|
|
{
|
|
KURL::List urls = nodeBuildDragList( 0 );
|
|
KMultipleDrag *md = new KMultipleDrag( viewport() );
|
|
md->addDragObject( KListView::dragObject() );
|
|
KURLDrag* ud = new KURLDrag( urls, viewport() );
|
|
md->addDragObject( ud );
|
|
md->setPixmap( CollectionDB::createDragPixmap( urls ),
|
|
TQPoint( CollectionDB::DRAGPIXMAP_OFFSET_X, CollectionDB::DRAGPIXMAP_OFFSET_Y ) );
|
|
return md;
|
|
}
|
|
|
|
|
|
KURL::List
|
|
MediaView::nodeBuildDragList( MediaItem* item, int flags )
|
|
{
|
|
KURL::List items;
|
|
MediaItem* fi;
|
|
|
|
if ( !item )
|
|
{
|
|
fi = static_cast<MediaItem*>(firstChild());
|
|
}
|
|
else
|
|
fi = item;
|
|
|
|
while ( fi )
|
|
{
|
|
if( fi->isVisible() )
|
|
{
|
|
if ( fi->isSelected() || !(flags & OnlySelected ) )
|
|
{
|
|
if( fi->isLeafItem() || fi->type() == MediaItem::DIRECTORY )
|
|
items += fi->url();
|
|
else
|
|
{
|
|
if(fi->childCount() )
|
|
items += nodeBuildDragList( static_cast<MediaItem*>(fi->firstChild()), None );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( fi->childCount() )
|
|
items += nodeBuildDragList( static_cast<MediaItem*>(fi->firstChild()), OnlySelected );
|
|
}
|
|
}
|
|
fi = static_cast<MediaItem*>(fi->nextSibling());
|
|
}
|
|
return items;
|
|
}
|
|
|
|
int
|
|
MediaView::getSelectedLeaves( MediaItem *parent, TQPtrList<MediaItem> *list, int flags )
|
|
{
|
|
int numFiles = 0;
|
|
if( !list )
|
|
list = new TQPtrList<MediaItem>;
|
|
|
|
MediaItem *it;
|
|
if( !parent )
|
|
it = dynamic_cast<MediaItem *>(firstChild());
|
|
else
|
|
it = dynamic_cast<MediaItem *>(parent->firstChild());
|
|
|
|
for( ; it; it = dynamic_cast<MediaItem*>(it->nextSibling()))
|
|
{
|
|
if( it->isVisible() )
|
|
{
|
|
if( it->childCount() && !( it->type() == MediaItem::DIRECTORY && it->isSelected() ) )
|
|
{
|
|
int f = flags;
|
|
if( it->isSelected() )
|
|
f &= ~OnlySelected;
|
|
numFiles += getSelectedLeaves(it, list, f );
|
|
}
|
|
if( it->isSelected() || !(flags & OnlySelected) )
|
|
{
|
|
if( it->type() == MediaItem::TRACK ||
|
|
it->type() == MediaItem::DIRECTORY ||
|
|
it->type() == MediaItem::PODCASTITEM ||
|
|
it->type() == MediaItem::PLAYLISTITEM||
|
|
it->type() == MediaItem::INVISIBLE ||
|
|
it->type() == MediaItem::ORPHANED )
|
|
{
|
|
if( flags & OnlyPlayed )
|
|
{
|
|
if( it->played() > 0 )
|
|
numFiles++;
|
|
}
|
|
else
|
|
numFiles++;
|
|
}
|
|
if( ( it->isLeafItem() && (!(flags & OnlyPlayed) || it->played()>0) )
|
|
|| it->type() == MediaItem::DIRECTORY )
|
|
list->append( it );
|
|
}
|
|
}
|
|
}
|
|
return numFiles;
|
|
}
|
|
|
|
|
|
bool
|
|
MediaView::acceptDrag( TQDropEvent *e ) const
|
|
{
|
|
if( e->source() == MediaBrowser::queue()->viewport() )
|
|
return false;
|
|
|
|
TQString data;
|
|
TQCString subtype;
|
|
TQTextDrag::decode( e, data, subtype );
|
|
|
|
return e->source() == viewport()
|
|
|| subtype == "amarok-sql"
|
|
|| KURLDrag::canDecode( e );
|
|
}
|
|
|
|
void
|
|
MediaView::contentsDropEvent( TQDropEvent *e )
|
|
{
|
|
cleanDropVisualizer();
|
|
cleanItemHighlighter();
|
|
|
|
if(e->source() == viewport())
|
|
{
|
|
const TQPoint p = contentsToViewport( e->pos() );
|
|
MediaItem *item = dynamic_cast<MediaItem *>(itemAt( p ));
|
|
|
|
if( !item && MediaBrowser::instance()->currentDevice()->m_type != "generic-mediadevice" )
|
|
return;
|
|
|
|
TQPtrList<MediaItem> items;
|
|
|
|
if( !item || item->type() == MediaItem::DIRECTORY ||
|
|
item->type() == MediaItem::TRACK )
|
|
{
|
|
TQPtrList<MediaItem> items;
|
|
getSelectedLeaves( 0, &items );
|
|
m_device->addToDirectory( item, items );
|
|
}
|
|
else if( item->type() == MediaItem::PLAYLIST )
|
|
{
|
|
MediaItem *list = item;
|
|
MediaItem *after = 0;
|
|
for(MediaItem *it = dynamic_cast<MediaItem *>(item->firstChild());
|
|
it;
|
|
it = dynamic_cast<MediaItem *>(it->nextSibling()))
|
|
after = it;
|
|
|
|
getSelectedLeaves( 0, &items );
|
|
m_device->addToPlaylist( list, after, items );
|
|
}
|
|
else if( item->type() == MediaItem::PLAYLISTITEM )
|
|
{
|
|
MediaItem *list = dynamic_cast<MediaItem *>(item->parent());
|
|
MediaItem *after = 0;
|
|
for(MediaItem *it = dynamic_cast<MediaItem*>(item->parent()->firstChild());
|
|
it;
|
|
it = dynamic_cast<MediaItem *>(it->nextSibling()))
|
|
{
|
|
if(it == item)
|
|
break;
|
|
after = it;
|
|
}
|
|
|
|
getSelectedLeaves( 0, &items );
|
|
m_device->addToPlaylist( list, after, items );
|
|
}
|
|
else if( item->type() == MediaItem::PLAYLISTSROOT )
|
|
{
|
|
TQPtrList<MediaItem> items;
|
|
getSelectedLeaves( 0, &items );
|
|
TQString base( i18n("New Playlist") );
|
|
TQString name = base;
|
|
int i=1;
|
|
while( item->findItem(name) )
|
|
{
|
|
TQString num;
|
|
num.setNum(i);
|
|
name = base + ' ' + num;
|
|
i++;
|
|
}
|
|
MediaItem *pl = m_device->newPlaylist(name, item, items);
|
|
ensureItemVisible(pl);
|
|
rename(pl, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
TQString data;
|
|
TQCString subtype;
|
|
TQTextDrag::decode( e, data, subtype );
|
|
KURL::List list;
|
|
|
|
if( subtype == "amarok-sql" )
|
|
{
|
|
TQString playlist = data.section( "\n", 0, 0 );
|
|
TQString query = data.section( "\n", 1 );
|
|
TQStringList values = CollectionDB::instance()->query( query );
|
|
list = CollectionDB::instance()->URLsFromSqlDrag( values );
|
|
MediaBrowser::queue()->addURLs( list, playlist );
|
|
}
|
|
else if ( KURLDrag::decode( e, list ) )
|
|
{
|
|
MediaBrowser::queue()->addURLs( list );
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
MediaView::viewportPaintEvent( TQPaintEvent *e )
|
|
{
|
|
KListView::viewportPaintEvent( e );
|
|
|
|
// Superimpose bubble help:
|
|
|
|
if ( !MediaBrowser::instance()->currentDevice() || !MediaBrowser::instance()->currentDevice()->isConnected() )
|
|
{
|
|
TQPainter p( viewport() );
|
|
|
|
TQSimpleRichText t( i18n(
|
|
"<div align=center>"
|
|
"<h3>Media Device Browser</h3>"
|
|
"Configure your media device and then "
|
|
"click the Connect button to access your media device. "
|
|
"Drag and drop files to enqueue them for transfer."
|
|
"</div>" ), TQApplication::font() );
|
|
|
|
t.setWidth( width() - 50 );
|
|
|
|
const uint w = t.width() + 20;
|
|
const uint h = t.height() + 20;
|
|
|
|
p.setBrush( colorGroup().background() );
|
|
p.drawRoundRect( 15, 15, w, h, (8*200)/w, (8*200)/h );
|
|
t.draw( &p, 20, 20, TQRect(), colorGroup() );
|
|
}
|
|
MediaBrowser::instance()->updateButtons();
|
|
}
|
|
|
|
void
|
|
MediaView::rmbPressed( TQListViewItem *item, const TQPoint &p, int i )
|
|
{
|
|
if( m_device->isConnected() )
|
|
m_device->rmbPressed( item, p, i );
|
|
}
|
|
|
|
MediaItem *
|
|
MediaView::newDirectory( MediaItem *parent )
|
|
{
|
|
bool ok;
|
|
const TQString name = KInputDialog::getText(i18n("Add Directory"), i18n("Directory Name:"), TQString(), &ok, this);
|
|
|
|
if( ok && !name.isEmpty() )
|
|
{
|
|
m_device->newDirectory( name, parent );
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::mediumAdded( const Medium *medium, TQString /*name*/, bool /*constructing*/ )
|
|
{
|
|
debug() << "mediumAdded: " << (medium? medium->properties():"null") << endl;
|
|
if( medium )
|
|
{
|
|
TQString handler = Amarok::config( "MediaBrowser" )->readEntry( medium->id() );
|
|
if( handler.isEmpty() )
|
|
{
|
|
//Some people complained about the dialog, boohoo
|
|
//Just disable it for now I guess
|
|
/*if( !constructing && medium->isAutodetected() )
|
|
{
|
|
MediumPluginManagerDialog *mpm = new MediumPluginManagerDialog();
|
|
mpm->exec();
|
|
}*/
|
|
}
|
|
//debug() << "id=" << medium->id() << ", handler=" << handler << endl;
|
|
MediaDevice *device = loadDevicePlugin( handler );
|
|
if( device )
|
|
{
|
|
device->m_medium = *medium;
|
|
addDevice( device );
|
|
if( m_currentDevice == m_devices.begin() || m_currentDevice == m_devices.end() )
|
|
activateDevice( m_devices.count()-1, false );
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
MediaBrowser::pluginSelected( const Medium *medium, const TQString plugin )
|
|
{
|
|
DEBUG_BLOCK
|
|
if( !plugin.isEmpty() )
|
|
{
|
|
debug() << "Medium id is " << medium->id() << " and plugin selected is: " << plugin << endl;
|
|
Amarok::config( "MediaBrowser" )->writeEntry( medium->id(), plugin );
|
|
|
|
bool success = true;
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( (*it)->uniqueId() == medium->id() )
|
|
{
|
|
debug() << "removing " << medium->deviceNode() << endl;
|
|
if( (*it)->isConnected() )
|
|
{
|
|
if( (*it)->disconnectDevice( false ) )
|
|
removeDevice( *it );
|
|
else
|
|
success = false;
|
|
}
|
|
else
|
|
removeDevice( *it );
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( success )
|
|
{
|
|
mediumAdded( medium, "doesntmatter", false );
|
|
}
|
|
else
|
|
{
|
|
debug() << "Cannot change plugin while operation is in progress" << endl;
|
|
Amarok::StatusBar::instance()->longMessage(
|
|
i18n( "Cannot change plugin while operation is in progress" ),
|
|
KDE::StatusBar::Warning );
|
|
}
|
|
}
|
|
else
|
|
debug() << "Medium id is " << medium->id() << " and you opted not to use a plugin" << endl;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::showPluginManager()
|
|
{
|
|
MediumPluginManagerDialog* mpm = new MediumPluginManagerDialog();
|
|
mpm->exec();
|
|
delete mpm;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::mediumChanged( const Medium *medium, TQString /*name*/ )
|
|
{
|
|
if( medium )
|
|
{
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( (*it)->uniqueId() == medium->id() )
|
|
{
|
|
(*it)->m_medium = const_cast<Medium *>(medium);
|
|
if( !(*it)->isConnected() && medium->isMounted() )
|
|
(*it)->connectDevice();
|
|
#if 0
|
|
else if( (*it)->isConnected() && !medium->isMounted() )
|
|
{
|
|
Amarok::StatusBar::instance()->longMessage(
|
|
i18n( "The device %1 was unmounted before it was synchronized. "
|
|
"In order to avoid data loss, press the \"Disconnect\" button "
|
|
"before unmounting the device." ).arg( name ),
|
|
KDE::StatusBar::Warning );
|
|
//(*it)->disconnectDevice();
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
MediaBrowser::mediumRemoved( const Medium *medium, TQString name )
|
|
{
|
|
if( medium )
|
|
{
|
|
for( TQValueList<MediaDevice *>::iterator it = m_devices.begin();
|
|
it != m_devices.end();
|
|
it++ )
|
|
{
|
|
if( (*it)->uniqueId() == medium->id() )
|
|
{
|
|
if( (*it)->isConnected() )
|
|
{
|
|
if( (*it)->disconnectDevice() )
|
|
removeDevice( *it );
|
|
Amarok::StatusBar::instance()->longMessage(
|
|
i18n( "The device %1 was removed before it was disconnected. "
|
|
"In order to avoid possible data loss, press the \"Disconnect\" "
|
|
"button before disconnecting the device." ).arg( name ),
|
|
KDE::StatusBar::Warning );
|
|
}
|
|
else
|
|
removeDevice( *it );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
MediaDevice *
|
|
MediaBrowser::loadDevicePlugin( const TQString &deviceType )
|
|
{
|
|
DEBUG_BLOCK
|
|
|
|
if( deviceType == "ignore" )
|
|
return 0;
|
|
|
|
TQString query = "[X-KDE-Amarok-plugintype] == 'mediadevice' and [X-KDE-Amarok-name] == '%1'";
|
|
Amarok::Plugin *plugin = PluginManager::createFromQuery( query.arg( deviceType ) );
|
|
|
|
if( plugin )
|
|
{
|
|
debug() << "Returning plugin!" << endl;
|
|
MediaDevice *device = static_cast<MediaDevice *>( plugin );
|
|
device->init( this );
|
|
device->m_type = deviceType;
|
|
return device;
|
|
}
|
|
|
|
debug() << "no plugin for " << deviceType << endl;
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::unloadDevicePlugin( MediaDevice *device )
|
|
{
|
|
DEBUG_BLOCK
|
|
|
|
if( !device )
|
|
return;
|
|
|
|
disconnect( device ); // disconnect all signals
|
|
|
|
if( dynamic_cast<DummyMediaDevice *>(device) )
|
|
{
|
|
delete device;
|
|
}
|
|
else
|
|
{
|
|
PluginManager::unload( device );
|
|
}
|
|
}
|
|
|
|
bool
|
|
MediaBrowser::config()
|
|
{
|
|
if( m_deviceCombo->currentText() == "No Device Selected" )
|
|
{
|
|
showPluginManager();
|
|
return true;
|
|
}
|
|
|
|
DeviceConfigureDialog* dcd = new DeviceConfigureDialog( currentDevice()->m_medium );
|
|
dcd->exec();
|
|
bool successful = dcd->successful();
|
|
delete dcd;
|
|
return successful;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::configSelectPlugin( int index )
|
|
{
|
|
Q_UNUSED( index );
|
|
|
|
if( m_currentDevice == m_devices.begin() )
|
|
{
|
|
AmarokConfig::setDeviceType( m_pluginName[m_configPluginCombo->currentText()] );
|
|
}
|
|
else if( currentDevice() )
|
|
{
|
|
KConfig *config = Amarok::config( "MediaBrowser" );
|
|
config->writeEntry( currentDevice()->uniqueId(), m_pluginName[m_configPluginCombo->currentText()] );
|
|
}
|
|
|
|
if( !currentDevice() )
|
|
activateDevice( 0, false );
|
|
|
|
if( !currentDevice() )
|
|
return;
|
|
|
|
if( m_pluginName[m_configPluginCombo->currentText()] != currentDevice()->deviceType() )
|
|
{
|
|
MediaDevice *dev = currentDevice();
|
|
dev->removeConfigElements( m_configBox );
|
|
if( dev->isConnected() )
|
|
{
|
|
dev->disconnectDevice( false );
|
|
}
|
|
unloadDevicePlugin( dev );
|
|
*m_currentDevice = loadDevicePlugin( AmarokConfig::deviceType() );
|
|
if( !*m_currentDevice )
|
|
{
|
|
*m_currentDevice = new DummyMediaDevice();
|
|
if( AmarokConfig::deviceType() != "dummy-mediadevice" )
|
|
{
|
|
TQString msg = i18n( "The requested media device could not be loaded" );
|
|
Amarok::StatusBar::instance()->shortMessage( msg );
|
|
}
|
|
}
|
|
dev = currentDevice();
|
|
dev->init( this );
|
|
dev->loadConfig();
|
|
|
|
m_configBox->hide();
|
|
dev->addConfigElements( m_configBox );
|
|
m_configBox->show();
|
|
|
|
dev->view()->show();
|
|
|
|
if( dev->autoConnect() )
|
|
{
|
|
dev->connectDevice( true );
|
|
updateButtons();
|
|
}
|
|
|
|
updateDevices();
|
|
}
|
|
}
|
|
|
|
void
|
|
MediaBrowser::updateButtons()
|
|
{
|
|
if( !m_toolbar->getButton(CONNECT) ||
|
|
!m_toolbar->getButton(DISCONNECT) ||
|
|
!m_toolbar->getButton(TRANSFER) ) //TODO add CUSTOM
|
|
return;
|
|
|
|
if( currentDevice() )
|
|
{
|
|
if( currentDevice()->m_transfer )
|
|
m_toolbar->showItem( TRANSFER );
|
|
else
|
|
m_toolbar->hideItem( TRANSFER );
|
|
|
|
if( currentDevice()->m_customButton )
|
|
m_toolbar->showItem( CUSTOM );
|
|
else
|
|
m_toolbar->hideItem( CUSTOM );
|
|
|
|
if( currentDevice()->m_configure )
|
|
m_toolbar->showItem( CONFIGURE );
|
|
else
|
|
m_toolbar->hideItem( CONFIGURE );
|
|
|
|
m_toolbar->getButton(CONNECT)->setEnabled( !currentDevice()->isConnected() );
|
|
m_toolbar->getButton(DISCONNECT)->setEnabled( currentDevice()->isConnected() );
|
|
m_toolbar->getButton(TRANSFER)->setEnabled( currentDevice()->isConnected() && m_queue->childCount() > 0 );
|
|
m_toolbar->getButton( CUSTOM )->setEnabled( true );
|
|
}
|
|
else
|
|
{
|
|
m_toolbar->getButton( CONNECT )->setEnabled( false );
|
|
m_toolbar->getButton( DISCONNECT )->setEnabled( false );
|
|
m_toolbar->getButton( TRANSFER )->setEnabled( false );
|
|
m_toolbar->getButton( CUSTOM )->setEnabled( false );
|
|
}
|
|
}
|
|
|
|
void
|
|
MediaBrowser::updateStats()
|
|
{
|
|
if( !m_stats )
|
|
return;
|
|
|
|
KIO::filesize_t queued = m_queue->totalSize();
|
|
|
|
TQString text = i18n( "1 track in queue", "%n tracks in queue", m_queue->childCount() );
|
|
if(m_queue->childCount() > 0)
|
|
{
|
|
text += i18n(" (%1)").arg( KIO::convertSize( queued ) );
|
|
}
|
|
|
|
KIO::filesize_t total, avail;
|
|
if( currentDevice() && currentDevice()->getCapacity(&total, &avail) )
|
|
{
|
|
text += i18n( " - %1 of %2 available" ).arg( KIO::convertSize( avail ) ).arg( KIO::convertSize( total ) );
|
|
|
|
m_stats->m_used = total-avail;
|
|
m_stats->m_total = total;
|
|
m_stats->m_scheduled = queued;
|
|
}
|
|
else
|
|
{
|
|
m_stats->m_used = 0;
|
|
m_stats->m_total = 0;
|
|
m_stats->m_scheduled = queued;
|
|
}
|
|
|
|
m_stats->setText(text);
|
|
TQToolTip::add( m_stats, text );
|
|
}
|
|
|
|
|
|
bool
|
|
MediaView::setFilter( const TQString &filter, MediaItem *parent )
|
|
{
|
|
bool advanced = ExpressionParser::isAdvancedExpression( filter );
|
|
TQValueList<int> defaultColumns;
|
|
defaultColumns << MetaBundle::Album;
|
|
defaultColumns << MetaBundle::Title;
|
|
defaultColumns << MetaBundle::Artist;
|
|
|
|
bool root = false;
|
|
MediaItem *it;
|
|
if( !parent )
|
|
{
|
|
root = true;
|
|
it = dynamic_cast<MediaItem *>(firstChild());
|
|
}
|
|
else
|
|
{
|
|
it = dynamic_cast<MediaItem *>(parent->firstChild());
|
|
}
|
|
|
|
bool childrenVisible = false;
|
|
for( ; it; it = dynamic_cast<MediaItem *>(it->nextSibling()))
|
|
{
|
|
bool visible = true;
|
|
if(it->isLeafItem())
|
|
{
|
|
if( advanced )
|
|
{
|
|
ParsedExpression parsed = ExpressionParser::parse( filter );
|
|
visible = it->bundle() && it->bundle()->matchesParsedExpression( parsed, defaultColumns );
|
|
}
|
|
else
|
|
{
|
|
visible = it->bundle() && it->bundle()->matchesSimpleExpression( filter, defaultColumns );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
visible = setFilter(filter, it);
|
|
if(it->type()==MediaItem::PLAYLISTSROOT || it->type()==MediaItem::PLAYLIST)
|
|
{
|
|
visible = true;
|
|
}
|
|
else if(it->type()==MediaItem::DIRECTORY)
|
|
{
|
|
bool match = true;
|
|
TQStringList list = TQStringList::split( " ", filter );
|
|
for( TQStringList::iterator i = list.begin();
|
|
i != list.end();
|
|
++i )
|
|
{
|
|
if( !(*it).text(0).contains( *i ) )
|
|
{
|
|
match = false;
|
|
break;
|
|
}
|
|
}
|
|
if( match )
|
|
visible = true;
|
|
}
|
|
}
|
|
if( filter.isEmpty() )
|
|
visible = true;
|
|
it->setVisible( visible );
|
|
if(visible)
|
|
childrenVisible = true;
|
|
}
|
|
|
|
if( root && m_device )
|
|
m_device->updateRootItems();
|
|
|
|
return childrenVisible;
|
|
}
|
|
|
|
MediaDevice::MediaDevice()
|
|
: Amarok::Plugin()
|
|
, m_hasMountPoint( true )
|
|
, m_autoDeletePodcasts( false )
|
|
, m_syncStats( false )
|
|
, m_transcode( false )
|
|
, m_transcodeAlways( false )
|
|
, m_transcodeRemove( false )
|
|
, sysProc ( 0 )
|
|
, m_parent( 0 )
|
|
, m_view( 0 )
|
|
, m_wait( false )
|
|
, m_requireMount( false )
|
|
, m_canceled( false )
|
|
, m_transferring( false )
|
|
, m_deleting( false )
|
|
, m_deferredDisconnect( false )
|
|
, m_scheduledDisconnect( false )
|
|
, m_transfer( true )
|
|
, m_configure( true )
|
|
, m_customButton( false )
|
|
, m_playlistItem( 0 )
|
|
, m_podcastItem( 0 )
|
|
, m_invisibleItem( 0 )
|
|
, m_staleItem( 0 )
|
|
, m_orphanedItem( 0 )
|
|
{
|
|
sysProc = new KShellProcess(); Q_CHECK_PTR(sysProc);
|
|
}
|
|
|
|
void MediaDevice::init( MediaBrowser* parent )
|
|
{
|
|
m_parent = parent;
|
|
if( !m_view )
|
|
m_view = new MediaView( m_parent->m_views, this );
|
|
m_view->hide();
|
|
}
|
|
|
|
MediaDevice::~MediaDevice()
|
|
{
|
|
delete m_view;
|
|
delete sysProc;
|
|
}
|
|
|
|
bool
|
|
MediaDevice::isSpecialItem( MediaItem *item )
|
|
{
|
|
return (item == m_playlistItem) ||
|
|
(item == m_podcastItem) ||
|
|
(item == m_invisibleItem) ||
|
|
(item == m_staleItem) ||
|
|
(item == m_orphanedItem);
|
|
}
|
|
|
|
void
|
|
MediaDevice::loadConfig()
|
|
{
|
|
m_transcode = configBool( "Transcode" );
|
|
m_transcodeAlways = configBool( "TranscodeAlways" );
|
|
m_transcodeRemove = configBool( "TranscodeRemove" );
|
|
m_preconnectcmd = configString( "PreConnectCommand" );
|
|
if( m_preconnectcmd.isEmpty() )
|
|
m_preconnectcmd = configString( "MountCommand" );
|
|
m_postdisconnectcmd = configString( "PostDisconnectCommand" );
|
|
if( m_postdisconnectcmd.isEmpty() )
|
|
m_postdisconnectcmd = configString( "UmountCommand" );
|
|
if( m_requireMount && m_postdisconnectcmd.isEmpty() )
|
|
m_postdisconnectcmd = "kdeeject -q %d";
|
|
}
|
|
|
|
TQString
|
|
MediaDevice::configString( const TQString &name, const TQString &defValue )
|
|
{
|
|
TQString configName = "MediaDevice";
|
|
if( !uniqueId().isEmpty() )
|
|
configName += '_' + uniqueId();
|
|
KConfig *config = Amarok::config( configName );
|
|
return config->readEntry( name, defValue );
|
|
}
|
|
|
|
void
|
|
MediaDevice::setConfigString( const TQString &name, const TQString &value )
|
|
{
|
|
TQString configName = "MediaDevice";
|
|
if( !uniqueId().isEmpty() )
|
|
configName += '_' + uniqueId();
|
|
KConfig *config = Amarok::config( configName );
|
|
config->writeEntry( name, value );
|
|
}
|
|
|
|
bool
|
|
MediaDevice::configBool( const TQString &name, bool defValue )
|
|
{
|
|
TQString configName = "MediaDevice";
|
|
if( !uniqueId().isEmpty() )
|
|
configName += '_' + uniqueId();
|
|
KConfig *config = Amarok::config( configName );
|
|
return config->readBoolEntry( name, defValue );
|
|
}
|
|
|
|
void
|
|
MediaDevice::setConfigBool( const TQString &name, bool value )
|
|
{
|
|
TQString configName = "MediaDevice";
|
|
if( !uniqueId().isEmpty() )
|
|
configName += '_' + uniqueId();
|
|
KConfig *config = Amarok::config( configName );
|
|
config->writeEntry( name, value );
|
|
}
|
|
|
|
MediaView *
|
|
MediaDevice::view()
|
|
{
|
|
return m_view;
|
|
}
|
|
|
|
void
|
|
MediaDevice::hideProgress()
|
|
{
|
|
m_parent->m_progressBox->hide();
|
|
}
|
|
|
|
void
|
|
MediaDevice::updateRootItems()
|
|
{
|
|
if(m_podcastItem)
|
|
m_podcastItem->setVisible(m_podcastItem->childCount() > 0);
|
|
if(m_invisibleItem)
|
|
m_invisibleItem->setVisible(m_invisibleItem->childCount() > 0);
|
|
if(m_staleItem)
|
|
m_staleItem->setVisible(m_staleItem->childCount() > 0);
|
|
if(m_orphanedItem)
|
|
m_orphanedItem->setVisible(m_orphanedItem->childCount() > 0);
|
|
}
|
|
|
|
void
|
|
MediaQueue::syncPlaylist( const TQString &name, const TQString &query, bool loading )
|
|
{
|
|
MediaItem* item = new MediaItem( this, lastItem() );
|
|
item->setType( MediaItem::PLAYLIST );
|
|
item->setExpandable( false );
|
|
item->setData( query );
|
|
item->m_playlistName = name;
|
|
item->setText( 0, name );
|
|
item->m_flags |= MediaItem::SmartPlaylist;
|
|
m_parent->m_progress->setTotalSteps( m_parent->m_progress->totalSteps() + 1 );
|
|
itemCountChanged();
|
|
if( !loading )
|
|
URLsAdded();
|
|
}
|
|
|
|
void
|
|
MediaQueue::syncPlaylist( const TQString &name, const KURL &url, bool loading )
|
|
{
|
|
MediaItem* item = new MediaItem( this, lastItem() );
|
|
item->setType( MediaItem::PLAYLIST );
|
|
item->setExpandable( false );
|
|
item->setData( url.url() );
|
|
item->m_playlistName = name;
|
|
item->setText( 0, name );
|
|
m_parent->m_progress->setTotalSteps( m_parent->m_progress->totalSteps() + 1 );
|
|
itemCountChanged();
|
|
if( !loading )
|
|
URLsAdded();
|
|
}
|
|
|
|
BundleList
|
|
MediaDevice::bundlesToSync( const TQString &name, const KURL &url )
|
|
{
|
|
BundleList bundles;
|
|
if( !PlaylistFile::isPlaylistFile( url ) )
|
|
{
|
|
Amarok::StatusBar::instance()->longMessage( i18n( "Not a playlist file: %1" ).arg( url.path() ),
|
|
KDE::StatusBar::Sorry );
|
|
return bundles;
|
|
}
|
|
|
|
PlaylistFile playlist( url.path() );
|
|
if( playlist.isError() )
|
|
{
|
|
Amarok::StatusBar::instance()->longMessage( i18n( "Failed to load playlist: %1" ).arg( url.path() ),
|
|
KDE::StatusBar::Sorry );
|
|
return bundles;
|
|
}
|
|
|
|
for( BundleList::iterator it = playlist.bundles().begin();
|
|
it != playlist.bundles().end();
|
|
++it )
|
|
{
|
|
bundles += MetaBundle( (*it).url() );
|
|
}
|
|
preparePlaylistForSync( name, bundles );
|
|
return bundles;
|
|
}
|
|
|
|
BundleList
|
|
MediaDevice::bundlesToSync( const TQString &name, const TQString &query )
|
|
{
|
|
const TQStringList values = CollectionDB::instance()->query( query );
|
|
|
|
BundleList bundles;
|
|
for( for_iterators( TQStringList, values ); it != end; ++it )
|
|
bundles += CollectionDB::instance()->bundleFromQuery( &it );
|
|
preparePlaylistForSync( name, bundles );
|
|
return bundles;
|
|
}
|
|
|
|
void
|
|
MediaDevice::preparePlaylistForSync( const TQString &name, const BundleList &bundles )
|
|
{
|
|
if( ! m_playlistItem ) // might be syncing a new playlist from the playlist browser
|
|
return;
|
|
MediaItem *pl = m_playlistItem->findItem( name );
|
|
if( pl )
|
|
{
|
|
MediaItem *next = 0;
|
|
for( MediaItem *it = static_cast<MediaItem *>(pl->firstChild());
|
|
it;
|
|
it = next )
|
|
{
|
|
next = static_cast<MediaItem *>(it->nextSibling());
|
|
const MetaBundle *bundle = (*it).bundle();
|
|
if( !bundle )
|
|
continue;
|
|
if( isOnOtherPlaylist( name, *bundle ) )
|
|
continue;
|
|
if( isInBundleList( bundles, *bundle ) )
|
|
continue;
|
|
deleteItemFromDevice( it );
|
|
}
|
|
deleteItemFromDevice( pl, None );
|
|
}
|
|
purgeEmptyItems();
|
|
}
|
|
|
|
bool
|
|
MediaDevice::bundleMatch( const MetaBundle &b1, const MetaBundle &b2 )
|
|
{
|
|
if( b1.track() != b2.track() )
|
|
return false;
|
|
if( b1.title() != b2.title() )
|
|
return false;
|
|
if( b1.album() != b2.album() )
|
|
return false;
|
|
if( b1.artist() != b2.artist() )
|
|
return false;
|
|
#if 0
|
|
if( b1.discNumber() != b2.discNumber() )
|
|
return false;
|
|
if( b1.composer() != b2.composer() )
|
|
return false;
|
|
#endif
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
MediaDevice::isInBundleList( const BundleList &bundles, const MetaBundle &b )
|
|
{
|
|
for( BundleList::const_iterator it = bundles.begin();
|
|
it != bundles.end();
|
|
++it )
|
|
{
|
|
if( bundleMatch( b, *it ) )
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
MediaDevice::isOnOtherPlaylist( const TQString &playlistToAvoid, const MetaBundle &bundle )
|
|
{
|
|
for( MediaItem *it = static_cast<MediaItem *>(m_playlistItem->firstChild());
|
|
it;
|
|
it = static_cast<MediaItem *>(it->nextSibling()) )
|
|
{
|
|
if( it->text( 0 ) == playlistToAvoid )
|
|
continue;
|
|
if( isOnPlaylist( *it, bundle ) )
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
MediaDevice::isOnPlaylist( const MediaItem &playlist, const MetaBundle &bundle )
|
|
{
|
|
for( MediaItem *it = static_cast<MediaItem *>(playlist.firstChild());
|
|
it;
|
|
it = static_cast<MediaItem *>(it->nextSibling()) )
|
|
{
|
|
const MetaBundle *b = (*it).bundle();
|
|
if( !b )
|
|
continue;
|
|
if( bundleMatch( *b, bundle ) )
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void
|
|
MediaQueue::addURL( const KURL& url2, MetaBundle *bundle, const TQString &playlistName )
|
|
{
|
|
KURL url = Amarok::mostLocalURL( url2 );
|
|
|
|
if( PlaylistFile::isPlaylistFile( url ) )
|
|
{
|
|
TQString name = TQString(url.path().section( "/", -1 ).section( ".", 0, -2 )).replace( "_", " " );
|
|
PlaylistFile playlist( url.path() );
|
|
|
|
if( playlist.isError() )
|
|
{
|
|
Amarok::StatusBar::instance()->longMessage( i18n( "Failed to load playlist: %1" ).arg( url.path() ),
|
|
KDE::StatusBar::Sorry );
|
|
return;
|
|
}
|
|
|
|
for( BundleList::iterator it = playlist.bundles().begin();
|
|
it != playlist.bundles().end();
|
|
++it )
|
|
{
|
|
addURL( (*it).url(), 0, name );
|
|
}
|
|
return;
|
|
}
|
|
else if( ContextBrowser::hasContextProtocol( url ) )
|
|
{
|
|
KURL::List urls = ContextBrowser::expandURL( url );
|
|
|
|
for( KURL::List::iterator it = urls.begin();
|
|
it != urls.end();
|
|
++it )
|
|
{
|
|
addURL( *it );
|
|
}
|
|
return;
|
|
}
|
|
else if( url.protocol() == "file" && TQFileInfo( url.path() ).isDir() )
|
|
{
|
|
KURL::List urls = Amarok::recursiveUrlExpand( url );
|
|
foreachType( KURL::List, urls )
|
|
addURL( *it );
|
|
return;
|
|
}
|
|
|
|
if( playlistName.isNull() )
|
|
{
|
|
for( MediaItem *it = static_cast<MediaItem *>(firstChild());
|
|
it;
|
|
it = static_cast<MediaItem *>(it->nextSibling()) )
|
|
{
|
|
if( it->url() == url )
|
|
{
|
|
Amarok::StatusBar::instance()->shortMessage(
|
|
i18n( "Track already queued for transfer: %1" ).arg( url.url() ) );
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!bundle)
|
|
bundle = new MetaBundle( url );
|
|
|
|
MediaItem* item = new MediaItem( this, lastItem() );
|
|
item->setExpandable( false );
|
|
item->setDropEnabled( true );
|
|
item->setBundle( bundle );
|
|
if(bundle->podcastBundle() )
|
|
{
|
|
item->setType( MediaItem::PODCASTITEM );
|
|
}
|
|
item->m_playlistName = playlistName;
|
|
|
|
TQString text = item->bundle()->prettyTitle();
|
|
if( text.isEmpty() || (!item->bundle()->isValidMedia() && !item->bundle()->podcastBundle()) )
|
|
text = item->bundle()->url().prettyURL();
|
|
if( !item->m_playlistName.isNull() )
|
|
{
|
|
text += " (" + item->m_playlistName + ')';
|
|
}
|
|
item->setText( 0, text);
|
|
|
|
m_parent->updateButtons();
|
|
m_parent->m_progress->setTotalSteps( m_parent->m_progress->totalSteps() + 1 );
|
|
addItemToSize( item );
|
|
itemCountChanged();
|
|
}
|
|
|
|
void
|
|
MediaQueue::addURL( const KURL &url, MediaItem *item )
|
|
{
|
|
DEBUG_BLOCK
|
|
MediaItem *newitem = new MediaItem( this, lastItem() );
|
|
newitem->setExpandable( false );
|
|
newitem->setDropEnabled( true );
|
|
MetaBundle *bundle = new MetaBundle( *item->bundle() );
|
|
KURL filepath(url);
|
|
filepath.addPath( bundle->filename() );
|
|
bundle->setUrl( filepath );
|
|
newitem->m_device = item->m_device;
|
|
if(bundle->podcastBundle() )
|
|
{
|
|
item->setType( MediaItem::PODCASTITEM );
|
|
}
|
|
TQString text = item->bundle()->prettyTitle();
|
|
if( text.isEmpty() || (!item->bundle()->isValidMedia() && !item->bundle()->podcastBundle()) )
|
|
text = item->bundle()->url().prettyURL();
|
|
if( item->m_playlistName != TQString() )
|
|
{
|
|
text += " (" + item->m_playlistName + ')';
|
|
}
|
|
newitem->setText( 0, text);
|
|
newitem->setBundle( bundle );
|
|
m_parent->updateButtons();
|
|
m_parent->m_progress->setTotalSteps( m_parent->m_progress->totalSteps() + 1 );
|
|
addItemToSize( item );
|
|
itemCountChanged();
|
|
|
|
}
|
|
|
|
void
|
|
MediaQueue::addURLs( const KURL::List urls, const TQString &playlistName )
|
|
{
|
|
KURL::List::ConstIterator it = urls.begin();
|
|
for ( ; it != urls.end(); ++it )
|
|
addURL( *it, 0, playlistName );
|
|
|
|
URLsAdded();
|
|
}
|
|
|
|
void
|
|
MediaQueue::URLsAdded()
|
|
{
|
|
m_parent->updateStats();
|
|
m_parent->updateButtons();
|
|
if( m_parent->currentDevice()
|
|
&& m_parent->currentDevice()->isConnected()
|
|
&& m_parent->currentDevice()->asynchronousTransfer()
|
|
&& !m_parent->currentDevice()->isTransferring() )
|
|
m_parent->currentDevice()->transferFiles();
|
|
|
|
save( Amarok::saveLocation() + "transferlist.xml" );
|
|
}
|
|
|
|
void
|
|
MediaDevice::copyTrackFromDevice( MediaItem *item )
|
|
{
|
|
debug() << "copyTrackFromDevice: not copying " << item->url() << ": not implemented" << endl;
|
|
}
|
|
|
|
TQDragObject *
|
|
MediaQueue::dragObject()
|
|
{
|
|
KURL::List urls;
|
|
for( TQListViewItem *it = firstChild(); it; it = it->nextSibling() )
|
|
{
|
|
if( it->isVisible() && it->isSelected() && dynamic_cast<MediaItem *>(it) )
|
|
urls += static_cast<MediaItem *>(it)->url();
|
|
}
|
|
|
|
KMultipleDrag *md = new KMultipleDrag( viewport() );
|
|
TQDragObject *d = KListView::dragObject();
|
|
KURLDrag* urldrag = new KURLDrag( urls, viewport() );
|
|
md->addDragObject( d );
|
|
md->addDragObject( urldrag );
|
|
md->setPixmap( CollectionDB::createDragPixmap( urls ),
|
|
TQPoint( CollectionDB::DRAGPIXMAP_OFFSET_X, CollectionDB::DRAGPIXMAP_OFFSET_Y ) );
|
|
return md;
|
|
}
|
|
|
|
TQString
|
|
MediaDevice::replaceVariables( const TQString &cmd )
|
|
{
|
|
TQString result = cmd;
|
|
result.replace( "%d", deviceNode() );
|
|
result.replace( "%m", mountPoint() );
|
|
return result;
|
|
}
|
|
|
|
int MediaDevice::runPreConnectCommand()
|
|
{
|
|
if( m_preconnectcmd.isEmpty() )
|
|
return 0;
|
|
|
|
TQString cmd = replaceVariables( m_preconnectcmd );
|
|
|
|
debug() << "running pre-connect command: [" << cmd << "]" << endl;
|
|
int e=sysCall(cmd);
|
|
debug() << "pre-connect: e=" << e << endl;
|
|
return e;
|
|
}
|
|
|
|
int MediaDevice::runPostDisconnectCommand()
|
|
{
|
|
if( m_postdisconnectcmd.isEmpty() )
|
|
return 0;
|
|
|
|
TQString cmd = replaceVariables( m_postdisconnectcmd );
|
|
debug() << "running post-disconnect command: [" << cmd << "]" << endl;
|
|
int e=sysCall(cmd);
|
|
debug() << "post-disconnect: e=" << e << endl;
|
|
|
|
return e;
|
|
}
|
|
|
|
int MediaDevice::sysCall( const TQString &command )
|
|
{
|
|
if ( sysProc->isRunning() ) return -1;
|
|
|
|
sysProc->clearArguments();
|
|
(*sysProc) << command;
|
|
if (!sysProc->start( KProcess::Block, KProcess::AllOutput ))
|
|
kdFatal() << i18n("could not execute %1").arg(command.local8Bit().data()) << endl;
|
|
|
|
return (sysProc->exitStatus());
|
|
}
|
|
|
|
void
|
|
MediaDevice::abortTransfer()
|
|
{
|
|
setCanceled( true );
|
|
cancelTransfer();
|
|
}
|
|
|
|
bool
|
|
MediaDevice::kioCopyTrack( const KURL &src, const KURL &dst )
|
|
{
|
|
m_wait = true;
|
|
|
|
KIO::FileCopyJob *job = KIO::file_copy( src, dst,
|
|
-1 /* permissions */,
|
|
false /* overwrite */,
|
|
false /* resume */,
|
|
false /* show progress */ );
|
|
connect( job, TQT_SIGNAL( result( KIO::Job * ) ),
|
|
this, TQT_SLOT( fileTransferred( KIO::Job * ) ) );
|
|
|
|
bool tryToRemove = false;
|
|
while ( m_wait )
|
|
{
|
|
if( isCanceled() )
|
|
{
|
|
job->kill( false /* still emit result */ );
|
|
tryToRemove = true;
|
|
m_wait = false;
|
|
}
|
|
else
|
|
{
|
|
usleep(10000);
|
|
kapp->processEvents( 100 );
|
|
}
|
|
}
|
|
|
|
if( !tryToRemove )
|
|
{
|
|
if(m_copyFailed)
|
|
{
|
|
tryToRemove = true;
|
|
Amarok::StatusBar::instance()->longMessage(
|
|
i18n( "Media Device: Copying %1 to %2 failed" )
|
|
.arg( src.prettyURL(), dst.prettyURL() ),
|
|
KDE::StatusBar::Error );
|
|
}
|
|
else
|
|
{
|
|
MetaBundle bundle2(dst);
|
|
if(!bundle2.isValidMedia() && bundle2.filesize()==MetaBundle::Undetermined)
|
|
{
|
|
tryToRemove = true;
|
|
// probably s.th. went wrong
|
|
Amarok::StatusBar::instance()->longMessage(
|
|
i18n( "Media Device: Reading tags from %1 failed" ).arg( dst.prettyURL() ),
|
|
KDE::StatusBar::Error );
|
|
}
|
|
}
|
|
}
|
|
|
|
if( tryToRemove )
|
|
{
|
|
TQFile::remove( dst.path() );
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
MediaDevice::fileTransferred( KIO::Job *job ) //SLOT
|
|
{
|
|
if(job->error())
|
|
{
|
|
m_copyFailed = true;
|
|
debug() << "file transfer failed: " << job->errorText() << endl;
|
|
}
|
|
else
|
|
{
|
|
m_copyFailed = false;
|
|
}
|
|
|
|
m_wait = false;
|
|
}
|
|
|
|
void
|
|
MediaBrowser::cancelClicked()
|
|
{
|
|
DEBUG_BLOCK
|
|
|
|
m_waitForTranscode = false;
|
|
if( currentDevice() )
|
|
currentDevice()->abortTransfer();
|
|
}
|
|
|
|
void
|
|
MediaBrowser::transferClicked()
|
|
{
|
|
m_toolbar->getButton(TRANSFER)->setEnabled( false );
|
|
if( currentDevice()
|
|
&& currentDevice()->isConnected()
|
|
&& !currentDevice()->isTransferring() )
|
|
{
|
|
if( !currentDevice()->hasTransferDialog() )
|
|
currentDevice()->transferFiles();
|
|
else
|
|
{
|
|
currentDevice()->runTransferDialog();
|
|
//may not work with non-TransferDialog-class object, but maybe some run time introspection could solve it?
|
|
if( currentDevice()->getTransferDialog() &&
|
|
( reinterpret_cast<TransferDialog *>(currentDevice()->getTransferDialog()))->isAccepted() )
|
|
currentDevice()->transferFiles();
|
|
else
|
|
updateButtons();
|
|