diff --git a/src/libtomahawk/resolvers/JSInfoPlugin.cpp b/src/libtomahawk/resolvers/JSInfoPlugin.cpp index 0aeaf9340..672b4637f 100644 --- a/src/libtomahawk/resolvers/JSInfoPlugin.cpp +++ b/src/libtomahawk/resolvers/JSInfoPlugin.cpp @@ -29,9 +29,9 @@ JSInfoPlugin::JSInfoPlugin( int id, JSResolver *resolver ) { Q_ASSERT( resolver ); - // read in supported GetTypes and PushTypes - m_supportedGetTypes = parseSupportedTypes( callMethodOnInfoPlugin( "supportedGetTypes" ) ); - m_supportedPushTypes = parseSupportedTypes( callMethodOnInfoPlugin( "supportedPushTypes" ) ); + // read in supported GetTypes and PushTypes - we can do this safely because we are still in WebKit thread here + m_supportedGetTypes = parseSupportedTypes( callMethodOnInfoPluginWithResult( "supportedGetTypes" ) ); + m_supportedPushTypes = parseSupportedTypes( callMethodOnInfoPluginWithResult( "supportedPushTypes" ) ); setFriendlyName( QString( "JSInfoPlugin: %1" ).arg( resolver->name() ) ); } @@ -104,6 +104,12 @@ JSInfoPlugin::addInfoRequestResult( int requestId, qint64 maxAge, const QVariant { Q_D( JSInfoPlugin ); + if ( QThread::currentThread() != thread() ) + { + QMetaObject::invokeMethod( this, "addInfoRequestResult", Qt::QueuedConnection, Q_ARG( int, requestId ), Q_ARG( qint64, maxAge ), Q_ARG( QVariantMap, returnedData ) ); + return; + } + // retrieve requestData from cache and delete it Tomahawk::InfoSystem::InfoRequestData requestData = d->requestDataCache[ requestId ]; d->requestDataCache.remove( requestId ); @@ -123,6 +129,12 @@ JSInfoPlugin::emitGetCachedInfo( int requestId, const QVariantMap& criteria, int { Q_D( JSInfoPlugin ); + if ( QThread::currentThread() != thread() ) + { + QMetaObject::invokeMethod( this, "emitGetCachedInfo", Qt::QueuedConnection, Q_ARG( int, requestId ), Q_ARG( QVariantMap, criteria ), Q_ARG( int, newMaxAge ) ); + return; + } + emit getCachedInfo( convertQVariantMapToInfoStringHash( criteria ), newMaxAge, d->requestDataCache[ requestId ]); } @@ -132,6 +144,12 @@ JSInfoPlugin::emitInfo( int requestId, const QVariantMap& output ) { Q_D( JSInfoPlugin ); + if ( QThread::currentThread() != thread() ) + { + QMetaObject::invokeMethod( this, "emitInfo", Qt::QueuedConnection, Q_ARG( int, requestId ), Q_ARG( QVariantMap, output ) ); + return; + } + emit info( d->requestDataCache[ requestId ], output ); } @@ -144,8 +162,8 @@ JSInfoPlugin::serviceGetter() const return QString( "Tomahawk.InfoSystem.getInfoPlugin(%1)" ).arg( d->id ); } - -QVariant +// TODO: DRY, really move things into base class +void JSInfoPlugin::callMethodOnInfoPlugin( const QString& scriptSource ) { Q_D( JSInfoPlugin ); @@ -154,10 +172,24 @@ JSInfoPlugin::callMethodOnInfoPlugin( const QString& scriptSource ) tLog() << Q_FUNC_INFO << eval; - return d->resolver->evaluateJavaScript( eval ); + d->resolver->evaluateJavaScript( eval ); } +QVariant +JSInfoPlugin::callMethodOnInfoPluginWithResult(const QString& scriptSource) +{ + Q_D( JSInfoPlugin ); + + QString eval = QString( "%1.%2" ).arg( serviceGetter() ).arg( scriptSource ); + + tLog() << Q_FUNC_INFO << eval; + + return d->resolver->evaluateJavaScriptWithResult( eval ); +} + + + QSet< Tomahawk::InfoSystem::InfoType > JSInfoPlugin::parseSupportedTypes( const QVariant& variant ) { diff --git a/src/libtomahawk/resolvers/JSInfoPlugin.h b/src/libtomahawk/resolvers/JSInfoPlugin.h index 5087993ad..bb399d288 100644 --- a/src/libtomahawk/resolvers/JSInfoPlugin.h +++ b/src/libtomahawk/resolvers/JSInfoPlugin.h @@ -40,9 +40,9 @@ public: virtual ~JSInfoPlugin(); - void addInfoRequestResult( int requestId, qint64 maxAge, const QVariantMap& returnedData ); - void emitGetCachedInfo( int requestId, const QVariantMap& criteria, int newMaxAge ); - void emitInfo( int requestId, const QVariantMap& output ); + Q_INVOKABLE void addInfoRequestResult( int requestId, qint64 maxAge, const QVariantMap& returnedData ); + Q_INVOKABLE void emitGetCachedInfo( int requestId, const QVariantMap& criteria, int newMaxAge ); + Q_INVOKABLE void emitInfo( int requestId, const QVariantMap& output ); protected slots: void init() override; @@ -54,7 +54,8 @@ protected slots: protected: // TODO: create JSPlugin base class and move these methods there QString serviceGetter() const; // = 0 - QVariant callMethodOnInfoPlugin( const QString& scriptSource ); + void callMethodOnInfoPlugin( const QString& scriptSource ); + QVariant callMethodOnInfoPluginWithResult( const QString& scriptSource ); private: static QSet< Tomahawk::InfoSystem::InfoType > parseSupportedTypes(const QVariant& variant); diff --git a/src/libtomahawk/resolvers/JSResolver.cpp b/src/libtomahawk/resolvers/JSResolver.cpp index 431be5aa0..1c40aa84e 100644 --- a/src/libtomahawk/resolvers/JSResolver.cpp +++ b/src/libtomahawk/resolvers/JSResolver.cpp @@ -478,7 +478,7 @@ JSResolver::lookupUrl( const QString& url ) QVariant -JSResolver::evaluateJavaScript ( const QString& scriptSource ) +JSResolver::evaluateJavaScriptInternal(const QString& scriptSource) { Q_D( JSResolver ); @@ -486,6 +486,30 @@ JSResolver::evaluateJavaScript ( const QString& scriptSource ) } +void +JSResolver::evaluateJavaScript( const QString& scriptSource ) +{ + Q_D( JSResolver ); + + if ( QThread::currentThread() != thread() ) + { + QMetaObject::invokeMethod( this, "evaluateJavaScript", Qt::QueuedConnection, Q_ARG( QString, scriptSource ) ); + return; + } + + evaluateJavaScriptInternal( scriptSource ); +} + + +QVariant +JSResolver::evaluateJavaScriptWithResult( const QString& scriptSource ) +{ + Q_ASSERT( QThread::currentThread() == thread() ); + + return evaluateJavaScriptInternal( scriptSource ); +} + + Tomahawk::ExternalResolver::ErrorState JSResolver::error() const diff --git a/src/libtomahawk/resolvers/JSResolver.h b/src/libtomahawk/resolvers/JSResolver.h index 0105bba24..f2163e3d7 100644 --- a/src/libtomahawk/resolvers/JSResolver.h +++ b/src/libtomahawk/resolvers/JSResolver.h @@ -61,7 +61,16 @@ public: bool canParseUrl( const QString& url, UrlType type ) override; - QVariant evaluateJavaScript( const QString& scriptSource ); + /** + * Evaluate JavaScript on the WebKit thread + */ + Q_INVOKABLE void evaluateJavaScript( const QString& scriptSource ); + + /** + * This method must be called from the WebKit thread + */ + QVariant evaluateJavaScriptWithResult( const QString& scriptSource ); + public slots: void resolve( const Tomahawk::query_ptr& query ) override; @@ -97,6 +106,11 @@ private: void loadScript( const QString& path ); void loadScripts( const QStringList& paths ); + /** + * Wrap the pure evaluateJavaScript call in here, while the threadings guards are in public methods + */ + QVariant evaluateJavaScriptInternal( const QString& scriptSource ); + // encapsulate javascript calls QVariantMap resolverSettings(); QVariantMap resolverUserConfig();