/* * 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 class QNetworkReply; 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 url; QString name; QString subText; CategoryItem *parent; virtual bool isCategory() const { return false; } }; 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()) : Item(u, n, p), state(Initial), isAll(false), isBookmarks(false), supportsBookmarks(false), canBookmark(false), icon(i), cacheName(cn), bookmarksName(bn) { } virtual ~CategoryItem() { qDeleteAll(children); } virtual bool isCategory() const { return true; } virtual bool canConfigure() const { return false; } virtual bool isFavourites() const { return false; } virtual void removeCache(); bool isTopLevel() const { return parent && 0==parent->parent; } bool canReload() const { return !cacheName.isEmpty() || isTopLevel() || !url.isEmpty(); } void removeBookmarks(); void saveBookmarks(); QList loadBookmarks(); CategoryItem * getBookmarksCategory(); CategoryItem * createBookmarksCategory(); void saveCache() const; 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); 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 QList children; QIcon icon; QString cacheName; QString bookmarksName; }; 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; } }; 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) { } bool canConfigure() const { return true; } }; static const QString constPrefix; static const QString constCacheDir; static const QString constCacheExt; 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; } const QIcon & tuneInIcon() const { return tuneIn->icon; } 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; } 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); QList parseShoutCastResponse(QIODevice *dev, CategoryItem *cat, const QString &origUrl); QList parseShoutCastLinks(QXmlStreamReader &doc, CategoryItem *cat); 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(); private: QMap jobs; CategoryItem *root; CategoryItem *favourites; CategoryItem *tuneIn; CategoryItem *listenLive; bool favouritesIsWriteable; bool favouritesModified; QTimer *favouritesSaveTimer; Action *addBookmarkAction; Action *addToFavouritesAction; Action *configureAction; Action *reloadAction; }; #endif