/* * Cantata * * Copyright (c) 2011-2014 Craig Drummond * * ---- * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #ifndef STREAMSMODEL_H #define STREAMSMODEL_H #include "actionmodel.h" #include "config.h" #ifdef ENABLE_STREAMS #include "mpd/stream.h" #include "mpd/playlist.h" #include #include #include #include #include class NetworkJob; class QNetworkRequest; class QXmlStreamReader; class QIODevice; class StreamsModel : public ActionModel { Q_OBJECT public: struct CategoryItem; struct Item : public Stream { Item(const QString &u, const QString &n=QString(), CategoryItem *p=0, const QString &sub=QString()) : Stream(u, n), subText(sub), parent(p) { } virtual ~Item() { } QString modifiedName() const; QString subText; CategoryItem *parent; virtual bool isCategory() const { return false; } CategoryItem *getTopLevelCategory() const; }; struct CategoryItem : public Item { enum State { Initial, Fetching, Fetched }; CategoryItem(const QString &u, const QString &n=QString(), CategoryItem *p=0, const QIcon &i=QIcon(), const QString &cn=QString(), const QString &bn=QString(), bool modName=false) : Item(u, n, p), state(Initial), isAll(false), isBookmarks(false), supportsBookmarks(false), canBookmark(false), addCatToModifiedName(modName), icon(i), cacheName(cn), bookmarksName(bn), configName(cn.isEmpty() ? bn : cn) { } virtual ~CategoryItem() { qDeleteAll(children); } virtual bool isCategory() const { return true; } virtual bool canConfigure() const { return false; } virtual bool isFavourites() const { return false; } virtual bool isBuiltIn() const { return true; } virtual bool isDi() const { return false; } virtual bool isSoma() const { return false; } virtual bool isListenLive() const { return false; } virtual void removeCache(); bool isTopLevel() const { return parent && 0==parent->parent; } virtual bool canReload() const { return !cacheName.isEmpty() || isTopLevel() || !url.isEmpty(); } void removeBookmarks(); void saveBookmarks(); QList loadBookmarks(); CategoryItem * getBookmarksCategory(); CategoryItem * createBookmarksCategory(); void saveCache() const; virtual QList loadCache(); bool saveXml(const QString &fileName, bool format=false) const; bool saveXml(QIODevice *dev, bool format=false) const; QList loadXml(const QString &fileName); virtual QList loadXml(QIODevice *dev); virtual void addHeaders(QNetworkRequest &) { } virtual NetworkJob * fetchSecondardyUrl() { return 0; } State state; bool isAll : 1; bool isBookmarks : 1; // 'Virtual' bookmarks category... bool supportsBookmarks : 1; // Intended for top-level items, indicates if bookmarks can be added bool canBookmark : 1; // Can this category be bookmark'ed in top-level parent? can have the cache bool addCatToModifiedName : 1; // When adding to playqueue/favourites, should name contian category? QList children; QIcon icon; QString cacheName; QString bookmarksName; QString configName; }; struct FavouritesCategoryItem : public CategoryItem { FavouritesCategoryItem(const QString &u, const QString &n, CategoryItem *p, const QIcon &i) : CategoryItem(u, n, p, i), importedOld(false) { } QList loadXml(QIODevice *dev); bool isFavourites() const { return true; } bool canReload() const { return false; } QDateTime lastModified; bool importedOld; }; struct IceCastCategoryItem : public CategoryItem { IceCastCategoryItem(const QString &u, const QString &n=QString(), CategoryItem *p=0, const QIcon &i=QIcon(), const QString &cn=QString(), const QString &bn=QString()) : CategoryItem(u, n, p, i, cn, bn) { } void addHeaders(QNetworkRequest &req); }; struct ShoutCastCategoryItem : public CategoryItem { ShoutCastCategoryItem(const QString &u, const QString &n=QString(), CategoryItem *p=0, const QIcon &i=QIcon(), const QString &cn=QString(), const QString &bn=QString()) : CategoryItem(u, n, p, i, cn, bn) { } void addHeaders(QNetworkRequest &req); NetworkJob * fetchSecondardyUrl(); }; struct DirbleCategoryItem : public CategoryItem { DirbleCategoryItem(const QString &u, const QString &n=QString(), CategoryItem *p=0, const QIcon &i=QIcon(), const QString &cn=QString(), const QString &bn=QString()) : CategoryItem(u, n, p, i, cn, bn) { } NetworkJob * fetchSecondardyUrl(); }; struct ListenLiveCategoryItem : public CategoryItem { ListenLiveCategoryItem(const QString &u, const QString &n, CategoryItem *p, const QIcon &i, const QString &cn) : CategoryItem(u, n, p, i, cn) { } bool isListenLive() const { return true; } bool isBuiltIn() const { return false; } }; struct DiCategoryItem : public CategoryItem { DiCategoryItem(const QString &u, const QString &n, CategoryItem *p, const QIcon &i, const QString &cn) : CategoryItem(u, n, p, i, cn, QString(), true) { } bool canConfigure() const { return true; } void addHeaders(QNetworkRequest &req); bool isDi() const { return true; } bool isBuiltIn() const { return false; } }; struct SomaCategoryItem : public CategoryItem { SomaCategoryItem(const QString &u, const QString &n, CategoryItem *p, const QIcon &i, const QString &cn, bool mod) : CategoryItem(u, n, p, i, cn, QString(), mod) { } bool isSoma() const { return true; } bool isBuiltIn() const { return false; } }; struct XmlCategoryItem : public CategoryItem { XmlCategoryItem(const QString &n, CategoryItem *p, const QIcon &i, const QString &cn) : CategoryItem(QLatin1String("-"), n, p, i, cn, QString(), true) { } QList loadCache(); bool canReload() const { return false; } void removeCache() { } bool isBuiltIn() const { return false; } }; struct Category { Category(const QString &n, const QIcon &i, const QString &k, bool b, bool h, bool c) : name(n), icon(i), key(k), builtin(b), hidden(h), configurable(c) { } QString name; QIcon icon; QString key; bool builtin; bool hidden; bool configurable; }; static const QString constPrefix; static const QString constPlayListName; static const QString constSubDir; static const QString constCacheExt; static const QString constShoutCastApiKey; static const QString constShoutCastHost; static const QString constDirbleApiKey; static const QString constDirbleHost; static const QString constCompressedXmlFile; static const QString constXmlFile; static const QString constSettingsFile; static const QString constPngIcon; static const QString constSvgIcon; static const QStringList constInstallFiles; static StreamsModel * self(); static QString favouritesDir(); static QString modifyUrl(const QString &u, bool addPrefix=true, const QString &name=QString()); // static bool validProtocol(const QString &file); StreamsModel(QObject *parent = 0); ~StreamsModel(); QModelIndex index(int, int, const QModelIndex & = QModelIndex()) const; QModelIndex parent(const QModelIndex &) const; QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const; int rowCount(const QModelIndex &parent = QModelIndex()) const; int columnCount(const QModelIndex &) const; QVariant data(const QModelIndex &, int) const; Qt::ItemFlags flags(const QModelIndex &index) const; bool hasChildren(const QModelIndex &index) const; bool canFetchMore(const QModelIndex &index) const; void fetchMore(const QModelIndex &index); void reload(const QModelIndex &index); bool exportFavourites(const QString &fileName); void importIntoFavourites(const QString &fileName); bool haveFavourites() const { return !favourites->children.isEmpty(); } void removeFromFavourites(const QModelIndexList &indexes); bool addToFavourites(const QString &url, const QString &name); QString favouritesNameForUrl(const QString &u); bool nameExistsInFavourites(const QString &n); void updateFavouriteStream(const QString &url, const QString &name, const QModelIndex &idx); bool addBookmark(const QString &url, const QString &name, CategoryItem *bookmarkParentCat); void removeBookmark(const QModelIndex &index); void removeAllBookmarks(const QModelIndex &index); QModelIndex favouritesIndex() const; const QIcon & favouritesIcon() const { return favourites->icon; } bool isTuneIn(const CategoryItem *cat) const { return tuneIn==cat; } bool isShoutCast(const CategoryItem *cat) const { return shoutCast==cat; } bool isDirble(const CategoryItem *cat) const { return dirble==cat; } CategoryItem * tuneInCat() const { return tuneIn; } QStringList filenames(const QModelIndexList &indexes, bool addPrefix) const; QMimeData * mimeData(const QModelIndexList &indexes) const; QStringList mimeTypes() const; Action *addBookmarkAct() { return addBookmarkAction; } Action *addToFavouritesAct() { return addToFavouritesAction; } Action *configureAct() { return configureAction; } Action *reloadAct() { return reloadAction; } Action *searchAct() { return searchAction; } void save(); QList getCategories() const; void setHiddenCategories(const QSet &cats); CategoryItem * addInstalledProvider(const QString &name, const QIcon &icon, const QString &streamsFileName, bool replace); void removeInstalledProvider(const QString &key); QModelIndex categoryIndex(const CategoryItem *cat) const; Q_SIGNALS: void loading(); void loaded(); void error(const QString &msg); void categoriesChanged(); // Streams stored on MPD... void listFavouriteStreams(); void saveFavouriteStream(const QString &url, const QString &name); void removeFavouriteStreams(const QList &positions); void editFavouriteStream(const QString &url, const QString &name, quint32 position); void favouritesLoaded(); void addedToFavourites(const QString &name); public: static QList parseRadioTimeResponse(QIODevice *dev, CategoryItem *cat, bool parseSubText=false); static QList parseIceCastResponse(QIODevice *dev, CategoryItem *cat); static QList parseSomaFmResponse(QIODevice *dev, CategoryItem *cat); static QList parseDigitallyImportedResponse(QIODevice *dev, CategoryItem *cat); static QList parseListenLiveResponse(QIODevice *dev, CategoryItem *cat); static QList parseShoutCastSearchResponse(QIODevice *dev, CategoryItem *cat); QList parseShoutCastResponse(QIODevice *dev, CategoryItem *cat); static QList parseShoutCastLinks(QXmlStreamReader &doc, CategoryItem *cat); static QList parseShoutCastStations(QXmlStreamReader &doc, CategoryItem *cat); QList parseDirbleResponse(QIODevice *dev, CategoryItem *cat, const QString &origUrl); static QList parseDirbleStations(QIODevice *dev, CategoryItem *cat); static Item * parseRadioTimeEntry(QXmlStreamReader &doc, CategoryItem *parent, bool parseSubText=false); static Item * parseSomaFmEntry(QXmlStreamReader &doc, CategoryItem *parent); private Q_SLOTS: void jobFinished(); void tooltipUpdated(QAction *act); // Responses from MPD... void storedPlaylists(const QList &list); void savedFavouriteStream(const QString &url, const QString &name); void removedFavouriteStreams(const QList &removed); // void editedFavouriteStream(const QString &url, const QString &name, quint32 position); void favouriteStreams(const QList &streams); private: bool loadCache(CategoryItem *cat); Item * toItem(const QModelIndex &index) const { return index.isValid() ? static_cast(index.internalPointer()) : root; } void importOldFavourites(); void loadInstalledProviders(); private: QMap jobs; CategoryItem *root; FavouritesCategoryItem *favourites; CategoryItem *tuneIn; CategoryItem *shoutCast; CategoryItem *dirble; Action *addBookmarkAction; Action *addToFavouritesAction; Action *configureAction; Action *reloadAction; Action *searchAction; QList hiddenCategories; }; #elif defined ENABLE_KDE_SUPPORT #include class StreamsModel : public QObject { Q_OBJECT public: static const QString constPrefix; static const QString constPlayListName; static QString modifyUrl(const QString &u, bool addPrefix=true, const QString &name=QString()); static bool validProtocol(const QString &file); }; #else namespace StreamsModel { extern const QString constPrefix; extern const QString constPlayListName; extern QString modifyUrl(const QString &u, bool addPrefix=true, const QString &name=QString()); extern bool validProtocol(const QString &file); } #endif // ENABLE_STREAMS #endif