/* * Cantata * * Copyright (c) 2011-2013 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 #include #include #include #include class NetworkJob; class QNetworkRequest; class QXmlStreamReader; class QIODevice; class QTimer; class StreamsModel : public ActionModel { Q_OBJECT public: struct CategoryItem; struct Item { Item(const QString &u, const QString &n=QString(), CategoryItem *p=0, const QString &sub=QString()) : url(u), name(n), subText(sub), parent(p) { } virtual ~Item() { } QString modifiedName() const; QString url; QString name; 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 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, bool importing=false); virtual QList loadXml(QIODevice *dev, bool importing=false); virtual void addHeaders(QNetworkRequest &) { } 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) { } QList loadXml(QIODevice *dev, bool importing=false); bool isFavourites() const { return true; } bool canReload() const { return false; } }; 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); }; struct ListenLiveCategoryItem : public CategoryItem { ListenLiveCategoryItem(const QString &n, CategoryItem *p, const QIcon &i=QIcon()) : CategoryItem(QString(), n, p, i) { } void removeCache(); }; 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); }; struct XmlCategoryItem : public CategoryItem { XmlCategoryItem(const QString &n, CategoryItem *p, const QIcon &i, const QString &cn) : CategoryItem("-", n, p, i, cn, QString(), true) { } QList loadCache(); bool canReload() const { return false; } void removeCache() { } bool isBuiltIn() const; }; struct Category { Category(const QString &n, const QIcon &i, const QString &k, bool b, bool h) : name(n), icon(i), key(k), builtin(b), hidden(h) { } QString name; QIcon icon; QString key; bool builtin; bool hidden; }; static const QString constPrefix; static const QString constSubDir; static const QString constCacheExt; static const QString constShoutCastApiKey; static const QString constShoutCastHost; static const QString constCompressedXmlFile; static const QString constXmlFile; static const QString constPngIcon; static const QString constSvgIcon; 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); void saveFavourites(bool force=false); bool exportFavourites(const QString &fileName); bool importIntoFavourites(const QString &fileName) { return loadFavourites(fileName, favouritesIndex(), true); } bool haveFavourites() const { return !favourites->children.isEmpty(); } bool isFavoritesWritable() { return favouritesIsWriteable; } bool checkFavouritesWritable(); void reloadFavourites() { reload(favouritesIndex()); } void removeFromFavourites(const QModelIndex &index); bool addToFavourites(const QString &url, const QString &name); QString favouritesNameForUrl(const QString &u); bool nameExistsInFavourites(const QString &n); void updateFavouriteStream(Item *item); 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; } 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; } void save(); QList getCategories() const; void setHiddenCategories(const QSet &cats); CategoryItem * addXmlCategory(const QString &name, const QIcon &icon, const QString &xmlFileName, bool replace); void removeXmlCategory(const QString &key); Q_SIGNALS: void loading(); void loaded(); void error(const QString &msg); 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, const QString &origUrl); static QList parseShoutCastLinks(QXmlStreamReader &doc, CategoryItem *cat); static QList parseShoutCastStations(QXmlStreamReader &doc, 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 persistFavourites(); private: bool loadCache(CategoryItem *cat); Item * toItem(const QModelIndex &index) const { return index.isValid() ? static_cast(index.internalPointer()) : root; } bool loadFavourites(const QString &fileName, const QModelIndex &index, bool importing=false); void buildListenLive(); void buildXml(); private: QMap jobs; CategoryItem *root; CategoryItem *favourites; CategoryItem *tuneIn; CategoryItem *shoutCast; CategoryItem *listenLive; bool favouritesIsWriteable; bool favouritesModified; QTimer *favouritesSaveTimer; Action *addBookmarkAction; Action *addToFavouritesAction; Action *configureAction; Action *reloadAction; QList hiddenCategories; }; #endif