diff --git a/admin/mac/add-Qt-to-bundle.sh b/admin/mac/add-Qt-to-bundle.sh
index e5c8f319f..8720e3f8d 100755
--- a/admin/mac/add-Qt-to-bundle.sh
+++ b/admin/mac/add-Qt-to-bundle.sh
@@ -54,8 +54,8 @@ cp -R $QT_PLUGINS_DIR/imageformats/libqgif.dylib Contents/MacOS/imageformats/
cp -R $QT_PLUGINS_DIR/imageformats/libqjpeg.dylib Contents/MacOS/imageformats/
cp -R $QT_PLUGINS_DIR/imageformats/libqico.dylib Contents/MacOS/imageformats/
cp -R $QT_PLUGINS_DIR/imageformats/libqmng.dylib Contents/MacOS/imageformats/
-cp -R $QT_PLUGINS_DIR/imageformats/libqsvg.dylib Contents/MacOS/imageformats/
-cp -R $QT_PLUGINS_DIR/imageformats/libqtiff.dylib Contents/MacOS/imageformats/
+#cp -R $QT_PLUGINS_DIR/imageformats/libqsvg.dylib Contents/MacOS/imageformats/
+#cp -R $QT_PLUGINS_DIR/imageformats/libqtiff.dylib Contents/MacOS/imageformats/
cp -R $QT_PLUGINS_DIR/crypto/libqca-ossl.dylib Contents/MacOS/crypto/
cp -R $QT_PLUGINS_DIR/phonon_backend/phonon_vlc.so Contents/MacOS/phonon_backend/
diff --git a/admin/mac/deposx.sh b/admin/mac/deposx.sh
index 8da4cd3e0..131cfea37 100755
--- a/admin/mac/deposx.sh
+++ b/admin/mac/deposx.sh
@@ -50,7 +50,11 @@ function deposx_change
install_name_tool -change /usr/local/Cellar/qt/$QTVERSION/lib/$y.framework/Versions/4/$y \
@executable_path/../Frameworks/$y.framework/Versions/4/$y \
- "$1"
+ "$1"
+
+ install_name_tool -change /usr/X11/lib/libpng12.0.dylib \
+ @executable_path/libpng12.0.dylib \
+ "$1"
done
for y in $LIBS
@@ -126,6 +130,7 @@ import_lib /usr/local/Cellar/kde-phonon/4.5.0/lib/libphonon.4.dylib
import_lib /usr/local/Cellar/vlc-git/HEAD/lib/libvlc.5.dylib
import_lib /usr/local/Cellar/vlc-git/HEAD/lib/libvlccore.4.dylib
import_lib /usr/local/Cellar/gettext/0.18.1.1/lib/libintl.8.dylib
+import_lib /usr/X11/lib/libpng12.0.dylib
import_lib $ORIGROOT/libjreen.dylib
import_lib $ORIGROOT/libtomahawklib.dylib
diff --git a/data/sql/dbmigrate-23_to_24.sql b/data/sql/dbmigrate-23_to_24.sql
new file mode 100644
index 000000000..31c35f9ab
--- /dev/null
+++ b/data/sql/dbmigrate-23_to_24.sql
@@ -0,0 +1,22 @@
+-- Script to migate from db version 23 to 24.
+-- Added the social_attributes table.
+--
+-- Separate each command with %%
+
+CREATE TABLE IF NOT EXISTS social_attributes (
+ id INTEGER REFERENCES track(id) ON DELETE CASCADE ON UPDATE CASCADE DEFERRABLE INITIALLY DEFERRED, -- track id
+ source INTEGER REFERENCES source(id) ON DELETE CASCADE ON UPDATE CASCADE, -- DEFERRABLE INITIALLY DEFERRED,
+ k TEXT NOT NULL,
+ v TEXT NOT NULL,
+ timestamp INTEGER NOT NULL DEFAULT 0
+);
+
+CREATE INDEX social_attrib_id ON social_attributes(id);
+
+CREATE INDEX social_attrib_source ON social_attributes(source);
+
+CREATE INDEX social_attrib_k ON social_attributes(k);
+
+CREATE INDEX social_attrib_timestamp ON social_attributes(timestamp);
+
+UPDATE settings SET v = '24' WHERE k == 'schema_version';
diff --git a/resources.qrc b/resources.qrc
index 5194ea273..4e88a8098 100644
--- a/resources.qrc
+++ b/resources.qrc
@@ -102,5 +102,6 @@
./data/www/auth.na.html
./data/www/tomahawk_banner_small.png
./data/sql/dbmigrate-22_to_23.sql
+ ./data/sql/dbmigrate-23_to_24.sql
diff --git a/src/audiocontrols.cpp b/src/audiocontrols.cpp
index 01536818c..7b060b327 100644
--- a/src/audiocontrols.cpp
+++ b/src/audiocontrols.cpp
@@ -314,10 +314,10 @@ AudioControls::onPlaybackResumed()
/* m_playAction->setEnabled( false );
m_pauseAction->setEnabled( true ); */
- ui->pauseButton->setVisible( true );
- ui->pauseButton->setEnabled( true );
ui->playPauseButton->setVisible( false );
ui->playPauseButton->setEnabled( false );
+ ui->pauseButton->setVisible( true );
+ ui->pauseButton->setEnabled( true );
}
diff --git a/src/libtomahawk/database/databasecommand_resolve.cpp b/src/libtomahawk/database/databasecommand_resolve.cpp
index 876a9685f..053283b47 100644
--- a/src/libtomahawk/database/databasecommand_resolve.cpp
+++ b/src/libtomahawk/database/databasecommand_resolve.cpp
@@ -36,6 +36,13 @@ DatabaseCommand_Resolve::DatabaseCommand_Resolve( const query_ptr& query )
void
DatabaseCommand_Resolve::exec( DatabaseImpl* lib )
{
+ /*
+ * Resolving is a 2 stage process.
+ * 1) find list of trk/art/alb IDs that are reasonable matches to the metadata given
+ * 2) find files in database by permitted sources and calculate score, ignoring
+ * results that are less than MINSCORE
+ */
+
if ( !m_query->resultHint().isEmpty() )
{
qDebug() << "Using result-hint to speed up resolving:" << m_query->resultHint();
@@ -64,20 +71,12 @@ void
DatabaseCommand_Resolve::resolve( DatabaseImpl* lib )
{
QList res;
-
- /*
- Resolving is a 2 stage process.
- 1) find list of trk/art/alb IDs that are reasonable matches to the metadata given
- 2) find files in database by permitted sources and calculate score, ignoring
- results that are less than MINSCORE
- */
-
typedef QPair scorepair_t;
// STEP 1
- QList< int > artists = lib->searchTable( "artist", m_query->artist(), 10 );
- QList< int > tracks = lib->searchTable( "track", m_query->track(), 10 );
- QList< int > albums = lib->searchTable( "album", m_query->album(), 10 );
+ QList< QPair > artists = lib->searchTable( "artist", m_query->artist(), 10 );
+ QList< QPair > tracks = lib->searchTable( "track", m_query->track(), 10 );
+ QList< QPair > albums = lib->searchTable( "album", m_query->album(), 10 );
if ( artists.length() == 0 || tracks.length() == 0 )
{
@@ -90,10 +89,10 @@ DatabaseCommand_Resolve::resolve( DatabaseImpl* lib )
TomahawkSqlQuery files_query = lib->newquery();
QStringList artsl, trksl;
- foreach( int i, artists )
- artsl.append( QString::number( i ) );
- foreach( int i, tracks )
- trksl.append( QString::number( i ) );
+ for ( int k = 0; k < artists.count(); k++ )
+ artsl.append( QString::number( artists.at( k ).first ) );
+ for ( int k = 0; k < tracks.count(); k++ )
+ trksl.append( QString::number( tracks.at( k ).first ) );
QString artsToken = QString( "file_join.artist IN (%1)" ).arg( artsl.join( "," ) );
QString trksToken = QString( "file_join.track IN (%1)" ).arg( trksl.join( "," ) );
@@ -174,7 +173,7 @@ DatabaseCommand_Resolve::resolve( DatabaseImpl* lib )
float score = how_similar( m_query, result );
result->setScore( score );
- if ( m_query->fullTextQuery().isEmpty() && score < MINSCORE )
+ if ( score < MINSCORE )
continue;
result->setCollection( s->collection() );
@@ -189,20 +188,12 @@ void
DatabaseCommand_Resolve::fullTextResolve( DatabaseImpl* lib )
{
QList res;
-
- /*
- * Resolving is a 2 stage process.
- * 1) find list of trk/art/alb IDs that are reasonable matches to the metadata given
- * 2) find files in database by permitted sources and calculate score, ignoring
- * results that are less than MINSCORE
- */
-
typedef QPair scorepair_t;
// STEP 1
- QList< int > artists = lib->searchTable( "artist", m_query->fullTextQuery(), 10 );
- QList< int > tracks = lib->searchTable( "track", m_query->fullTextQuery(), 10 );
- QList< int > albums = lib->searchTable( "album", m_query->fullTextQuery(), 10 );
+ QList< QPair > artists = lib->searchTable( "artist", m_query->fullTextQuery(), 10 );
+ QList< QPair > tracks = lib->searchTable( "track", m_query->fullTextQuery(), 10 );
+ QList< QPair > albums = lib->searchTable( "album", m_query->fullTextQuery(), 10 );
if ( artists.length() == 0 && tracks.length() == 0 && albums.length() == 0 )
{
@@ -215,12 +206,12 @@ DatabaseCommand_Resolve::fullTextResolve( DatabaseImpl* lib )
TomahawkSqlQuery files_query = lib->newquery();
QStringList artsl, trksl, albsl;
- foreach( int i, artists )
- artsl.append( QString::number( i ) );
- foreach( int i, tracks )
- trksl.append( QString::number( i ) );
- foreach( int i, albums )
- albsl.append( QString::number( i ) );
+ for ( int k = 0; k < artists.count(); k++ )
+ artsl.append( QString::number( artists.at( k ).first ) );
+ for ( int k = 0; k < tracks.count(); k++ )
+ trksl.append( QString::number( tracks.at( k ).first ) );
+ for ( int k = 0; k < albums.count(); k++ )
+ albsl.append( QString::number( albums.at( k ).first ) );
QString artsToken = QString( "file_join.artist IN (%1)" ).arg( artsl.join( "," ) );
QString trksToken = QString( "file_join.track IN (%1)" ).arg( trksl.join( "," ) );
@@ -241,10 +232,8 @@ DatabaseCommand_Resolve::fullTextResolve( DatabaseImpl* lib )
"artist.id = file_join.artist AND "
"track.id = file_join.track AND "
"file.id = file_join.file AND "
- "(%1 OR %2 OR %3)" )
- .arg( artists.length() > 0 ? artsToken : QString( "0" ) )
- .arg( tracks.length() > 0 ? trksToken : QString( "0" ) )
- .arg( albums.length() > 0 ? albsToken : QString( "0" ) );
+ "%1" )
+ .arg( tracks.length() > 0 ? trksToken : QString( "0" ) );
files_query.prepare( sql );
files_query.exec();
@@ -288,6 +277,15 @@ DatabaseCommand_Resolve::fullTextResolve( DatabaseImpl* lib )
result->setId( files_query.value( 9 ).toUInt() );
result->setYear( files_query.value( 17 ).toUInt() );
+ for ( int k = 0; k < tracks.count(); k++ )
+ {
+ if ( tracks.at( k ).first == (int)result->dbid() )
+ {
+ result->setScore( tracks.at( k ).second );
+ break;
+ }
+ }
+
TomahawkSqlQuery attrQuery = lib->newquery();
QVariantMap attr;
@@ -301,11 +299,6 @@ DatabaseCommand_Resolve::fullTextResolve( DatabaseImpl* lib )
result->setAttributes( attr );
- float score = how_similar( m_query, result );
- result->setScore( score );
- if ( m_query->fullTextQuery().isEmpty() && score < MINSCORE )
- continue;
-
result->setCollection( s->collection() );
res << result;
}
diff --git a/src/libtomahawk/database/databaseimpl.cpp b/src/libtomahawk/database/databaseimpl.cpp
index b41852ad4..d8c21b024 100644
--- a/src/libtomahawk/database/databaseimpl.cpp
+++ b/src/libtomahawk/database/databaseimpl.cpp
@@ -38,7 +38,7 @@
*/
#include "schema.sql.h"
-#define CURRENT_SCHEMA_VERSION 23
+#define CURRENT_SCHEMA_VERSION 24
DatabaseImpl::DatabaseImpl( const QString& dbname, Database* parent )
@@ -47,67 +47,48 @@ DatabaseImpl::DatabaseImpl( const QString& dbname, Database* parent )
, m_lastalbid( 0 )
, m_lasttrkid( 0 )
{
- db = QSqlDatabase::addDatabase( "QSQLITE", "tomahawk" );
- db.setDatabaseName( dbname );
- if ( !db.open() )
- {
- qDebug() << "FAILED TO OPEN DB";
- throw "failed to open db"; // TODO
- }
-
- QSqlQuery qry = QSqlQuery( db );
-
bool schemaUpdated = false;
- qry.exec( "SELECT v FROM settings WHERE k='schema_version'" );
- if ( qry.next() )
+ int version = getDatabaseVersion( dbname );
+
+ if ( version > 0 && version != CURRENT_SCHEMA_VERSION )
{
- int v = qry.value( 0 ).toInt();
- qDebug() << "Current schema is" << v << this->thread();
- if ( v != CURRENT_SCHEMA_VERSION )
+ QString newname = QString( "%1.v%2" ).arg( dbname ).arg( version );
+ qDebug() << endl << "****************************" << endl;
+ qDebug() << "Schema version too old: " << version << ". Current version is:" << CURRENT_SCHEMA_VERSION;
+ qDebug() << "Moving" << dbname << newname;
+ qDebug() << "If the migration fails, you can recover your DB by copying" << newname << "back to" << dbname;
+ qDebug() << endl << "****************************" << endl;
+
+ QFile::copy( dbname, newname );
{
+ db = QSqlDatabase::addDatabase( "QSQLITE", "tomahawk" );
+ db.setDatabaseName( dbname );
+ if( !db.open() )
+ throw "db moving failed";
- QString newname = QString("%1.v%2").arg(dbname).arg(v);
- qDebug() << endl << "****************************" << endl;
- qDebug() << "Schema version too old: " << v << ". Current version is:" << CURRENT_SCHEMA_VERSION;
- qDebug() << "Moving" << dbname << newname;
- qDebug() << "If the migration fails, you can recover your DB by copying" << newname << "back to" << dbname;
- qDebug() << endl << "****************************" << endl;
+ TomahawkSqlQuery query = newquery();
+ query.exec( "PRAGMA auto_vacuum = FULL" );
- qry.clear();
- qry.finish();
-
- db.close();
- db.removeDatabase( "tomahawk" );
-
- if( QFile::copy( dbname, newname ) )
- {
- db = QSqlDatabase::addDatabase( "QSQLITE", "tomahawk" );
- db.setDatabaseName( dbname );
- if( !db.open() )
- throw "db moving failed";
-
- TomahawkSqlQuery query = newquery();
- query.exec( "PRAGMA auto_vacuum = FULL" );
- schemaUpdated = updateSchema( v );
-
- if( !schemaUpdated )
- {
- Q_ASSERT( false );
- QTimer::singleShot( 0, qApp, SLOT( quit() ) );
- }
-
- }
- else
+ schemaUpdated = updateSchema( version );
+ if ( !schemaUpdated )
{
Q_ASSERT( false );
QTimer::singleShot( 0, qApp, SLOT( quit() ) );
- return;
}
}
}
else
{
- schemaUpdated = updateSchema( 0 );
+ db = QSqlDatabase::addDatabase( "QSQLITE", "tomahawk" );
+ db.setDatabaseName( dbname );
+ if ( !db.open() )
+ {
+ qDebug() << "Failed to open database" << dbname;
+ throw "failed to open db"; // TODO
+ }
+
+ if ( version < 0 )
+ schemaUpdated = updateSchema( 0 );
}
TomahawkSqlQuery query = newquery();
@@ -422,30 +403,23 @@ DatabaseImpl::albumId( int artistid, const QString& name_orig, bool& isnew )
}
-QList< int >
+QList< QPair >
DatabaseImpl::searchTable( const QString& table, const QString& name, uint limit )
{
Q_UNUSED( limit );
- QList< int > results;
- if( table != "artist" && table != "track" && table != "album" )
- return results;
-
- QMap< int, float > resultsmap = m_fuzzyIndex->search( table, name );
QList< QPair > resultslist;
+ if( table != "artist" && table != "track" && table != "album" )
+ return resultslist;
+
+ QMap< int, float > resultsmap = m_fuzzyIndex->search( table, name );
foreach( int i, resultsmap.keys() )
{
resultslist << QPair( i, (float)resultsmap.value( i ) );
}
qSort( resultslist.begin(), resultslist.end(), DatabaseImpl::scorepairSorter );
- for( int k = 0; k < resultslist.count(); k++ )
- {
- results << resultslist.at( k ).first;
- }
-
-// qDebug() << "Returning" << results.count() << "results";
- return results;
+ return resultslist;
}
@@ -639,3 +613,31 @@ DatabaseImpl::resultFromHint( const Tomahawk::query_ptr& origquery )
return res;
}
+
+
+int
+DatabaseImpl::getDatabaseVersion( const QString& dbname )
+{
+ int version = -1;
+ {
+ QSqlDatabase db = QSqlDatabase::addDatabase( "QSQLITE", "tomahawk" );
+ db.setDatabaseName( dbname );
+ if ( !db.open() )
+ {
+ qDebug() << "Failed to open database" << dbname;
+ throw "failed to open db"; // TODO
+ }
+
+ QSqlQuery qry = QSqlQuery( db );
+ qry.exec( "SELECT v FROM settings WHERE k='schema_version'" );
+ if ( qry.next() )
+ {
+ version = qry.value( 0 ).toInt();
+ qDebug() << "Database schema of" << dbname << "is" << version;
+ }
+ }
+
+ QSqlDatabase::removeDatabase( "tomahawk" );
+
+ return version;
+}
diff --git a/src/libtomahawk/database/databaseimpl.h b/src/libtomahawk/database/databaseimpl.h
index 243f04edd..151c5ee27 100644
--- a/src/libtomahawk/database/databaseimpl.h
+++ b/src/libtomahawk/database/databaseimpl.h
@@ -45,6 +45,8 @@ friend class FuzzyIndex;
friend class DatabaseCommand_UpdateSearchIndex;
public:
+ static int getDatabaseVersion( const QString& dbname );
+
DatabaseImpl( const QString& dbname, Database* parent = 0 );
~DatabaseImpl();
@@ -55,7 +57,7 @@ public:
int trackId( int artistid, const QString& name_orig, bool& isnew );
int albumId( int artistid, const QString& name_orig, bool& isnew );
- QList< int > searchTable( const QString& table, const QString& name, uint limit = 10 );
+ QList< QPair > searchTable( const QString& table, const QString& name, uint limit = 10 );
QList< int > getTrackFids( int tid );
static QString sortname( const QString& str );
@@ -85,18 +87,15 @@ public slots:
private:
QString cleanSql( const QString& sql );
-
- bool m_ready;
-
bool updateSchema( int oldVersion );
+ bool m_ready;
QSqlDatabase db;
QString m_lastart, m_lastalb, m_lasttrk;
int m_lastartid, m_lastalbid, m_lasttrkid;
QString m_dbid;
-
FuzzyIndex* m_fuzzyIndex;
};
diff --git a/src/libtomahawk/database/schema.sql b/src/libtomahawk/database/schema.sql
index e2f6ce90d..1f5e5593b 100644
--- a/src/libtomahawk/database/schema.sql
+++ b/src/libtomahawk/database/schema.sql
@@ -227,6 +227,24 @@ CREATE TABLE IF NOT EXISTS track_attributes (
CREATE INDEX track_attrib_id ON track_attributes(id);
CREATE INDEX track_attrib_k ON track_attributes(k);
+-- social attributes connected to the track.
+-- like love, hate, comments, recommendations
+-- love=[comment], hate=[comment], comment=Some text
+-- NB: since all values are text, numeric values should be zero-padded to a set amount
+-- so that we can always do range queries.
+
+CREATE TABLE IF NOT EXISTS social_attributes (
+ id INTEGER REFERENCES track(id) ON DELETE CASCADE ON UPDATE CASCADE DEFERRABLE INITIALLY DEFERRED, -- track id
+ source INTEGER REFERENCES source(id) ON DELETE CASCADE ON UPDATE CASCADE, -- DEFERRABLE INITIALLY DEFERRED,
+ k TEXT NOT NULL,
+ v TEXT NOT NULL,
+ timestamp INTEGER NOT NULL DEFAULT 0
+);
+CREATE INDEX social_attrib_id ON social_attributes(id);
+CREATE INDEX social_attrib_source ON social_attributes(source);
+CREATE INDEX social_attrib_k ON social_attributes(k);
+CREATE INDEX social_attrib_timestamp ON social_attributes(timestamp);
+
-- playback history
@@ -264,4 +282,4 @@ CREATE TABLE IF NOT EXISTS settings (
v TEXT NOT NULL DEFAULT ''
);
-INSERT INTO settings(k,v) VALUES('schema_version', '23');
+INSERT INTO settings(k,v) VALUES('schema_version', '24');
diff --git a/src/libtomahawk/database/schema.sql.h b/src/libtomahawk/database/schema.sql.h
index 864766dc6..bd085067a 100644
--- a/src/libtomahawk/database/schema.sql.h
+++ b/src/libtomahawk/database/schema.sql.h
@@ -1,5 +1,5 @@
/*
- This file was automatically generated from schema.sql on Fri Apr 15 15:55:52 EDT 2011.
+ This file was automatically generated from ./schema.sql on Sun Jun 12 05:17:25 CEST 2011.
*/
static const char * tomahawk_schema_sql =
@@ -151,6 +151,17 @@ static const char * tomahawk_schema_sql =
");"
"CREATE INDEX track_attrib_id ON track_attributes(id);"
"CREATE INDEX track_attrib_k ON track_attributes(k);"
+"CREATE TABLE IF NOT EXISTS social_attributes ("
+" id INTEGER REFERENCES track(id) ON DELETE CASCADE ON UPDATE CASCADE DEFERRABLE INITIALLY DEFERRED, "
+" source INTEGER REFERENCES source(id) ON DELETE CASCADE ON UPDATE CASCADE, "
+" k TEXT NOT NULL,"
+" v TEXT NOT NULL,"
+" timestamp INTEGER NOT NULL DEFAULT 0"
+");"
+"CREATE INDEX social_attrib_id ON social_attributes(id);"
+"CREATE INDEX social_attrib_source ON social_attributes(source);"
+"CREATE INDEX social_attrib_k ON social_attributes(k);"
+"CREATE INDEX social_attrib_timestamp ON social_attributes(timestamp);"
"CREATE TABLE IF NOT EXISTS playback_log ("
" id INTEGER PRIMARY KEY AUTOINCREMENT,"
" source INTEGER REFERENCES source(id) ON DELETE CASCADE ON UPDATE CASCADE DEFERRABLE INITIALLY DEFERRED,"
@@ -172,7 +183,7 @@ static const char * tomahawk_schema_sql =
" k TEXT NOT NULL PRIMARY KEY,"
" v TEXT NOT NULL DEFAULT ''"
");"
-"INSERT INTO settings(k,v) VALUES('schema_version', '23');"
+"INSERT INTO settings(k,v) VALUES('schema_version', '24');"
;
const char * get_tomahawk_sql()
diff --git a/src/libtomahawk/globalactionmanager.cpp b/src/libtomahawk/globalactionmanager.cpp
index bf60833f6..ad85da31c 100644
--- a/src/libtomahawk/globalactionmanager.cpp
+++ b/src/libtomahawk/globalactionmanager.cpp
@@ -78,9 +78,9 @@ GlobalActionManager::openLinkFromQuery( const Tomahawk::query_ptr& query ) const
}
QUrl
-GlobalActionManager::openLink( const QString& title, const QString& artist, const QString& album, bool tomahk ) const
+GlobalActionManager::openLink( const QString& title, const QString& artist, const QString& album ) const
{
- QUrl link( tomahk ? "http://toma.hk/open/track/" : "tomahawk://open/track/" );
+ QUrl link( QString( "%1/open/track/" ).arg( hostname() ) );
if( !title.isEmpty() )
link.addQueryItem( "title", title );
@@ -95,7 +95,7 @@ GlobalActionManager::openLink( const QString& title, const QString& artist, cons
QString
GlobalActionManager::copyPlaylistToClipboard( const Tomahawk::dynplaylist_ptr& playlist )
{
- QUrl link( QString( "tomahawk://%1/create/" ).arg( playlist->mode() == Tomahawk::OnDemand ? "station" : "autoplaylist" ) );
+ QUrl link( QString( "%1/%2/create/" ).arg( hostname() ).arg( playlist->mode() == Tomahawk::OnDemand ? "station" : "autoplaylist" ) );
if( playlist->generator()->type() != "echonest" ) {
qDebug() << "Only echonest generators are supported";
@@ -676,4 +676,8 @@ GlobalActionManager::waitingForResolved( bool success )
m_waitingToPlay.clear();
}
-
+QString
+GlobalActionManager::hostname() const
+{
+ return QString( "http://toma.hk" );
+}
diff --git a/src/libtomahawk/globalactionmanager.h b/src/libtomahawk/globalactionmanager.h
index d8186559a..f54605da8 100644
--- a/src/libtomahawk/globalactionmanager.h
+++ b/src/libtomahawk/globalactionmanager.h
@@ -36,7 +36,7 @@ public:
virtual ~GlobalActionManager();
QUrl openLinkFromQuery( const Tomahawk::query_ptr& query ) const;
- QUrl openLink( const QString& title, const QString& artist, const QString& album, bool tomahk=false ) const;
+ QUrl openLink( const QString& title, const QString& artist, const QString& album ) const;
void copyToClipboard( const Tomahawk::query_ptr& query ) const;
QString copyPlaylistToClipboard( const Tomahawk::dynplaylist_ptr& playlist );
@@ -68,6 +68,7 @@ private:
bool handleOpenCommand(const QUrl& url );
bool doQueueAdd( const QStringList& parts, const QList< QPair< QString, QString > >& queryItems );
+ QString hostname() const;
Tomahawk::playlist_ptr m_toShow;
Tomahawk::query_ptr m_waitingToBookmark;
diff --git a/src/libtomahawk/infosystem/infoplugins/adiumplugin.cpp b/src/libtomahawk/infosystem/infoplugins/adiumplugin.cpp
index 44bdaeb10..0d47a1b36 100644
--- a/src/libtomahawk/infosystem/infoplugins/adiumplugin.cpp
+++ b/src/libtomahawk/infosystem/infoplugins/adiumplugin.cpp
@@ -18,6 +18,8 @@
#include
+#include
+
#include "infosystem/infosystemworker.h"
#include "artist.h"
#include "result.h"
@@ -65,6 +67,10 @@ AdiumPlugin::AdiumPlugin()
connect( TomahawkSettings::instance(), SIGNAL( changed() ),
SLOT( settingsChanged() ), Qt::QueuedConnection );
+
+ m_pauseTimer = new QTimer( this );
+ connect( m_pauseTimer, SIGNAL( timeout() ),
+ this, SLOT( clearStatus() ) );
}
AdiumPlugin::~AdiumPlugin()
@@ -73,6 +79,13 @@ AdiumPlugin::~AdiumPlugin()
setStatus( "" );
}
+void
+AdiumPlugin::clearStatus()
+{
+ qDebug() << Q_FUNC_INFO;
+ setStatus( "" );
+}
+
void
AdiumPlugin::settingsChanged()
{
@@ -107,9 +120,9 @@ AdiumPlugin::pushInfo( const QString caller, const Tomahawk::InfoSystem::InfoTyp
case InfoNowPlaying:
audioStarted( input );
break;
- // case InfoNowPaused:
- // audioPaused();
- // break;
+ case InfoNowPaused:
+ audioPaused();
+ return;
case InfoNowResumed:
audioResumed( input );
break;
@@ -120,6 +133,9 @@ AdiumPlugin::pushInfo( const QString caller, const Tomahawk::InfoSystem::InfoTyp
default:
return;
}
+
+ // Stop the pause timer always, unless pausing of course
+ m_pauseTimer->stop();
}
/** Audio state slots */
@@ -136,11 +152,14 @@ AdiumPlugin::audioStarted( const QVariant &input )
return;
QString nowPlaying = "";
- nowPlaying.append( hash["title"] );
- nowPlaying.append(" - ");
nowPlaying.append( hash["artist"] );
+ nowPlaying.append(" - ");
+ nowPlaying.append( hash["title"] );
nowPlaying.append( " " );
+ // Escape quotes, or Applescript gets confused
+ nowPlaying.replace( "\"", "\\\"" );
nowPlaying.append( openLinkFromHash( hash ).toEncoded() );
+ qDebug() << "nowPlaying: " << nowPlaying;
setStatus( nowPlaying );
}
@@ -157,7 +176,7 @@ AdiumPlugin::openLinkFromHash( const Tomahawk::InfoSystem::InfoCriteriaHash& has
album = hash["album"];
}
- return GlobalActionManager::instance()->openLink( title, artist, album, true );
+ return GlobalActionManager::instance()->openLink( title, artist, album );
}
void
@@ -177,14 +196,12 @@ void
AdiumPlugin::audioPaused()
{
qDebug() << Q_FUNC_INFO;
-
- //setStatus( "Paused" );
+ m_pauseTimer->start( 60 * 1000 );
}
void
AdiumPlugin::audioResumed( const QVariant &input )
{
qDebug() << Q_FUNC_INFO;
- // TODO: audio resumed, so push update status to Adium with playing track
audioStarted( input );
}
diff --git a/src/libtomahawk/infosystem/infoplugins/adiumplugin.h b/src/libtomahawk/infosystem/infoplugins/adiumplugin.h
index 724c97ca3..e4a7f85c9 100644
--- a/src/libtomahawk/infosystem/infoplugins/adiumplugin.h
+++ b/src/libtomahawk/infosystem/infoplugins/adiumplugin.h
@@ -24,6 +24,8 @@
#include
#include
+class QTimer;
+
namespace Tomahawk {
namespace InfoSystem {
@@ -44,6 +46,9 @@ public slots:
void namChangedSlot( QNetworkAccessManager *nam ) {} // unused
void notInCacheSlot( const Tomahawk::InfoSystem::InfoCriteriaHash criteria, const QString caller, const Tomahawk::InfoSystem::InfoType type, const QVariant input, const Tomahawk::InfoSystem::InfoCustomData customData ) {} // unused
+private slots:
+ void clearStatus();
+
private:
void settingsChanged();
@@ -59,6 +64,8 @@ private:
QString m_beforeStatus;
QString m_afterStatus;
+ QTimer* m_pauseTimer;
+
};
diff --git a/src/libtomahawk/playlist.cpp b/src/libtomahawk/playlist.cpp
index d3e88a764..bb3ff7a6a 100644
--- a/src/libtomahawk/playlist.cpp
+++ b/src/libtomahawk/playlist.cpp
@@ -220,6 +220,8 @@ Playlist::load( const QString& guid )
bool
Playlist::remove( const playlist_ptr& playlist )
{
+ playlist->aboutToBeDeleted( playlist );
+
TomahawkSettings *s = TomahawkSettings::instance();
s->removePlaylistSettings( playlist->guid() );
diff --git a/src/libtomahawk/playlist.h b/src/libtomahawk/playlist.h
index ba705328c..dfbccb8e8 100644
--- a/src/libtomahawk/playlist.h
+++ b/src/libtomahawk/playlist.h
@@ -190,6 +190,12 @@ signals:
/// renamed etc.
void changed();
+ /**
+ * delete command is scheduled but not completed. Do not call remove() again once this
+ * is emitted.
+ */
+ void aboutToBeDeleted( const Tomahawk::playlist_ptr& pl );
+
/// was deleted, eh?
void deleted( const Tomahawk::playlist_ptr& pl );
diff --git a/src/libtomahawk/playlist/collectionproxymodel.cpp b/src/libtomahawk/playlist/collectionproxymodel.cpp
index 46972c6c8..a7eb2f572 100644
--- a/src/libtomahawk/playlist/collectionproxymodel.cpp
+++ b/src/libtomahawk/playlist/collectionproxymodel.cpp
@@ -29,109 +29,3 @@ CollectionProxyModel::CollectionProxyModel( QObject* parent )
: TrackProxyModel( parent )
{
}
-
-
-bool
-CollectionProxyModel::lessThan( const QModelIndex& left, const QModelIndex& right ) const
-{
- TrackModelItem* p1 = itemFromIndex( left );
- TrackModelItem* p2 = itemFromIndex( right );
-
- if ( !p1 )
- return true;
- if ( !p2 )
- return false;
-
- const Tomahawk::query_ptr& q1 = p1->query();
- const Tomahawk::query_ptr& q2 = p2->query();
-
- QString artist1 = q1->artist();
- QString artist2 = q2->artist();
- QString album1 = q1->album();
- QString album2 = q2->album();
- QString track1 = q1->track();
- QString track2 = q2->track();
- unsigned int albumpos1 = 0, albumpos2 = 0;
- unsigned int bitrate1 = 0, bitrate2 = 0;
- unsigned int mtime1 = 0, mtime2 = 0;
- unsigned int id1 = 0, id2 = 0;
- unsigned int size1 = 0, size2 = 0;
-
- if ( q1->numResults() )
- {
- const Tomahawk::result_ptr& r = q1->results().at( 0 );
- artist1 = r->artist()->name();
- album1 = r->album()->name();
- track1 = r->track();
- albumpos1 = r->albumpos();
- bitrate1 = r->bitrate();
- mtime1 = r->modificationTime();
- id1 = r->dbid();
- size1 = r->size();
- }
- if ( q2->numResults() )
- {
- const Tomahawk::result_ptr& r = q2->results().at( 0 );
- artist2 = r->artist()->name();
- album2 = r->album()->name();
- track2 = r->track();
- albumpos2 = r->albumpos();
- bitrate2 = r->bitrate();
- mtime2 = r->modificationTime();
- id2 = r->dbid();
- size2 = r->size();
- }
-
- if ( left.column() == TrackModel::Artist ) // sort by artist
- {
- if ( artist1 == artist2 )
- {
- if ( album1 == album2 )
- {
- if ( albumpos1 == albumpos2 )
- return id1 < id2;
-
- return albumpos1 < albumpos2;
- }
-
- return QString::localeAwareCompare( album1, album2 ) < 0;
- }
-
- return QString::localeAwareCompare( artist1, artist2 ) < 0;
- }
- else if ( left.column() == TrackModel::Album ) // sort by album
- {
- if ( album1 == album2 )
- {
- if ( albumpos1 == albumpos2 )
- return id1 < id2;
-
- return albumpos1 < albumpos2;
- }
-
- return QString::localeAwareCompare( album1, album2 ) < 0;
- }
- else if ( left.column() == TrackModel::Bitrate ) // sort by bitrate
- {
- if ( bitrate1 == bitrate2 )
- return id1 < id2;
-
- return bitrate1 < bitrate2;
- }
- else if ( left.column() == TrackModel::Age ) // sort by mtime
- {
- if ( mtime1 == mtime2 )
- return id1 < id2;
-
- return mtime1 < mtime2;
- }
- else if ( left.column() == TrackModel::Filesize ) // sort by file size
- {
- if ( size1 == size2 )
- return id1 < id2;
-
- return size1 < size2;
- }
- return QString::localeAwareCompare( sourceModel()->data( left ).toString(),
- sourceModel()->data( right ).toString() ) < 0;
-}
diff --git a/src/libtomahawk/playlist/collectionproxymodel.h b/src/libtomahawk/playlist/collectionproxymodel.h
index 63c615f40..61abd3df4 100644
--- a/src/libtomahawk/playlist/collectionproxymodel.h
+++ b/src/libtomahawk/playlist/collectionproxymodel.h
@@ -1,5 +1,5 @@
/* === This file is part of Tomahawk Player - ===
- *
+ *
* Copyright 2010-2011, Christian Muehlhaeuser
*
* Tomahawk is free software: you can redistribute it and/or modify
@@ -31,9 +31,6 @@ public:
explicit CollectionProxyModel( QObject* parent = 0 );
virtual PlaylistInterface::ViewMode viewMode() const { return PlaylistInterface::Flat; }
-
-protected:
- bool lessThan( const QModelIndex& left, const QModelIndex& right ) const;
};
#endif // COLLECTIONPROXYMODEL_H
diff --git a/src/libtomahawk/playlist/collectionview.cpp b/src/libtomahawk/playlist/collectionview.cpp
index 1b4d02c0f..8d3895389 100644
--- a/src/libtomahawk/playlist/collectionview.cpp
+++ b/src/libtomahawk/playlist/collectionview.cpp
@@ -63,8 +63,12 @@ void
CollectionView::setTrackModel( TrackModel* model )
{
TrackView::setTrackModel( model );
+ setColumnHidden( TrackModel::Score, true ); // Hide age column per default
setGuid( "collectionview" );
+ setSortingEnabled( true );
+ sortByColumn( 0, Qt::AscendingOrder );
+
connect( model, SIGNAL( trackCountChanged( unsigned int ) ), SLOT( onTrackCountChanged( unsigned int ) ) );
}
@@ -83,14 +87,14 @@ CollectionView::setupMenus()
m_playItemAction = m_itemMenu.addAction( tr( "&Play" ) );
m_addItemsToQueueAction = m_itemMenu.addAction( tr( "Add to &Queue" ) );
- m_itemMenu.addSeparator();
+ m_itemMenu.addSeparator();
- foreach( QAction* a, actions() )
- m_itemMenu.addAction( a );
+ foreach( QAction* a, actions() )
+ m_itemMenu.addAction( a );
// m_addItemsToPlaylistAction = m_itemMenu.addAction( tr( "&Add to Playlist" ) );
- connect( m_playItemAction, SIGNAL( triggered() ), SLOT( playItem() ) );
- connect( m_addItemsToQueueAction, SIGNAL( triggered() ), SLOT( addItemsToQueue() ) );
+ connect( m_playItemAction, SIGNAL( triggered() ), SLOT( playItem() ) );
+ connect( m_addItemsToQueueAction, SIGNAL( triggered() ), SLOT( addItemsToQueue() ) );
// connect( m_addItemsToPlaylistAction, SIGNAL( triggered() ), SLOT( addItemsToPlaylist() ) );
}
diff --git a/src/libtomahawk/playlist/dynamic/DynamicPlaylist.cpp b/src/libtomahawk/playlist/dynamic/DynamicPlaylist.cpp
index e69dbb320..205c7e8e2 100644
--- a/src/libtomahawk/playlist/dynamic/DynamicPlaylist.cpp
+++ b/src/libtomahawk/playlist/dynamic/DynamicPlaylist.cpp
@@ -26,6 +26,7 @@
#include "database/databasecommand_setdynamicplaylistrevision.h"
#include "database/databasecommand_loaddynamicplaylist.h"
#include "database/databasecommand_deletedynamicplaylist.h"
+#include "tomahawksettings.h"
using namespace Tomahawk;
@@ -249,6 +250,10 @@ DynamicPlaylist::loadRevision( const QString& rev )
bool
DynamicPlaylist::remove( const Tomahawk::dynplaylist_ptr& playlist )
{
+ playlist->aboutToBeDeleted( playlist );
+
+ TomahawkSettings::instance()->removePlaylistSettings( playlist->guid() );
+
DatabaseCommand_DeletePlaylist* cmd = new DatabaseCommand_DeleteDynamicPlaylist( playlist->author(), playlist->guid() );
Database::instance()->enqueue( QSharedPointer(cmd) );
diff --git a/src/libtomahawk/playlist/dynamic/DynamicPlaylist.h b/src/libtomahawk/playlist/dynamic/DynamicPlaylist.h
index c53c96dd2..90a63da6e 100644
--- a/src/libtomahawk/playlist/dynamic/DynamicPlaylist.h
+++ b/src/libtomahawk/playlist/dynamic/DynamicPlaylist.h
@@ -115,6 +115,7 @@ signals:
/// emitted when the playlist revision changes (whenever the playlist changes)
void dynamicRevisionLoaded( Tomahawk::DynamicPlaylistRevision );
+ void aboutToBeDeleted( const Tomahawk::dynplaylist_ptr& pl );
void deleted( const Tomahawk::dynplaylist_ptr& pl );
public slots:
diff --git a/src/libtomahawk/playlist/playlistitemdelegate.cpp b/src/libtomahawk/playlist/playlistitemdelegate.cpp
index e06bdb9c4..4bdb0ac0e 100644
--- a/src/libtomahawk/playlist/playlistitemdelegate.cpp
+++ b/src/libtomahawk/playlist/playlistitemdelegate.cpp
@@ -18,6 +18,7 @@
#include "playlistitemdelegate.h"
+#include
#include
#include
@@ -36,6 +37,7 @@
PlaylistItemDelegate::PlaylistItemDelegate( TrackView* parent, TrackProxyModel* proxy )
: QStyledItemDelegate( (QObject*)parent )
+ , m_style( Detailed )
, m_view( parent )
, m_model( proxy )
{
@@ -43,6 +45,13 @@ PlaylistItemDelegate::PlaylistItemDelegate( TrackView* parent, TrackProxyModel*
}
+void
+PlaylistItemDelegate::setStyle( PlaylistItemDelegate::PlaylistItemStyle style )
+{
+ m_style = style;
+}
+
+
void
PlaylistItemDelegate::updateRowSize( const QModelIndex& index )
{
@@ -70,6 +79,29 @@ PlaylistItemDelegate::createEditor( QWidget* parent, const QStyleOptionViewItem&
void
PlaylistItemDelegate::paint( QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index ) const
+{
+ switch ( m_style )
+ {
+ case Detailed:
+ paintDetailed( painter, option, index );
+ break;
+
+ case Short:
+ paintShort( painter, option, index );
+ break;
+ }
+}
+
+
+void
+PlaylistItemDelegate::paintShort( QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index ) const
+{
+
+}
+
+
+void
+PlaylistItemDelegate::paintDetailed( QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index ) const
{
TrackModelItem* item = m_model->itemFromIndex( m_model->mapToSource( index ) );
if ( !item || item->query().isNull() )
@@ -82,13 +114,51 @@ PlaylistItemDelegate::paint( QPainter* painter, const QStyleOptionViewItem& opti
opacity = qMax( (float)0.3, opacity );
QColor textColor = TomahawkUtils::alphaBlend( option.palette.color( QPalette::Foreground ), option.palette.color( QPalette::Background ), opacity );
+ QStyleOptionViewItemV4 opt = option;
+ initStyleOption( &opt, index );
+
if ( item->isPlaying() )
{
-// painter->setRenderHint( QPainter::Antialiasing );
- painter->save();
+ opt.palette.setColor( QPalette::Highlight, opt.palette.color( QPalette::Mid ) );
+ opt.state |= QStyle::State_Selected;
+ }
+ if ( item->isPlaying() || index.column() == TrackModel::Score )
+ opt.text.clear();
+ if ( opt.state & QStyle::State_Selected )
+ opt.palette.setColor( QPalette::Text, opt.palette.color( QPalette::HighlightedText ) );
+ else
+ opt.palette.setColor( QPalette::Text, textColor );
+ qApp->style()->drawControl( QStyle::CE_ItemViewItem, &opt, painter );
+
+ painter->save();
+ if ( index.column() == TrackModel::Score )
+ {
+ if ( opt.state & QStyle::State_Selected )
+ painter->setPen( opt.palette.brightText().color() );
+ else
+ painter->setPen( opt.palette.highlight().color() );
+
+ QRect r = opt.rect.adjusted( 3, 3, -6, -5 );
+ painter->drawRect( r );
+
+ QRect fillR = r;
+ int fillerWidth = (int)( index.data().toFloat() * (float)fillR.width() );
+ fillR.adjust( 0, 0, -( fillR.width() - fillerWidth ), 0 );
+
+ if ( opt.state & QStyle::State_Selected )
+ painter->setBrush( opt.palette.brightText().color() );
+ else
+ painter->setBrush( opt.palette.highlight().color() );
+
+ painter->drawRect( fillR );
+ }
+ else if ( item->isPlaying() )
+ {
{
- QRect r = option.rect.adjusted( 3, 0, 0, 0 );
+ QRect r = opt.rect.adjusted( 3, 0, 0, 0 );
+
+ // Paint Now Playing Speaker Icon
if ( m_view->header()->visualIndex( index.column() ) == 0 )
{
r.adjust( 0, 0, 0, -3 );
@@ -96,34 +166,22 @@ PlaylistItemDelegate::paint( QPainter* painter, const QStyleOptionViewItem& opti
r.adjust( 22, 0, 0, 3 );
}
- painter->setPen( option.palette.text().color() );
+ painter->setPen( opt.palette.text().color() );
QTextOption to( Qt::AlignVCenter );
QString text = painter->fontMetrics().elidedText( index.data().toString(), Qt::ElideRight, r.width() - 3 );
painter->drawText( r.adjusted( 0, 1, 0, 0 ), text, to );
}
-// if ( m_view->header()->visualIndex( index.column() ) == m_view->header()->visibleSectionCount() - 1 )
- {
- QRect r = QRect( 3, option.rect.y() + 1, m_view->viewport()->width() - 6, option.rect.height() - 2 );
- painter->setPen( option.palette.highlight().color() );
+ // Paint Now Playing Frame
+/* {
+ QRect r = QRect( 3, opt.rect.y() + 1, m_view->viewport()->width() - 6, opt.rect.height() - 2 );
+ painter->setPen( opt.palette.highlight().color() );
QPen pen = painter->pen();
pen.setWidth( 1.0 );
painter->setPen( pen );
painter->drawRoundedRect( r, 3.0, 3.0 );
- }
-
- painter->restore();
- }
- else
- {
- if ( const QStyleOptionViewItem *vioption = qstyleoption_cast(&option))
- {
- QStyleOptionViewItemV4 o( *vioption );
- o.palette.setColor( QPalette::Text, textColor );
- QStyledItemDelegate::paint( painter, o, index );
- }
- else
- QStyledItemDelegate::paint( painter, option, index );
+ }*/
}
+ painter->restore();
}
diff --git a/src/libtomahawk/playlist/playlistitemdelegate.h b/src/libtomahawk/playlist/playlistitemdelegate.h
index 5ba66bbd9..eabe03c1b 100644
--- a/src/libtomahawk/playlist/playlistitemdelegate.h
+++ b/src/libtomahawk/playlist/playlistitemdelegate.h
@@ -1,5 +1,5 @@
/* === This file is part of Tomahawk Player - ===
- *
+ *
* Copyright 2010-2011, Christian Muehlhaeuser
*
* Tomahawk is free software: you can redistribute it and/or modify
@@ -31,8 +31,12 @@ class DLLEXPORT PlaylistItemDelegate : public QStyledItemDelegate
Q_OBJECT
public:
+ enum PlaylistItemStyle
+ { Detailed = 0, Short = 1 };
+
PlaylistItemDelegate( TrackView* parent = 0, TrackProxyModel* proxy = 0 );
+ void setStyle( PlaylistItemDelegate::PlaylistItemStyle style );
void updateRowSize( const QModelIndex& index );
public slots:
@@ -41,13 +45,16 @@ public slots:
protected:
void paint( QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index ) const;
QSize sizeHint( const QStyleOptionViewItem& option, const QModelIndex& index ) const;
-
QWidget* createEditor( QWidget* parent, const QStyleOptionViewItem& option, const QModelIndex& index ) const;
private:
+ void paintDetailed( QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index ) const;
+ void paintShort( QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index ) const;
+
unsigned int m_removalProgress;
QPixmap m_nowPlayingIcon;
+ PlaylistItemStyle m_style;
TrackView* m_view;
TrackProxyModel* m_model;
};
diff --git a/src/libtomahawk/playlist/playlistmodel.cpp b/src/libtomahawk/playlist/playlistmodel.cpp
index 65c157d9c..d480db8e1 100644
--- a/src/libtomahawk/playlist/playlistmodel.cpp
+++ b/src/libtomahawk/playlist/playlistmodel.cpp
@@ -509,6 +509,7 @@ PlaylistModel::removeIndex( const QModelIndex& index, bool moreToCome )
}
}
+
bool
PlaylistModel::isTemporary() const
{
diff --git a/src/libtomahawk/playlist/playlistview.cpp b/src/libtomahawk/playlist/playlistview.cpp
index 8d59cb6d4..1a3b4084a 100644
--- a/src/libtomahawk/playlist/playlistview.cpp
+++ b/src/libtomahawk/playlist/playlistview.cpp
@@ -73,10 +73,8 @@ PlaylistView::setPlaylistModel( PlaylistModel* model )
else
{
setGuid( "playlistview" );
-
- m_model->title();
- m_model->description();
}
+
connect( m_model, SIGNAL( trackCountChanged( unsigned int ) ), SLOT( onTrackCountChanged( unsigned int ) ) );
connect( m_model, SIGNAL( playlistDeleted() ), SLOT( onDeleted() ) );
connect( m_model, SIGNAL( playlistChanged() ), SLOT( onChanged() ) );
@@ -148,12 +146,6 @@ PlaylistView::keyPressEvent( QKeyEvent* event )
}
-void
-PlaylistView::addItemsToPlaylist()
-{
-}
-
-
void
PlaylistView::deleteItems()
{
@@ -189,6 +181,7 @@ PlaylistView::onDeleted()
deleteLater();
}
+
void
PlaylistView::onChanged()
{
@@ -197,12 +190,9 @@ PlaylistView::onChanged()
emit nameChanged( m_model->playlist()->title() );
}
+
bool
PlaylistView::isTemporaryPage() const
{
- if ( m_model ) {
- return m_model->isTemporary();
- } else {
- return false;
- }
+ return ( m_model && m_model->isTemporary() );
}
diff --git a/src/libtomahawk/playlist/playlistview.h b/src/libtomahawk/playlist/playlistview.h
index c42ff1248..12bbd934b 100644
--- a/src/libtomahawk/playlist/playlistview.h
+++ b/src/libtomahawk/playlist/playlistview.h
@@ -65,7 +65,6 @@ private slots:
void onCustomContextMenu( const QPoint& pos );
void onTrackCountChanged( unsigned int tracks );
- void addItemsToPlaylist();
void deleteItems();
void onDeleted();
diff --git a/src/libtomahawk/playlist/trackheader.cpp b/src/libtomahawk/playlist/trackheader.cpp
index 750f207d2..962665d72 100644
--- a/src/libtomahawk/playlist/trackheader.cpp
+++ b/src/libtomahawk/playlist/trackheader.cpp
@@ -1,5 +1,5 @@
/* === This file is part of Tomahawk Player - ===
- *
+ *
* Copyright 2010-2011, Christian Muehlhaeuser
*
* Tomahawk is free software: you can redistribute it and/or modify
@@ -79,11 +79,15 @@ TrackHeader::checkState()
QByteArray state = TomahawkSettings::instance()->playlistColumnSizes( m_parent->guid() );
if ( !state.isEmpty() )
+ {
restoreState( state );
+ setSortIndicatorShown( true );
+ setSortIndicator( -1, Qt::AscendingOrder );
+ }
else
{
QList< double > m_columnWeights;
- m_columnWeights << 0.21 << 0.22 << 0.20 << 0.05 << 0.05 << 0.05 << 0.05 << 0.05; // << 0.12;
+ m_columnWeights << 0.20 << 0.20 << 0.18 << 0.05 << 0.05 << 0.05 << 0.05 << 0.05 << 0.12; // << 0.05;
for ( int i = 0; i < count() - 1; i++ )
{
diff --git a/src/libtomahawk/playlist/trackmodel.cpp b/src/libtomahawk/playlist/trackmodel.cpp
index 51ae0590a..79bff2a00 100644
--- a/src/libtomahawk/playlist/trackmodel.cpp
+++ b/src/libtomahawk/playlist/trackmodel.cpp
@@ -82,7 +82,7 @@ int
TrackModel::columnCount( const QModelIndex& parent ) const
{
Q_UNUSED( parent );
- return 9;
+ return 10;
}
@@ -183,6 +183,10 @@ TrackModel::data( const QModelIndex& index, int role ) const
case Origin:
return query->results().first()->friendlySource();
break;
+
+ case Score:
+ return query->results().first()->score();
+ break;
}
}
@@ -195,7 +199,7 @@ TrackModel::headerData( int section, Qt::Orientation orientation, int role ) con
{
Q_UNUSED( orientation );
QStringList headers;
- headers << tr( "Artist" ) << tr( "Track" ) << tr( "Album" ) << tr( "Duration" ) << tr( "Bitrate" ) << tr( "Age" ) << tr( "Year" ) << tr( "Size" ) << tr( "Origin" );
+ headers << tr( "Artist" ) << tr( "Track" ) << tr( "Album" ) << tr( "Duration" ) << tr( "Bitrate" ) << tr( "Age" ) << tr( "Year" ) << tr( "Size" ) << tr( "Origin" ) << tr( "Score" );
if ( role == Qt::DisplayRole && section >= 0 )
{
return headers.at( section );
diff --git a/src/libtomahawk/playlist/trackmodel.h b/src/libtomahawk/playlist/trackmodel.h
index 843e8f72a..5a935c8df 100644
--- a/src/libtomahawk/playlist/trackmodel.h
+++ b/src/libtomahawk/playlist/trackmodel.h
@@ -42,7 +42,8 @@ public:
Age = 5,
Year = 6,
Filesize = 7,
- Origin = 8
+ Origin = 8,
+ Score = 9
};
explicit TrackModel( QObject* parent = 0 );
diff --git a/src/libtomahawk/playlist/trackproxymodel.cpp b/src/libtomahawk/playlist/trackproxymodel.cpp
index dbc22c543..224f995cb 100644
--- a/src/libtomahawk/playlist/trackproxymodel.cpp
+++ b/src/libtomahawk/playlist/trackproxymodel.cpp
@@ -270,3 +270,111 @@ TrackProxyModel::removeIndexes( const QList& indexes )
sourceModel()->removeIndex( idx, b );
}
}
+
+
+bool
+TrackProxyModel::lessThan( const QModelIndex& left, const QModelIndex& right ) const
+{
+ qDebug() << Q_FUNC_INFO;
+
+ TrackModelItem* p1 = itemFromIndex( left );
+ TrackModelItem* p2 = itemFromIndex( right );
+
+ if ( !p1 )
+ return true;
+ if ( !p2 )
+ return false;
+
+ const Tomahawk::query_ptr& q1 = p1->query();
+ const Tomahawk::query_ptr& q2 = p2->query();
+
+ QString artist1 = q1->artist();
+ QString artist2 = q2->artist();
+ QString album1 = q1->album();
+ QString album2 = q2->album();
+ QString track1 = q1->track();
+ QString track2 = q2->track();
+ unsigned int albumpos1 = 0, albumpos2 = 0;
+ unsigned int bitrate1 = 0, bitrate2 = 0;
+ unsigned int mtime1 = 0, mtime2 = 0;
+ unsigned int id1 = 0, id2 = 0;
+ unsigned int size1 = 0, size2 = 0;
+
+ if ( q1->numResults() )
+ {
+ const Tomahawk::result_ptr& r = q1->results().at( 0 );
+ artist1 = r->artist()->name();
+ album1 = r->album()->name();
+ track1 = r->track();
+ albumpos1 = r->albumpos();
+ bitrate1 = r->bitrate();
+ mtime1 = r->modificationTime();
+ id1 = r->dbid();
+ size1 = r->size();
+ }
+ if ( q2->numResults() )
+ {
+ const Tomahawk::result_ptr& r = q2->results().at( 0 );
+ artist2 = r->artist()->name();
+ album2 = r->album()->name();
+ track2 = r->track();
+ albumpos2 = r->albumpos();
+ bitrate2 = r->bitrate();
+ mtime2 = r->modificationTime();
+ id2 = r->dbid();
+ size2 = r->size();
+ }
+
+ if ( left.column() == TrackModel::Artist ) // sort by artist
+ {
+ if ( artist1 == artist2 )
+ {
+ if ( album1 == album2 )
+ {
+ if ( albumpos1 == albumpos2 )
+ return id1 < id2;
+
+ return albumpos1 < albumpos2;
+ }
+
+ return QString::localeAwareCompare( album1, album2 ) < 0;
+ }
+
+ return QString::localeAwareCompare( artist1, artist2 ) < 0;
+ }
+ else if ( left.column() == TrackModel::Album ) // sort by album
+ {
+ if ( album1 == album2 )
+ {
+ if ( albumpos1 == albumpos2 )
+ return id1 < id2;
+
+ return albumpos1 < albumpos2;
+ }
+
+ return QString::localeAwareCompare( album1, album2 ) < 0;
+ }
+ else if ( left.column() == TrackModel::Bitrate ) // sort by bitrate
+ {
+ if ( bitrate1 == bitrate2 )
+ return id1 < id2;
+
+ return bitrate1 < bitrate2;
+ }
+ else if ( left.column() == TrackModel::Age ) // sort by mtime
+ {
+ if ( mtime1 == mtime2 )
+ return id1 < id2;
+
+ return mtime1 < mtime2;
+ }
+ else if ( left.column() == TrackModel::Filesize ) // sort by file size
+ {
+ if ( size1 == size2 )
+ return id1 < id2;
+
+ return size1 < size2;
+ }
+ return QString::localeAwareCompare( sourceModel()->data( left ).toString(),
+ sourceModel()->data( right ).toString() ) < 0;
+}
diff --git a/src/libtomahawk/playlist/trackproxymodel.h b/src/libtomahawk/playlist/trackproxymodel.h
index 853a512d8..ee8b65753 100644
--- a/src/libtomahawk/playlist/trackproxymodel.h
+++ b/src/libtomahawk/playlist/trackproxymodel.h
@@ -77,6 +77,7 @@ public slots:
protected:
bool filterAcceptsRow( int sourceRow, const QModelIndex& sourceParent ) const;
+ bool lessThan( const QModelIndex& left, const QModelIndex& right ) const;
private:
TrackModel* m_model;
diff --git a/src/libtomahawk/playlist/trackview.cpp b/src/libtomahawk/playlist/trackview.cpp
index daf3627f0..0ee1c8a8d 100644
--- a/src/libtomahawk/playlist/trackview.cpp
+++ b/src/libtomahawk/playlist/trackview.cpp
@@ -50,7 +50,6 @@ TrackView::TrackView( QWidget* parent )
, m_resizing( false )
, m_dragging( false )
{
- setSortingEnabled( false );
setAlternatingRowColors( true );
setSelectionMode( QAbstractItemView::ExtendedSelection );
setSelectionBehavior( QAbstractItemView::SelectRows );
@@ -63,9 +62,11 @@ TrackView::TrackView( QWidget* parent )
setRootIsDecorated( false );
setUniformRowHeights( true );
setMinimumWidth( 300 );
-// setVerticalScrollBarPolicy( Qt::ScrollBarAlwaysOn );
+ // setVerticalScrollBarPolicy( Qt::ScrollBarAlwaysOn );
setHeader( m_header );
+ setSortingEnabled( true );
+ sortByColumn( -1 );
#ifndef Q_WS_WIN
QFont f = font();
@@ -78,7 +79,7 @@ TrackView::TrackView( QWidget* parent )
setFont( f );
#endif
- QAction* createLinkAction = new QAction( tr( "Copy track link" ), this );
+ QAction* createLinkAction = new QAction( tr( "Copy Track Link" ), this );
connect( createLinkAction, SIGNAL( triggered( bool ) ), this, SLOT( copyLink() ) );
addAction( createLinkAction );
diff --git a/src/libtomahawk/utils/proxystyle.cpp b/src/libtomahawk/utils/proxystyle.cpp
index 038208a85..2e358751a 100644
--- a/src/libtomahawk/utils/proxystyle.cpp
+++ b/src/libtomahawk/utils/proxystyle.cpp
@@ -24,8 +24,8 @@
#include
#include
-#define ARROW_WIDTH 8
-#define ARROW_HEIGHT 8
+#define ARROW_WIDTH 7
+#define ARROW_HEIGHT 7
void
diff --git a/src/libtomahawk/utils/tomahawkutils.h b/src/libtomahawk/utils/tomahawkutils.h
index 331afa931..e4d461e99 100644
--- a/src/libtomahawk/utils/tomahawkutils.h
+++ b/src/libtomahawk/utils/tomahawkutils.h
@@ -24,6 +24,7 @@
#include
#include
#include
+#include
#define RESPATH ":/data/"
@@ -43,7 +44,7 @@ namespace TomahawkUtils
NetworkProxyFactory()
: m_proxy( QNetworkProxy::NoProxy )
{}
-
+
virtual ~NetworkProxyFactory() {}
virtual QList< QNetworkProxy > queryProxy( const QNetworkProxyQuery & query = QNetworkProxyQuery() );
diff --git a/src/libtomahawk/widgets/infowidgets/sourceinfowidget.cpp b/src/libtomahawk/widgets/infowidgets/sourceinfowidget.cpp
index 8ac75a8c6..a56fe386d 100644
--- a/src/libtomahawk/widgets/infowidgets/sourceinfowidget.cpp
+++ b/src/libtomahawk/widgets/infowidgets/sourceinfowidget.cpp
@@ -41,11 +41,11 @@ SourceInfoWidget::SourceInfoWidget( const Tomahawk::source_ptr& source, QWidget*
ui->historyView->overlay()->setEnabled( false );
m_recentCollectionModel = new CollectionFlatModel( ui->recentCollectionView );
- ui->recentCollectionView->setModel( m_recentCollectionModel );
+ ui->recentCollectionView->setTrackModel( m_recentCollectionModel );
m_recentCollectionModel->addFilteredCollection( source->collection(), 250, DatabaseCommand_AllTracks::ModificationTime );
m_historyModel = new PlaylistModel( ui->historyView );
- ui->historyView->setModel( m_historyModel );
+ ui->historyView->setPlaylistModel( m_historyModel );
m_historyModel->loadHistory( source );
connect( source.data(), SIGNAL( playbackFinished( Tomahawk::query_ptr ) ), SLOT( onPlaybackFinished( Tomahawk::query_ptr ) ) );
@@ -59,7 +59,7 @@ SourceInfoWidget::SourceInfoWidget( const Tomahawk::source_ptr& source, QWidget*
ui->historyView->setColumnHidden( TrackModel::Filesize, true );
m_recentAlbumModel = new AlbumModel( ui->recentAlbumView );
- ui->recentAlbumView->setModel( m_recentAlbumModel );
+ ui->recentAlbumView->setAlbumModel( m_recentAlbumModel );
m_recentAlbumModel->addFilteredCollection( source->collection(), 20, DatabaseCommand_AllAlbums::ModificationTime );
m_title = tr( "Info about %1" ).arg( source->isLocal() ? tr( "Your Collection" ) : source->friendlyName() );
diff --git a/src/libtomahawk/widgets/searchwidget.cpp b/src/libtomahawk/widgets/searchwidget.cpp
index 34350f4df..e4a734915 100644
--- a/src/libtomahawk/widgets/searchwidget.cpp
+++ b/src/libtomahawk/widgets/searchwidget.cpp
@@ -42,6 +42,7 @@ SearchWidget::SearchWidget( const QString& search, QWidget* parent )
m_resultsModel = new PlaylistModel( ui->resultsView );
ui->resultsView->setPlaylistModel( m_resultsModel );
ui->resultsView->overlay()->setEnabled( false );
+ ui->resultsView->sortByColumn( PlaylistModel::Score, Qt::DescendingOrder );
m_queries << Tomahawk::Query::get( search, uuid() );
diff --git a/src/resolvers/qtscriptresolver.h b/src/resolvers/qtscriptresolver.h
index ed90f8261..51d457c1b 100644
--- a/src/resolvers/qtscriptresolver.h
+++ b/src/resolvers/qtscriptresolver.h
@@ -22,6 +22,7 @@
#include "resolver.h"
#include "query.h"
#include "result.h"
+#include "utils/tomahawkutils.h"
#include
#include
@@ -41,6 +42,11 @@ public:
: QWebPage( (QObject*) parent )
, m_parent( parent )
{
+ settings()->setAttribute( QWebSettings::OfflineStorageDatabaseEnabled, true );
+ settings()->setOfflineStoragePath( TomahawkUtils::appDataDir().path() );
+ settings()->setAttribute(QWebSettings::LocalStorageEnabled, true);
+ settings()->setLocalStoragePath( TomahawkUtils::appDataDir().path() );
+ settings()->setAttribute( QWebSettings::LocalStorageDatabaseEnabled, true );
}
public slots:
diff --git a/src/resolvers/scriptresolver.cpp b/src/resolvers/scriptresolver.cpp
index c9f6b9f48..bac59fd16 100644
--- a/src/resolvers/scriptresolver.cpp
+++ b/src/resolvers/scriptresolver.cpp
@@ -39,7 +39,13 @@ ScriptResolver::ScriptResolver( const QString& exe )
connect( &m_proc, SIGNAL( readyReadStandardOutput() ), SLOT( readStdout() ) );
connect( &m_proc, SIGNAL( finished( int, QProcess::ExitStatus ) ), SLOT( cmdExited( int, QProcess::ExitStatus ) ) );
- m_proc.start( filePath() );
+ QString filepath = filePath();
+#ifdef WIN32
+ // have to enclose in quotes if path contains spaces on windows...
+ filepath = QString( "\"%1\"" ).arg( filepath );
+#endif
+
+ m_proc.start( filepath );
}
diff --git a/src/settingsdialog.cpp b/src/settingsdialog.cpp
index 8f881ac03..36393fb8e 100644
--- a/src/settingsdialog.cpp
+++ b/src/settingsdialog.cpp
@@ -136,11 +136,12 @@ SettingsDialog::SettingsDialog( QWidget *parent )
}
// NOW PLAYING
- #ifdef Q_WS_MAC
+#ifdef Q_WS_MAC
ui->checkBoxEnableAdium->setChecked( s->nowPlayingEnabled() );
- #else
+#else
+ ui->nowPlaying->hide();
ui->checkBoxEnableAdium->hide();
- #endif
+#endif
// LAST FM
ui->checkBoxEnableLastfm->setChecked( s->scrobblingEnabled() );
diff --git a/src/sip/jabber/jabber.cpp b/src/sip/jabber/jabber.cpp
index 29ee22a16..ae2b41e98 100644
--- a/src/sip/jabber/jabber.cpp
+++ b/src/sip/jabber/jabber.cpp
@@ -622,7 +622,8 @@ void JabberPlugin::onNewMessage(const Jreen::Message& message)
if( message.subtype() == Jreen::Message::Error )
{
- qDebug() << Q_FUNC_INFO << "Received error message from " << from << ", not answering... (Condition: " << message.error()->condition() << ")";
+ qDebug() << Q_FUNC_INFO << "Received error message from " << from << ", not answering... (Condition: "
+ << ( message.error().isNull() ? -1 : message.error()->condition() ) << ")";
return;
}
@@ -685,7 +686,7 @@ void JabberPlugin::onPresenceReceived( const Jreen::RosterItem::Ptr &item, const
Jreen::IQ featuresIq( Jreen::IQ::Get, jid );
featuresIq.addExtension( new Jreen::Disco::Info( node ) );
-
+
Jreen::IQReply *reply = m_client->send(featuresIq);
reply->setData(RequestDisco);
connect(reply, SIGNAL(received(Jreen::IQ)), SLOT(onNewIq(Jreen::IQ)));
diff --git a/src/sourcetree/items/collectionitem.cpp b/src/sourcetree/items/collectionitem.cpp
index 4aa76867b..984bb798c 100644
--- a/src/sourcetree/items/collectionitem.cpp
+++ b/src/sourcetree/items/collectionitem.cpp
@@ -33,13 +33,22 @@ CollectionItem::CollectionItem( SourcesModel* mdl, SourceTreeItem* parent, cons
, m_playlists( 0 )
, m_stations( 0 )
, m_tempItem( 0 )
+ , m_sourceInfoItem( 0 )
, m_curTempPage( 0 )
+ , m_sourceInfoPage( 0 )
{
if( m_source.isNull() ) { // super collection
connect( ViewManager::instance(), SIGNAL( tempPageActivated( Tomahawk::ViewPage*) ), this, SLOT( tempPageActivated( Tomahawk::ViewPage* ) ) );
return;
}
+
+ m_sourceInfoItem = new GenericPageItem( model(), this, tr( "New Additions" ), QIcon(),
+ boost::bind( &CollectionItem::sourceInfoClicked, this ),
+ boost::bind( &CollectionItem::getSourceInfoPage, this )
+ );
+ m_sourceInfoItem->setSortValue( -300 );
+
// create category items if there are playlists to show, or stations to show
QList< playlist_ptr > playlists = source->collection()->playlists();
QList< dynplaylist_ptr > autoplaylists = source->collection()->autoPlaylists();
@@ -71,18 +80,10 @@ CollectionItem::CollectionItem( SourcesModel* mdl, SourceTreeItem* parent, cons
connect( source->collection().data(), SIGNAL( playlistsAdded( QList ) ),
SLOT( onPlaylistsAdded( QList ) ), Qt::QueuedConnection );
- connect( source->collection().data(), SIGNAL( playlistsDeleted( QList ) ),
- SLOT( onPlaylistsDeleted( QList ) ), Qt::QueuedConnection );
-
connect( source->collection().data(), SIGNAL( autoPlaylistsAdded( QList< Tomahawk::dynplaylist_ptr > ) ),
SLOT( onAutoPlaylistsAdded( QList ) ), Qt::QueuedConnection );
- connect( source->collection().data(), SIGNAL( autoPlaylistsDeleted( QList ) ),
- SLOT( onAutoPlaylistsDeleted( QList ) ), Qt::QueuedConnection );
-
connect( source->collection().data(), SIGNAL( stationsAdded( QList ) ),
SLOT( onStationsAdded( QList ) ), Qt::QueuedConnection );
- connect( source->collection().data(), SIGNAL( stationsDeleted( QList ) ),
- SLOT( onStationsDeleted( QList ) ), Qt::QueuedConnection );
}
@@ -154,6 +155,14 @@ CollectionItem::playlistsAddedInternal( SourceTreeItem* parent, const QList< dyn
// qDebug() << "Dynamic Playlist added:" << p->title() << p->creator() << p->info();
p->loadRevision();
items << plItem;
+
+ if( p->mode() == Static ) {
+ connect( p.data(), SIGNAL( aboutToBeDeleted( Tomahawk::dynplaylist_ptr ) ),
+ SLOT( onAutoPlaylistDeleted( Tomahawk::dynplaylist_ptr ) ), Qt::QueuedConnection );
+ } else {
+ connect( p.data(), SIGNAL( aboutToBeDeleted( Tomahawk::dynplaylist_ptr ) ),
+ SLOT( onStationDeleted( Tomahawk::dynplaylist_ptr ) ), Qt::QueuedConnection );
+ }
}
parent->endRowsAdded();
}
@@ -161,20 +170,17 @@ CollectionItem::playlistsAddedInternal( SourceTreeItem* parent, const QList< dyn
template< typename T >
void
-CollectionItem::playlistsDeletedInternal( SourceTreeItem* parent, const QList< T >& playlists )
+CollectionItem::playlistDeletedInternal( SourceTreeItem* parent, const T& p )
{
Q_ASSERT( parent ); // How can we delete playlists if we have none?
- QList< SourceTreeItem* > items;
- foreach( const T& playlist, playlists ) {
- int curCount = parent->children().count();
- for( int i = 0; i < curCount; i++ ) {
- PlaylistItem* pl = qobject_cast< PlaylistItem* >( parent->children().at( i ) );
- if( pl && pl->playlist() == playlist ) {
- parent->beginRowsRemoved( i, i );
- parent->removeChild( pl );
- parent->endRowsRemoved();
- break;
- }
+ int curCount = parent->children().count();
+ for( int i = 0; i < curCount; i++ ) {
+ PlaylistItem* pl = qobject_cast< PlaylistItem* >( parent->children().at( i ) );
+ if( pl && pl->playlist() == p ) {
+ parent->beginRowsRemoved( i, i );
+ parent->removeChild( pl );
+ parent->endRowsRemoved();
+ break;
}
}
}
@@ -208,15 +214,19 @@ CollectionItem::onPlaylistsAdded( const QList< playlist_ptr >& playlists )
// qDebug() << "Playlist added:" << p->title() << p->creator() << p->info();
p->loadRevision();
items << plItem;
+
+ connect( p.data(), SIGNAL( aboutToBeDeleted( Tomahawk::playlist_ptr ) ),
+ SLOT( onPlaylistDeleted( Tomahawk::playlist_ptr ) ), Qt::QueuedConnection );
+
}
m_playlists->endRowsAdded();
}
void
-CollectionItem::onPlaylistsDeleted( const QList< playlist_ptr >& playlists )
+CollectionItem::onPlaylistDeleted( const playlist_ptr& playlist )
{
- playlistsDeletedInternal( m_playlists, playlists );
+ playlistDeletedInternal( m_playlists, playlist );
}
@@ -238,12 +248,12 @@ CollectionItem::onAutoPlaylistsAdded( const QList< dynplaylist_ptr >& playlists
void
-CollectionItem::onAutoPlaylistsDeleted( const QList< dynplaylist_ptr >& playlists )
+CollectionItem::onAutoPlaylistDeleted( const dynplaylist_ptr& playlist )
{
if( !m_playlists )
qDebug() << "NO playlist category item for a deleting playlist..";
- playlistsDeletedInternal( m_playlists, playlists );
+ playlistDeletedInternal( m_playlists, playlist );
}
@@ -265,9 +275,9 @@ CollectionItem::onStationsAdded( const QList< dynplaylist_ptr >& stations )
void
-CollectionItem::onStationsDeleted( const QList< dynplaylist_ptr >& stations )
+CollectionItem::onStationDeleted( const dynplaylist_ptr& station )
{
- playlistsDeletedInternal( m_stations, stations );
+ playlistDeletedInternal( m_stations, station );
}
void
@@ -306,3 +316,19 @@ CollectionItem::getTempPage() const
{
return m_curTempPage;
}
+
+ViewPage*
+CollectionItem::sourceInfoClicked()
+{
+ if( m_source.isNull() )
+ return 0;
+
+ m_sourceInfoPage = ViewManager::instance()->show( m_source );
+ return m_sourceInfoPage;
+}
+
+ViewPage*
+CollectionItem::getSourceInfoPage() const
+{
+ return m_sourceInfoPage;
+}
diff --git a/src/sourcetree/items/collectionitem.h b/src/sourcetree/items/collectionitem.h
index 8bb23bee7..90615dac0 100644
--- a/src/sourcetree/items/collectionitem.h
+++ b/src/sourcetree/items/collectionitem.h
@@ -45,27 +45,32 @@ public:
private slots:
void onPlaylistsAdded( const QList& playlists );
- void onPlaylistsDeleted( const QList& playlists );
+ void onPlaylistDeleted( const Tomahawk::playlist_ptr& playlists );
void onAutoPlaylistsAdded( const QList& playlists );
- void onAutoPlaylistsDeleted( const QList& playlists );
+ void onAutoPlaylistDeleted( const Tomahawk::dynplaylist_ptr& playlists );
void onStationsAdded( const QList& stations );
- void onStationsDeleted( const QList& stations );
+ void onStationDeleted( const Tomahawk::dynplaylist_ptr& stations );
void tempPageActivated( Tomahawk::ViewPage* );
Tomahawk::ViewPage* tempItemClicked();
Tomahawk::ViewPage* getTempPage() const;
+ Tomahawk::ViewPage* sourceInfoClicked();
+ Tomahawk::ViewPage* getSourceInfoPage() const;
+
private:
void playlistsAddedInternal( SourceTreeItem* parent, const QList< Tomahawk::dynplaylist_ptr >& playlists );
template< typename T >
- void playlistsDeletedInternal( SourceTreeItem* parent, const QList< T >& playlists );
+ void playlistDeletedInternal( SourceTreeItem* parent, const T& playlists );
Tomahawk::source_ptr m_source;
CategoryItem* m_playlists;
CategoryItem* m_stations;
GenericPageItem* m_tempItem;
+ GenericPageItem* m_sourceInfoItem;
Tomahawk::ViewPage* m_curTempPage;
+ Tomahawk::ViewPage* m_sourceInfoPage;
};
diff --git a/src/sourcetree/items/genericpageitems.cpp b/src/sourcetree/items/genericpageitems.cpp
index 3739fa7cb..9a2787b93 100644
--- a/src/sourcetree/items/genericpageitems.cpp
+++ b/src/sourcetree/items/genericpageitems.cpp
@@ -26,6 +26,7 @@ GenericPageItem::GenericPageItem( SourcesModel* model, SourceTreeItem* parent, c
: SourceTreeItem( model, parent, SourcesModel::GenericPage )
, m_icon( icon )
, m_text( text )
+ , m_sortValue( 0 )
, m_show( show )
, m_get( get )
{
diff --git a/src/sourcetree/items/genericpageitems.h b/src/sourcetree/items/genericpageitems.h
index 5a980f7c3..378224bb1 100644
--- a/src/sourcetree/items/genericpageitems.h
+++ b/src/sourcetree/items/genericpageitems.h
@@ -35,14 +35,17 @@ public:
virtual void activate();
virtual bool willAcceptDrag( const QMimeData* data ) const;
virtual QIcon icon() const;
+ virtual int peerSortValue() const { return m_sortValue; } // How to sort relative to peers in the tree.
void setText( const QString& text );
+ void setSortValue( int value ) { m_sortValue = value; }
signals:
void activated();
private:
QIcon m_icon;
QString m_text;
+ int m_sortValue;
boost::function< Tomahawk::ViewPage*() > m_show;
boost::function< Tomahawk::ViewPage*() > m_get;
};
diff --git a/src/sourcetree/sourcetreeview.cpp b/src/sourcetree/sourcetreeview.cpp
index 8b21bb176..f299dc0b4 100644
--- a/src/sourcetree/sourcetreeview.cpp
+++ b/src/sourcetree/sourcetreeview.cpp
@@ -38,6 +38,7 @@
using namespace Tomahawk;
+#define TREEVIEW_INDENT_ADD -7
class SourceDelegate : public QStyledItemDelegate
{
@@ -53,6 +54,8 @@ protected:
editor->setGeometry( option.rect.adjusted( 20, 0, 0, 0 ) );
else
QStyledItemDelegate::updateEditorGeometry( editor, option, index );
+
+ editor->setGeometry( editor->geometry().adjusted( 2*TREEVIEW_INDENT_ADD, 0, 0, 0 ) );
}
private:
@@ -79,7 +82,7 @@ SourceTreeView::SourceTreeView( QWidget* parent )
setDropIndicatorShown( false );
setAllColumnsShowFocus( true );
setUniformRowHeights( false );
- setIndentation( 16 );
+ setIndentation( 14 );
setSortingEnabled( true );
sortByColumn( 0, Qt::AscendingOrder );
@@ -595,6 +598,23 @@ SourceDelegate::paint( QPainter* painter, const QStyleOptionViewItem& option, co
else
{
QStyledItemDelegate::paint( painter, o, index );
+ /*QStyleOptionViewItemV4 opt = o;
+ initStyleOption( &opt, index );
+
+ // shrink the indentations. count how indented this item is and remove it
+ int indentMult = 0;
+ QModelIndex counter = index;
+ while ( counter.parent().isValid() )
+ {
+ indentMult++;
+ counter = counter.parent();
+ }
+ int realX = opt.rect.x() + indentMult * TREEVIEW_INDENT_ADD;
+
+ opt.rect.setX( realX );
+ const QWidget *widget = opt.widget;
+ QStyle *style = widget ? widget->style() : QApplication::style();
+ style->drawControl( QStyle::CE_ItemViewItem, &opt, painter, widget ); */
}
#ifdef Q_WS_MAC
diff --git a/src/tomahawkwindow.cpp b/src/tomahawkwindow.cpp
index 041390fd5..cc8d758a5 100644
--- a/src/tomahawkwindow.cpp
+++ b/src/tomahawkwindow.cpp
@@ -82,7 +82,12 @@ TomahawkWindow::TomahawkWindow( QWidget* parent )
, m_trayIcon( new TomahawkTrayIcon( this ) )
, m_sourcetree( 0 )
{
- qApp->setStyle( new ProxyStyle() );
+ // HACK QtCurve causes an infinite loop on startup. This is because setStyle calls setPalette, which calls ensureBaseStyle,
+ // which loads QtCurve. QtCurve calls setPalette, which creates an infinite loop. The UI will look like CRAP with QtCurve, but
+ // the user is asking for it explicitly... so he's gonna be stuck with an ugly UI.
+ if ( !QString( qApp->style()->metaObject()->className() ).toLower().contains( "qtcurve" ) )
+ qApp->setStyle( new ProxyStyle() );
+
setWindowIcon( QIcon( RESPATH "icons/tomahawk-icon-128x128.png" ) );
#ifdef Q_WS_MAC
@@ -294,6 +299,16 @@ TomahawkWindow::setupSignals()
connect( ui->actionCreate_New_Station, SIGNAL( triggered() ), SLOT( createStation() ));
connect( ui->actionAboutTomahawk, SIGNAL( triggered() ), SLOT( showAboutTomahawk() ) );
connect( ui->actionExit, SIGNAL( triggered() ), qApp, SLOT( quit() ) );
+
+ connect( ui->actionPlay, SIGNAL( triggered() ), AudioEngine::instance(), SLOT( playPause() ) );
+ connect( ui->actionNext, SIGNAL( triggered() ), AudioEngine::instance(), SLOT( previous() ) );
+ connect( ui->actionPrevious, SIGNAL( triggered() ), AudioEngine::instance(), SLOT( next() ) );
+
+ connect( AudioEngine::instance(), SIGNAL( started( Tomahawk::result_ptr ) ), this, SLOT( audioStarted() ) );
+ connect( AudioEngine::instance(), SIGNAL( resumed()), this, SLOT( audioStarted() ) );
+ connect( AudioEngine::instance(), SIGNAL( paused() ), this, SLOT( audioStopped() ) );
+ connect( AudioEngine::instance(), SIGNAL( stopped() ), this, SLOT( audioStopped() ) );
+
#if defined( Q_OS_DARWIN )
connect( ui->actionMinimize, SIGNAL( triggered() ), SLOT( minimize() ) );
connect( ui->actionZoom, SIGNAL( triggered() ), SLOT( maximize() ) );
@@ -528,6 +543,19 @@ TomahawkWindow::createPlaylist()
}
}
+void
+TomahawkWindow::audioStarted()
+{
+ ui->actionPlay->setText( tr( "Pause" ) );
+}
+
+void
+TomahawkWindow::audioStopped()
+{
+
+ ui->actionPlay->setText( tr( "Play" ) );
+}
+
void
TomahawkWindow::onPlaybackLoading( const Tomahawk::result_ptr& result )
diff --git a/src/tomahawkwindow.h b/src/tomahawkwindow.h
index 88f83a658..69992eab1 100644
--- a/src/tomahawkwindow.h
+++ b/src/tomahawkwindow.h
@@ -50,7 +50,6 @@ public:
~TomahawkWindow();
AudioControls* audioControls() { return m_audioControls; }
- QStackedWidget* playlistStack();
SourceTreeView* sourceTreeView() const { return m_sourcetree; }
void setWindowTitle( const QString& title );
@@ -84,6 +83,9 @@ private slots:
void onHistoryBackAvailable( bool avail );
void onHistoryForwardAvailable( bool avail );
+ void audioStarted();
+ void audioStopped();
+
void showAboutTomahawk();
void checkForUpdates();
diff --git a/src/tomahawkwindow.ui b/src/tomahawkwindow.ui
index 48024e480..0f506a40c 100644
--- a/src/tomahawkwindow.ui
+++ b/src/tomahawkwindow.ui
@@ -35,7 +35,7 @@
0
0
1000
- 21
+ 20
diff --git a/src/web/api_v1.cpp b/src/web/api_v1.cpp
index eba9823bb..5ae2ae149 100644
--- a/src/web/api_v1.cpp
+++ b/src/web/api_v1.cpp
@@ -180,7 +180,7 @@ Api_v1::send404( QxtWebRequestEvent* event )
qDebug() << "404" << event->url.toString();
QxtWebPageEvent* wpe = new QxtWebPageEvent( event->sessionID, event->requestID, "Not Found
" );
wpe->status = 404;
- wpe->statusMessage = "not feventound";
+ wpe->statusMessage = "no event found";
postEvent( wpe );
}
diff --git a/thirdparty/liblastfm2/src/ws/ws.cpp b/thirdparty/liblastfm2/src/ws/ws.cpp
index dffecbd46..93d06b929 100644
--- a/thirdparty/liblastfm2/src/ws/ws.cpp
+++ b/thirdparty/liblastfm2/src/ws/ws.cpp
@@ -1,5 +1,5 @@
/*
- Copyright 2009 Last.fm Ltd.
+ Copyright 2009 Last.fm Ltd.
- Primarily authored by Max Howell, Jono Cole and Doug Mansell
This file is part of liblastfm.
@@ -33,7 +33,7 @@ static QMap< QThread*, QNetworkAccessManager* > threadNamHash;
static QSet< QThread* > ourNamSet;
static QMutex namAccessMutex;
-QString
+QString
lastfm::ws::host()
{
QStringList const args = QCoreApplication::arguments();
@@ -57,8 +57,8 @@ static QUrl url()
}
static QString iso639()
-{
- return QLocale().name().left( 2 ).toLower();
+{
+ return QLocale().name().left( 2 ).toLower();
}
void autograph( QMap& params )
@@ -99,17 +99,17 @@ lastfm::ws::get( QMap params )
QByteArray const value = QUrl::toPercentEncoding( i.value() );
url.addEncodedQueryItem( key, value );
}
-
+
qDebug() << url;
-
+
return nam()->get( QNetworkRequest(url) );
}
QNetworkReply*
lastfm::ws::post( QMap params, bool sk )
-{
- sign( params, sk );
+{
+ sign( params, sk );
QByteArray query;
QMapIterator i( params );
while (i.hasNext()) {
@@ -120,7 +120,9 @@ lastfm::ws::post( QMap params, bool sk )
+ '&';
}
- return nam()->post( QNetworkRequest(url()), query );
+ QNetworkRequest req(url());
+ req.setHeader( QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded" );
+ return nam()->post( req, query );
}
@@ -133,7 +135,7 @@ lastfm::ws::parse( QNetworkReply* reply ) throw( ParseError )
if (!data.size())
throw MalformedResponse;
-
+
QDomDocument xml;
xml.setContent( data );
QDomElement lfm = xml.documentElement();
@@ -223,7 +225,7 @@ lastfm::setNetworkAccessManager( QNetworkAccessManager* nam )
if ( oldNam == nam )
return;
-
+
threadNamHash[thread] = nam;
ourNamSet.remove( thread );
@@ -286,7 +288,7 @@ namespace lastfm
const char* ApiKey;
/** if this is found set to "" we conjure ourselves a suitable one */
- const char* UserAgent = 0;
+ const char* UserAgent = 0;
}
}
diff --git a/thirdparty/qxt/qxtweb-standalone/qxtweb/qxthttpsessionmanager.cpp b/thirdparty/qxt/qxtweb-standalone/qxtweb/qxthttpsessionmanager.cpp
index ce9ccc907..c444018e3 100644
--- a/thirdparty/qxt/qxtweb-standalone/qxtweb/qxthttpsessionmanager.cpp
+++ b/thirdparty/qxt/qxtweb-standalone/qxtweb/qxthttpsessionmanager.cpp
@@ -682,6 +682,9 @@ void QxtHttpSessionManager::sendNextBlock(int requestID)
{
QSharedPointer& dataSource = connector()->getRequestDataSource( requestID );
QIODevice* device = connector()->getRequestConnection(requestID);
+ if (!device)
+ return;
+
if (!qxt_d().connectionState.contains(device)) return; // in case a disconnect signal and a bytesWritten signal get fired in the wrong order
QxtHttpSessionManagerPrivate::ConnectionState& state = qxt_d().connectionState[device];
if (state.finishedTransfer) return;
diff --git a/thirdparty/qxt/qxtweb-standalone/qxtweb/qxtmetaobject.cpp b/thirdparty/qxt/qxtweb-standalone/qxtweb/qxtmetaobject.cpp
index cad4f038f..e55fd3615 100644
--- a/thirdparty/qxt/qxtweb-standalone/qxtweb/qxtmetaobject.cpp
+++ b/thirdparty/qxt/qxtweb-standalone/qxtweb/qxtmetaobject.cpp
@@ -323,6 +323,11 @@ namespace QxtMetaObject
*/
bool connect(QObject* sender, const char* signal, QxtBoundFunction* slot, Qt::ConnectionType type)
{
+ if (!sender)
+ {
+ qWarning() << "Got connect() with a null sender!";
+ return false;
+ }
const QMetaObject* meta = sender->metaObject();
int methodID = meta->indexOfMethod(meta->normalizedSignature(signal).mid(1).constData());
if (methodID < 0)