/* This file is part of Clementine. Copyright 2010, David Sansome Clementine is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Clementine is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Clementine. If not, see . */ #include "internetmodel.h" #include #include #include "digitallyimportedservicebase.h" #include "groovesharkservice.h" #include "icecastservice.h" #include "internetmimedata.h" #include "internetservice.h" #include "jamendoservice.h" #include "magnatuneservice.h" #include "savedradio.h" #include "somafmservice.h" #include "soundcloudservice.h" #include "spotifyservice.h" #include "subsonicservice.h" #include "core/closure.h" #include "core/logging.h" #include "core/mergedproxymodel.h" #include "podcasts/podcastservice.h" #include "smartplaylists/generatormimedata.h" #ifdef HAVE_LIBLASTFM #include "lastfmservice.h" #endif #ifdef HAVE_GOOGLE_DRIVE #include "googledriveservice.h" #endif #ifdef HAVE_UBUNTU_ONE #include "ubuntuoneservice.h" #endif #ifdef HAVE_DROPBOX #include "dropboxservice.h" #endif #ifdef HAVE_SKYDRIVE #include "skydriveservice.h" #endif #ifdef HAVE_BOX #include "boxservice.h" #endif using smart_playlists::Generator; using smart_playlists::GeneratorMimeData; using smart_playlists::GeneratorPtr; QMap* InternetModel::sServices = nullptr; InternetModel::InternetModel(Application* app, QObject* parent) : QStandardItemModel(parent), app_(app), merged_model_(new MergedProxyModel(this)) { if (!sServices) { sServices = new QMap; } Q_ASSERT(sServices->isEmpty()); merged_model_->setSourceModel(this); AddService(new DigitallyImportedService(app, this)); AddService(new IcecastService(app, this)); AddService(new JamendoService(app, this)); #ifdef HAVE_LIBLASTFM AddService(new LastFMService(app, this)); #endif #ifdef HAVE_GOOGLE_DRIVE AddService(new GoogleDriveService(app, this)); #endif AddService(new GroovesharkService(app, this)); AddService(new JazzRadioService(app, this)); AddService(new MagnatuneService(app, this)); AddService(new PodcastService(app, this)); AddService(new RadioGFMService(app, this)); AddService(new RockRadioService(app, this)); AddService(new SavedRadio(app, this)); AddService(new SkyFmService(app, this)); AddService(new SomaFMService(app, this)); AddService(new SoundCloudService(app, this)); AddService(new SpotifyService(app, this)); AddService(new SubsonicService(app, this)); #ifdef HAVE_UBUNTU_ONE AddService(new UbuntuOneService(app, this)); #endif #ifdef HAVE_DROPBOX AddService(new DropboxService(app, this)); #endif #ifdef HAVE_SKYDRIVE AddService(new SkydriveService(app, this)); #endif #ifdef HAVE_BOX AddService(new BoxService(app, this)); #endif } void InternetModel::AddService(InternetService* service) { QStandardItem* root = service->CreateRootItem(); if (!root) { qLog(Warning) << "Internet service" << service->name() << "did not return a root item"; return; } root->setData(Type_Service, Role_Type); root->setData(QVariant::fromValue(service), Role_Service); invisibleRootItem()->appendRow(root); qLog(Debug) << "Adding internet service:" << service->name(); sServices->insert(service->name(), service); connect(service, SIGNAL(StreamError(QString)), SIGNAL(StreamError(QString))); connect(service, SIGNAL(StreamMetadataFound(QUrl, Song)), SIGNAL(StreamMetadataFound(QUrl, Song))); connect(service, SIGNAL(AddToPlaylistSignal(QMimeData*)), SIGNAL(AddToPlaylist(QMimeData*))); connect(service, SIGNAL(ScrollToIndex(QModelIndex)), SIGNAL(ScrollToIndex(QModelIndex))); connect(service, SIGNAL(destroyed()), SLOT(ServiceDeleted())); service->ReloadSettings(); } void InternetModel::RemoveService(InternetService* service) { if (!sServices->contains(service->name())) return; // Find and remove the root item that this service created for (int i = 0; i < invisibleRootItem()->rowCount(); ++i) { QStandardItem* item = invisibleRootItem()->child(i); if (!item || item->data(Role_Service).value() == service) { invisibleRootItem()->removeRow(i); break; } } // Remove the service from the list sServices->remove(service->name()); // Disconnect the service disconnect(service, 0, this, 0); } void InternetModel::ServiceDeleted() { InternetService* service = qobject_cast(sender()); if (service) RemoveService(service); } InternetService* InternetModel::ServiceByName(const QString& name) { if (sServices->contains(name)) return sServices->value(name); return nullptr; } InternetService* InternetModel::ServiceForItem(const QStandardItem* item) const { return ServiceForIndex(indexFromItem(item)); } InternetService* InternetModel::ServiceForIndex(const QModelIndex& index) const { QModelIndex current_index = index; while (current_index.isValid()) { InternetService* service = current_index.data(Role_Service).value(); if (service) { return service; } current_index = current_index.parent(); } return nullptr; } Qt::ItemFlags InternetModel::flags(const QModelIndex& index) const { Qt::ItemFlags flags = Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsDropEnabled; if (IsPlayable(index)) { flags |= Qt::ItemIsDragEnabled; } return flags; } bool InternetModel::hasChildren(const QModelIndex& parent) const { if (parent.data(Role_CanLazyLoad).toBool()) return true; return QStandardItemModel::hasChildren(parent); } int InternetModel::rowCount(const QModelIndex& parent) const { if (parent.data(Role_CanLazyLoad).toBool()) { QStandardItem* item = itemFromIndex(parent); InternetService* service = ServiceForItem(item); if (service) { item->setData(false, Role_CanLazyLoad); service->LazyPopulate(item); } } return QStandardItemModel::rowCount(parent); } bool InternetModel::IsPlayable(const QModelIndex& index) const { QVariant behaviour = index.data(Role_PlayBehaviour); if (!behaviour.isValid()) return false; PlayBehaviour pb = PlayBehaviour(behaviour.toInt()); return (pb == PlayBehaviour_MultipleItems || pb == PlayBehaviour_SingleItem || pb == PlayBehaviour_UseSongLoader); } QStringList InternetModel::mimeTypes() const { return QStringList() << "text/uri-list"; } QMimeData* InternetModel::mimeData(const QModelIndexList& indexes) const { // Special case for when the user double clicked on a special item. if (indexes.count() == 1 && indexes[0].data(Role_PlayBehaviour).toInt() == PlayBehaviour_DoubleClickAction) { InternetModel::ServiceForIndex(indexes[0]) ->ItemDoubleClicked(itemFromIndex(indexes[0])); return nullptr; } if (indexes.count() == 1 && indexes[0].data(Role_Type).toInt() == Type_SmartPlaylist) { GeneratorPtr generator = InternetModel::ServiceForIndex(indexes[0]) ->CreateGenerator(itemFromIndex(indexes[0])); if (!generator) return nullptr; GeneratorMimeData* data = new GeneratorMimeData(generator); data->setData(LibraryModel::kSmartPlaylistsMimeType, QByteArray()); data->name_for_new_playlist_ = this->data(indexes.first()).toString(); return data; } QList urls; QModelIndexList new_indexes; QModelIndex last_valid_index; for (const QModelIndex& index : indexes) { if (!IsPlayable(index)) continue; last_valid_index = index; if (index.data(Role_PlayBehaviour).toInt() == PlayBehaviour_MultipleItems) { // Get children int row = 0; int column = 0; QModelIndex child = index.child(row, column); while (child.isValid()) { new_indexes << child; urls << child.data(Role_Url).toUrl(); child = index.child(++row, column); } } else { new_indexes = indexes; urls << index.data(Role_Url).toUrl(); } } if (urls.isEmpty()) return nullptr; InternetMimeData* data = new InternetMimeData(this); data->setUrls(urls); data->indexes = new_indexes; data->name_for_new_playlist_ = InternetModel::ServiceForIndex(last_valid_index)->name(); return data; } bool InternetModel::dropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) { if (action == Qt::IgnoreAction) { return false; } if (parent.data(Role_CanBeModified).toBool()) { InternetModel::ServiceForIndex(parent)->DropMimeData(data, parent); } return true; } void InternetModel::ShowContextMenu( const QModelIndexList& selected_merged_model_indexes, const QModelIndex& current_merged_model_index, const QPoint& global_pos) { current_index_ = merged_model_->mapToSource(current_merged_model_index); selected_indexes_.clear(); for (const QModelIndex& index : selected_merged_model_indexes) { selected_indexes_ << merged_model_->mapToSource(index); } InternetService* service = ServiceForIndex(current_merged_model_index); if (service) service->ShowContextMenu(global_pos); } void InternetModel::ReloadSettings() { for (InternetService* service : sServices->values()) { service->ReloadSettings(); } }