/* * 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. */ #include "streamsmodel.h" #include "mpd/mpdconnection.h" #include "mpd/mpdparseutils.h" #include #ifdef ENABLE_STREAMS #include "widgets/icons.h" #include "network/networkaccessmanager.h" #include "support/localize.h" #include "gui/plurals.h" #include "support/utils.h" #include "gui/settings.h" #include "playqueuemodel.h" #include "roles.h" #include "support/action.h" #include "gui/stdactions.h" #include "support/actioncollection.h" #include "digitallyimported.h" #include "qtiocompressor/qtiocompressor.h" #include "support/utils.h" #include "config.h" #include "support/globalstatic.h" #include #include #include #include #include #include #include #include #include #include #include #include #if QT_VERSION >= 0x050000 #include #include #else #include "qjson/parser.h" #endif #if defined Q_OS_WIN #include #endif #include #include #if defined ENABLE_MODEL_TEST #include "modeltest.h" #endif #include GLOBAL_STATIC(StreamsModel, instance) const QString StreamsModel::constSubDir=QLatin1String("streams"); const QString StreamsModel::constCacheExt=QLatin1String(".xml.gz"); const QString StreamsModel::constShoutCastApiKey=QLatin1String("fa1669MuiRPorUBw"); const QString StreamsModel::constShoutCastHost=QLatin1String("api.shoutcast.com"); const QString StreamsModel::constDirbleApiKey=QLatin1String("1035d2834bdc7195b8929ad7f70a8410f02c633e"); const QString StreamsModel::constDirbleHost=QLatin1String("api.dirble.com"); const QString StreamsModel::constCompressedXmlFile=QLatin1String("streams.xml.gz"); const QString StreamsModel::constXmlFile=QLatin1String("streams.xml"); const QString StreamsModel::constSettingsFile=QLatin1String("settings.json"); const QString StreamsModel::constPngIcon=QLatin1String("icon.png"); const QString StreamsModel::constSvgIcon=QLatin1String("icon.svg"); static const char * constOrigUrlProperty = "orig-url"; static QString constRadioTimeHost=QLatin1String("opml.radiotime.com"); static QString constRadioTimeUrl=QLatin1String("http://")+constRadioTimeHost+QLatin1String("/Browse.ashx"); static QString constFavouritesUrl=QLatin1String("cantata://internal"); static QString constIceCastUrl=QLatin1String("http://dir.xiph.org/yp.xml"); static const QString constDiChannelListHost=QLatin1String("api.v2.audioaddict.com"); static const QString constDiChannelListUrl=QLatin1String("http://")+constDiChannelListHost+("/v1/%1/mobile/batch_update?asset_group_key=mobile_icons&stream_set_key="); static const QString constDiStdUrl=QLatin1String("http://%1/public3/%2.pls"); static QString constShoutCastUrl=QLatin1String("http://")+StreamsModel::constShoutCastHost+QLatin1String("/genre/primary?f=xml&k=")+StreamsModel::constShoutCastApiKey; static QString constDirbleUrl=QLatin1String("http://")+StreamsModel::constDirbleHost+QLatin1String("/v1/primaryCategories/apikey/")+StreamsModel::constDirbleApiKey; static const QLatin1String constBookmarksDir=QLatin1String("bookmarks"); static QIcon getIcon(const QString &name) { QIcon icon; icon.addFile(":"+name); return icon.isNull() ? Icons::self()->streamCategoryIcon : icon; } static QIcon getExternalIcon(const QString &path, QStringList files=QStringList() << StreamsModel::constSvgIcon << StreamsModel::constPngIcon) { foreach (const QString &file, files) { QString iconFile=path+Utils::constDirSep+file; if (QFile::exists(iconFile)) { QIcon icon; icon.addFile(iconFile); return icon; } } return QIcon(); } static QString categoryCacheName(const QString &name, bool createDir=false) { return Utils::cacheDir(StreamsModel::constSubDir, createDir)+name+StreamsModel::constCacheExt; } static QString categoryBookmarksName(const QString &name, bool createDir=false) { return Utils::dataDir(constBookmarksDir, createDir)+name+StreamsModel::constCacheExt; } QString StreamsModel::Item::modifiedName() const { if (isCategory()) { return name; } const CategoryItem *cat=getTopLevelCategory(); if (!cat || !cat->addCatToModifiedName || name.startsWith(cat->name)) { return name; } return cat->name+QLatin1String(" - ")+name; } StreamsModel::CategoryItem * StreamsModel::Item::getTopLevelCategory() const { StreamsModel::Item *item=const_cast(this); while (item->parent && item->parent->parent) { item=item->parent; } return item && item->isCategory() ? static_cast(item) : 0; } void StreamsModel::CategoryItem::removeBookmarks() { if (bookmarksName.isEmpty()) { return; } QString fileName=categoryBookmarksName(bookmarksName); if (QFile::exists(fileName)) { QFile::remove(fileName); } } void StreamsModel::CategoryItem::saveBookmarks() { if (bookmarksName.isEmpty() || !supportsBookmarks) { return; } foreach (Item *child, children) { if (child->isCategory()) { CategoryItem *cat=static_cast(child); if (cat->isBookmarks) { if (cat->children.count()) { QFile file(categoryBookmarksName(bookmarksName, true)); QtIOCompressor compressor(&file); compressor.setStreamFormat(QtIOCompressor::GzipFormat); if (compressor.open(QIODevice::WriteOnly)) { QXmlStreamWriter doc(&compressor); doc.writeStartDocument(); doc.writeStartElement("bookmarks"); doc.writeAttribute("version", "1.0"); doc.setAutoFormatting(false); foreach (Item *i, cat->children) { doc.writeStartElement("bookmark"); doc.writeAttribute("name", i->name); doc.writeAttribute("url", i->url); doc.writeEndElement(); } doc.writeEndElement(); doc.writeEndElement(); } } break; } } } } QList StreamsModel::CategoryItem::loadBookmarks() { QList newItems; if (bookmarksName.isEmpty() || !supportsBookmarks) { return newItems; } QString fileName=categoryBookmarksName(bookmarksName); if (!QFile::exists(fileName)) { return newItems; } QFile file(fileName); QtIOCompressor compressor(&file); compressor.setStreamFormat(QtIOCompressor::GzipFormat); if (compressor.open(QIODevice::ReadOnly)) { QXmlStreamReader doc(&compressor); while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("bookmark")==doc.name()) { QString name=doc.attributes().value("name").toString(); QString url=doc.attributes().value("url").toString(); if (!name.isEmpty() && !url.isEmpty()) { newItems.append(new CategoryItem(url, name, this)); } } } } return newItems; } StreamsModel::CategoryItem * StreamsModel::CategoryItem::getBookmarksCategory() { foreach (Item *i, children) { if (i->isCategory() && static_cast(i)->isBookmarks) { return static_cast(i); } } return 0; } StreamsModel::CategoryItem * StreamsModel::CategoryItem::createBookmarksCategory() { Icon icon=Icon("bookmarks"); if (icon.isNull()) { icon=Icon("user-bookmarks"); } CategoryItem *bookmarkCat = new CategoryItem(QString(), i18n("Bookmarks"), this, icon); bookmarkCat->state=CategoryItem::Fetched; bookmarkCat->isBookmarks=true; return bookmarkCat; } void StreamsModel::CategoryItem::removeCache() { if (!cacheName.isEmpty()) { QString cache=categoryCacheName(cacheName); if (QFile::exists(cache)) { QFile::remove(cache); } } } void StreamsModel::CategoryItem::saveCache() const { if (!cacheName.isEmpty()) { saveXml(categoryCacheName(cacheName, true)); } } QList StreamsModel::CategoryItem::loadCache() { if (!cacheName.isEmpty()) { QString cache=categoryCacheName(cacheName); if (!cache.isEmpty() && QFile::exists(cache)) { return loadXml(cache); } } return QList(); } QList StreamsModel::XmlCategoryItem::loadCache() { QList newItems; if (QFile::exists(cacheName)) { newItems=loadXml(cacheName); QString dir=Utils::getDir(cacheName); foreach (Item *i, newItems) { if (i->isCategory()) { StreamsModel::CategoryItem *cat=static_cast(i); QString name=cat->name; name=name.replace(Utils::constDirSep, "_"); cat->icon=getExternalIcon(dir, QStringList() << name+".svg" << name+".png"); } } } return newItems; } bool StreamsModel::CategoryItem::saveXml(const QString &fileName, bool format) const { if (children.isEmpty()) { // No children, so remove XML... return !QFile::exists(fileName) || QFile::remove(fileName); } QFile file(fileName); if (fileName.endsWith(".xml")) { return file.open(QIODevice::WriteOnly) && saveXml(&file, format); } else { QtIOCompressor compressor(&file); compressor.setStreamFormat(QtIOCompressor::GzipFormat); return compressor.open(QIODevice::WriteOnly) && saveXml(&compressor, format); } } static void saveStream(QXmlStreamWriter &doc, const StreamsModel::Item *item) { doc.writeStartElement("stream"); doc.writeAttribute("name", item->name); doc.writeAttribute("url", item->url); doc.writeEndElement(); } static void saveCategory(QXmlStreamWriter &doc, const StreamsModel::CategoryItem *cat) { if (cat->isBookmarks) { return; } doc.writeStartElement("category"); doc.writeAttribute("name", cat->name); if (cat->isAll) { doc.writeAttribute("isAll", "true"); } foreach (const StreamsModel::Item *i, cat->children) { if (i->isCategory()) { saveCategory(doc, static_cast(i)); } else { saveStream(doc, static_cast(i)); } } doc.writeEndElement(); } bool StreamsModel::CategoryItem::saveXml(QIODevice *dev, bool format) const { QXmlStreamWriter doc(dev); doc.writeStartDocument(); doc.writeStartElement("streams"); doc.writeAttribute("version", "1.0"); if (format) { doc.setAutoFormatting(true); doc.setAutoFormattingIndent(1); } else { doc.setAutoFormatting(false); } foreach (const Item *i, children) { if (i->isCategory()) { ::saveCategory(doc, static_cast(i)); } else { ::saveStream(doc, i); } } doc.writeEndElement(); doc.writeEndDocument(); return true; } QList StreamsModel::CategoryItem::loadXml(const QString &fileName) { QFile file(fileName); if (!file.open(QIODevice::ReadOnly)) { return QList(); } // Check for gzip header... QByteArray header=file.read(2); bool isCompressed=((unsigned char)header[0])==0x1f && ((unsigned char)header[1])==0x8b; file.seek(0); QtIOCompressor compressor(&file); if (isCompressed) { compressor.setStreamFormat(QtIOCompressor::GzipFormat); if (!compressor.open(QIODevice::ReadOnly)) { return QList(); } } return loadXml(isCompressed ? (QIODevice *)&compressor : (QIODevice *)&file); } QList StreamsModel::CategoryItem::loadXml(QIODevice *dev) { QList newItems; QXmlStreamReader doc(dev); CategoryItem *currentCat=this; CategoryItem *prevCat=this; while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement()) { if (QLatin1String("streams")==doc.name()) { if (QLatin1String("true")==doc.attributes().value("addCategoryName").toString()) { addCatToModifiedName=true; } } else if (QLatin1String("stream")==doc.name()) { QString name=doc.attributes().value("name").toString(); QString url=doc.attributes().value("url").toString(); if (currentCat==this) { newItems.append(new Item(url, name, currentCat)); } else { currentCat->children.append(new Item(url, name, currentCat)); } } else if (QLatin1String("category")==doc.name()) { prevCat=currentCat; QString name=doc.attributes().value("name").toString(); currentCat=new CategoryItem(QString(), name, prevCat); currentCat->state=CategoryItem::Fetched; currentCat->isAll=QLatin1String("true")==doc.attributes().value("isAll").toString(); newItems.append(currentCat); } } else if (doc.isEndElement() && QLatin1String("category")==doc.name()) { currentCat=prevCat; } } return newItems; } QList StreamsModel::FavouritesCategoryItem::loadXml(QIODevice *dev) { QList newItems; QXmlStreamReader doc(dev); QSet existingUrls; QSet existingNames; foreach (Item *i, children) { existingUrls.insert(i->url); existingNames.insert(i->name); } while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("stream")==doc.name()) { QString name=doc.attributes().value("name").toString(); QString origName=name; QString url=doc.attributes().value("url").toString(); if (!url.isEmpty() && !name.isEmpty() && !existingUrls.contains(url)) { int i=1; for (; i<100 && existingNames.contains(name); ++i) { name=origName+QLatin1String(" (")+QString::number(i)+QLatin1Char(')'); } if (i<100) { existingNames.insert(name); existingUrls.insert(url); newItems.append(new Item(url, name, this)); } } } } return newItems; } void StreamsModel::IceCastCategoryItem::addHeaders(QNetworkRequest &req) { req.setRawHeader("Accept-Encoding", "gzip"); } void StreamsModel::ShoutCastCategoryItem::addHeaders(QNetworkRequest &req) { req.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded"); } NetworkJob * StreamsModel::ShoutCastCategoryItem::fetchSecondardyUrl() { if (url!=constShoutCastUrl) { // Get stations... QUrl url(QLatin1String("http://")+constShoutCastHost+QLatin1String("/legacy/genresearch")); #if QT_VERSION < 0x050000 QUrl &query=url; #else QUrlQuery query; #endif query.addQueryItem("k", constShoutCastApiKey); query.addQueryItem("genre", name); #if QT_VERSION >= 0x050000 url.setQuery(query); #endif QNetworkRequest req(url); addHeaders(req); NetworkJob *job=NetworkAccessManager::self()->get(req); job->setProperty(constOrigUrlProperty, url.toString()); return job; } return 0; } NetworkJob * StreamsModel::DirbleCategoryItem::fetchSecondardyUrl() { if (url!=constDirbleUrl) { // Get stations... QString catId=url.split("/", QString::SkipEmptyParts).last(); QUrl url(QLatin1String("http://")+constDirbleHost+QLatin1String("/v1/stations/apikey/")+constDirbleApiKey+QLatin1String("/id/")+catId); QNetworkRequest req(url); addHeaders(req); NetworkJob *job=NetworkAccessManager::self()->get(req); job->setProperty(constOrigUrlProperty, url.toString()); return job; } return 0; } void StreamsModel::DiCategoryItem::addHeaders(QNetworkRequest &req) { DigitallyImported::self()->addAuthHeader(req); } StreamsModel::StreamsModel(QObject *parent) : ActionModel(parent) , root(new CategoryItem(QString(), "root")) { tuneIn=new CategoryItem(constRadioTimeUrl+QLatin1String("?locale=")+QLocale::system().name(), i18n("TuneIn"), root, getIcon("tunein"), QString(), "tunein"); tuneIn->supportsBookmarks=true; root->children.append(tuneIn); root->children.append(new IceCastCategoryItem(constIceCastUrl, i18n("IceCast"), root, getIcon("icecast"), "icecast")); shoutCast=new ShoutCastCategoryItem(constShoutCastUrl, i18n("ShoutCast"), root, getIcon("shoutcast")); shoutCast->configName="shoutcast"; root->children.append(shoutCast); dirble=new DirbleCategoryItem(constDirbleUrl, i18n("Dirble"), root, getIcon("dirble")); dirble->configName="dirble"; root->children.append(dirble); favourites=new FavouritesCategoryItem(constFavouritesUrl, i18n("Favorites"), root, getIcon("favourites")); root->children.append(favourites); loadInstalledProviders(); addBookmarkAction = ActionCollection::get()->createAction("bookmarkcategory", i18n("Bookmark Category"), Icon("bookmark-new")); addToFavouritesAction = ActionCollection::get()->createAction("addtofavourites", i18n("Add Stream To Favorites"), favouritesIcon()); configureAction = ActionCollection::get()->createAction("configurestreams", i18n("Configure Streams"), Icons::self()->configureIcon); reloadAction = ActionCollection::get()->createAction("reloadstreams", i18n("Reload"), Icon("view-refresh")); // For Mac/Unity we try to hide icons from menubar menus. However, search is used in the menubar AND in the streams view. We // need the icon on the streams view. Therefore, if the StdAction has no icon - we create a new one and forward all signals... if (StdActions::self()->searchAction->icon().isNull()) { searchAction = new Action(Icon("edit-find"), StdActions::self()->searchAction->text(), this); searchAction->setToolTip(StdActions::self()->searchAction->toolTip()); connect(searchAction, SIGNAL(triggered(bool)), StdActions::self()->searchAction, SIGNAL(triggered(bool))); connect(searchAction, SIGNAL(triggered()), StdActions::self()->searchAction, SIGNAL(triggered())); connect(ActionCollection::get(), SIGNAL(tooltipUpdated(QAction *)), SLOT(tooltipUpdated(QAction *))); } else { searchAction = StdActions::self()->searchAction; } QSet hidden=Settings::self()->hiddenStreamCategories().toSet(); foreach (Item *c, root->children) { if (c!=favourites) { CategoryItem *cat=static_cast(c); if (hidden.contains(cat->configName)) { hiddenCategories.append(c); root->children.removeAll(c); } } } #if defined ENABLE_MODEL_TEST new ModelTest(this, this); #endif connect(MPDConnection::self(), SIGNAL(playlistsRetrieved(QList)), SLOT(storedPlaylists(QList))); connect(MPDConnection::self(), SIGNAL(savedStream(QString,QString)), SLOT(savedFavouriteStream(QString,QString))); connect(MPDConnection::self(), SIGNAL(removedStreams(QList)), SLOT(removedFavouriteStreams(QList))); connect(MPDConnection::self(), SIGNAL(streamList(QList)), SLOT(favouriteStreams(QList))); connect(this, SIGNAL(listFavouriteStreams()), MPDConnection::self(), SLOT(listStreams())); connect(this, SIGNAL(saveFavouriteStream(QString,QString)), MPDConnection::self(), SLOT(saveStream(QString,QString))); connect(this, SIGNAL(removeFavouriteStreams(QList)), MPDConnection::self(), SLOT(removeStreams(QList))); connect(this, SIGNAL(editFavouriteStream(QString,QString,quint32)), MPDConnection::self(), SLOT(editStream(QString,QString,quint32))); } StreamsModel::~StreamsModel() { delete root; } QModelIndex StreamsModel::index(int row, int column, const QModelIndex &parent) const { if (!hasIndex(row, column, parent)) { return QModelIndex(); } const CategoryItem * p = parent.isValid() ? static_cast(parent.internalPointer()) : root; const Item * c = rowchildren.count() ? p->children.at(row) : 0; return c ? createIndex(row, column, (void *)c) : QModelIndex(); } QModelIndex StreamsModel::parent(const QModelIndex &index) const { if (!index.isValid()) { return QModelIndex(); } Item * parent = toItem(index)->parent; if (!parent || parent == root || !parent->parent) { return QModelIndex(); } return createIndex(static_cast(parent->parent)->children.indexOf(parent), 0, parent); } QVariant StreamsModel::headerData(int /*section*/, Qt::Orientation /*orientation*/, int /*role*/) const { return QVariant(); } int StreamsModel::rowCount(const QModelIndex &parent) const { if (parent.isValid()) { Item *item = toItem(parent); return item->isCategory() ? static_cast(item)->children.count() : 0; } return root->children.count(); } int StreamsModel::columnCount(const QModelIndex &) const { return 1; } QVariant StreamsModel::data(const QModelIndex &index, int role) const { if (!index.isValid()) { return QVariant(); } const Item *item = toItem(index); switch (role) { case Qt::DecorationRole: if (item->isCategory()) { const CategoryItem *cat=static_cast(item); return cat->icon.isNull() ? Icons::self()->streamCategoryIcon : cat->icon; } else { return Icons::self()->radioStreamIcon; } case Qt::DisplayRole: return item->name; case Qt::ToolTipRole: return item->isCategory() ? item->name : (item->name+QLatin1String("
")+item->url+QLatin1String("")); case Cantata::Role_SubText: if (item->isCategory()) { const CategoryItem *cat=static_cast(item); switch (cat->state) { case CategoryItem::Initial: return i18n("Not Loaded"); case CategoryItem::Fetching: return i18n("Loading..."); default: return Plurals::entries(cat->children.count()); } } break; case Cantata::Role_Actions: { QList actions; if (item->isCategory()){ const CategoryItem *cat=static_cast(item); if (cat->canConfigure()) { actions << configureAction; } if (cat->canReload()) { actions << reloadAction; } if (tuneIn==item || shoutCast==item || dirble==item || (cat->isTopLevel() && !cat->children.isEmpty())) { actions << searchAction; } if (cat->canBookmark) { actions << addBookmarkAction; } } else { actions << StdActions::self()->replacePlayQueueAction; if (item->parent!=favourites) { actions << addToFavouritesAction; } } if (!actions.isEmpty()) { QVariant v; v.setValue >(actions); return v; } break; } default: break; } return QVariant(); } Qt::ItemFlags StreamsModel::flags(const QModelIndex &index) const { if (index.isValid()) { if (toItem(index)->isCategory()) { return Qt::ItemIsSelectable | Qt::ItemIsEnabled; } else { return Qt::ItemIsSelectable | Qt::ItemIsDragEnabled | Qt::ItemIsEnabled; } } return Qt::NoItemFlags; } bool StreamsModel::hasChildren(const QModelIndex &index) const { return index.isValid() ? toItem(index)->isCategory() : true; } bool StreamsModel::canFetchMore(const QModelIndex &index) const { if (index.isValid()) { Item *item = toItem(index); return item->isCategory() && CategoryItem::Initial==static_cast(item)->state && !item->url.isEmpty() && !static_cast(item)->isFavourites(); } else { return false; } } void StreamsModel::fetchMore(const QModelIndex &index) { if (!index.isValid()) { return; } Item *item = toItem(index); if (item->isCategory() && !item->url.isEmpty()) { CategoryItem *cat=static_cast(item); if (!cat->isFavourites() && !loadCache(cat)) { QNetworkRequest req; if (cat->isDi()) { req=QNetworkRequest(constDiChannelListUrl.arg(cat->url.split(".").at(1))); } else { req=QNetworkRequest(cat->url); } cat->addHeaders(req); NetworkJob *job=NetworkAccessManager::self()->get(req); job->setProperty(constOrigUrlProperty, cat->url); if (jobs.isEmpty()) { emit loading(); } jobs.insert(job, cat); connect(job, SIGNAL(finished()), SLOT(jobFinished())); cat->state=CategoryItem::Fetching; job=cat->fetchSecondardyUrl(); if (job) { jobs.insert(job, cat); connect(job, SIGNAL(finished()), SLOT(jobFinished())); } } emit dataChanged(index, index); } } void StreamsModel::reload(const QModelIndex &index) { Item *item = toItem(index); if (!item->isCategory()) { return; } CategoryItem *cat=static_cast(item); if (!cat->children.isEmpty()) { cat->removeCache(); beginRemoveRows(index, 0, cat->children.count()-1); qDeleteAll(cat->children); cat->children.clear(); endRemoveRows(); } fetchMore(index); } bool StreamsModel::exportFavourites(const QString &fileName) { return favourites->saveXml(fileName, true); } void StreamsModel::importIntoFavourites(const QString &fileName) { QList newItems=favourites->CategoryItem::loadXml(fileName); if (!newItems.isEmpty()) { foreach (Item *i, newItems) { emit saveFavouriteStream(i->url, i->name); } qDeleteAll(newItems); } } void StreamsModel::removeFromFavourites(const QModelIndexList &indexes) { QList rows; foreach (const QModelIndex &idx, indexes) { rows.append(idx.row()); } if (!rows.isEmpty()) { emit removeFavouriteStreams(rows); } } bool StreamsModel::addToFavourites(const QString &url, const QString &name) { QSet existingNames; foreach (Item *i, favourites->children) { if (i->url==url) { return false; } existingNames.insert(i->name); } QString n=name; int i=1; for (; i<100 && existingNames.contains(n); ++i) { n=name+QLatin1String("_")+QString::number(i); } if (i<100) { emit saveFavouriteStream(url, name); return true; } return false; } QString StreamsModel::favouritesNameForUrl(const QString &u) { foreach (Item *i, favourites->children) { if (i->url==u) { return i->name; } } return QString(); } bool StreamsModel::nameExistsInFavourites(const QString &n) { foreach (Item *i, favourites->children) { if (i->name==n) { return true; } } return false; } void StreamsModel::updateFavouriteStream(const QString &url, const QString &name, const QModelIndex &idx) { emit editFavouriteStream(url, name, idx.row()); } bool StreamsModel::addBookmark(const QString &url, const QString &name, CategoryItem *bookmarkParentCat) { if (!bookmarkParentCat) { bookmarkParentCat=tuneIn; } if (bookmarkParentCat && !url.isEmpty() && !name.isEmpty()) { CategoryItem *bookmarkCat=bookmarkParentCat->getBookmarksCategory(); if (!bookmarkCat) { QModelIndex index=createIndex(bookmarkParentCat->parent->children.indexOf(bookmarkParentCat), 0, (void *)bookmarkParentCat); beginInsertRows(index, bookmarkParentCat->children.count(), bookmarkParentCat->children.count()); bookmarkCat=bookmarkParentCat->createBookmarksCategory(); bookmarkParentCat->children.append(bookmarkCat); endInsertRows(); } foreach (Item *i, bookmarkCat->children) { if (i->url==url) { return false; } } QModelIndex index=createIndex(bookmarkCat->parent->children.indexOf(bookmarkCat), 0, (void *)bookmarkCat); beginInsertRows(index, bookmarkCat->children.count(), bookmarkCat->children.count()); bookmarkCat->children.append(new CategoryItem(url, name, bookmarkCat)); endInsertRows(); bookmarkParentCat->saveBookmarks(); return true; } return false; } void StreamsModel::removeBookmark(const QModelIndex &index) { Item *item=toItem(index); if (item->isCategory() && item->parent && item->parent->isBookmarks) { CategoryItem *bookmarkCat=item->parent; // 'Bookmarks' CategoryItem *cat=bookmarkCat->parent; // e.g. 'TuneIn' if (1==bookmarkCat->children.count()) { // Only 1 bookark, so remove 'Bookmarks' folder... int pos=cat->children.indexOf(bookmarkCat); QModelIndex index=createIndex(cat->parent->children.indexOf(cat), 0, (void *)cat); beginRemoveRows(index, pos, pos); delete cat->children.takeAt(pos); endRemoveRows(); cat->removeBookmarks(); } else if (!bookmarkCat->children.isEmpty()) { // More than 1 bookmark... int pos=bookmarkCat->children.indexOf(item); QModelIndex index=createIndex(bookmarkCat->parent->children.indexOf(bookmarkCat), 0, (void *)bookmarkCat); beginRemoveRows(index, pos, pos); delete bookmarkCat->children.takeAt(pos); endRemoveRows(); cat->saveBookmarks(); } } } void StreamsModel::removeAllBookmarks(const QModelIndex &index) { Item *item=toItem(index); if (item->isCategory() && static_cast(item)->isBookmarks) { CategoryItem *cat=item->parent; // e.g. 'TuneIn' int pos=cat->children.indexOf(item); QModelIndex index=createIndex(cat->parent->children.indexOf(cat), 0, (void *)cat); beginRemoveRows(index, pos, pos); delete cat->children.takeAt(pos); endRemoveRows(); cat->removeBookmarks(); } } QModelIndex StreamsModel::favouritesIndex() const { return createIndex(root->children.indexOf(favourites), 0, (void *)favourites); } static QString addDiHash(const StreamsModel::Item *item) { return item->parent && item->parent->isDi() ? DigitallyImported::self()->modifyUrl(item->url) : item->url; } QStringList StreamsModel::filenames(const QModelIndexList &indexes, bool addPrefix) const { QStringList fnames; foreach(QModelIndex index, indexes) { Item *item=static_cast(index.internalPointer()); if (!item->isCategory()) { fnames << modifyUrl(addDiHash(item), addPrefix, addPrefix ? item->modifiedName() : item->name); } } return fnames; } QMimeData * StreamsModel::mimeData(const QModelIndexList &indexes) const { QMimeData *mimeData = new QMimeData(); PlayQueueModel::encode(*mimeData, PlayQueueModel::constFileNameMimeType, filenames(indexes, true)); return mimeData; } QStringList StreamsModel::mimeTypes() const { QStringList types; types << PlayQueueModel::constFileNameMimeType; return types; } void StreamsModel::save() { QStringList disabled; foreach (Item *i, hiddenCategories) { disabled.append(static_cast(i)->configName); } disabled.sort(); Settings::self()->saveHiddenStreamCategories(disabled); } QList StreamsModel::getCategories() const { QList categories; foreach (Item *i, hiddenCategories) { categories.append(Category(i->name, static_cast(i)->icon, static_cast(i)->configName, static_cast(i)->isBuiltIn(), true, static_cast(i)->canConfigure())); } foreach (Item *i, root->children) { if (i!=favourites) { categories.append(Category(i->name, static_cast(i)->icon, static_cast(i)->configName, static_cast(i)->isBuiltIn(), false, static_cast(i)->canConfigure())); } } return categories; } void StreamsModel::setHiddenCategories(const QSet &cats) { bool changed=false; foreach (Item *i, hiddenCategories) { if (!cats.contains(static_cast(i)->configName)) { beginInsertRows(QModelIndex(), root->children.count(), root->children.count()); root->children.append(i); hiddenCategories.removeAll(i); endInsertRows(); changed=true; } } foreach (Item *i, root->children) { if (cats.contains(static_cast(i)->configName)) { int row=root->children.indexOf(i); if (row>0) { beginRemoveRows(QModelIndex(), row, row); hiddenCategories.append(root->children.takeAt(row)); endRemoveRows(); changed=true; } } } if (changed) { emit categoriesChanged(); } } bool StreamsModel::loadCache(CategoryItem *cat) { QList newItems=cat->loadCache(); if (!newItems.isEmpty()) { QModelIndex index=createIndex(cat->parent->children.indexOf(cat), 0, (void *)cat); beginInsertRows(index, cat->children.count(), (cat->children.count()+newItems.count())-1); cat->children+=newItems; endInsertRows(); cat->state=CategoryItem::Fetched; emit dataChanged(index, index); return true; } return false; } void StreamsModel::jobFinished() { NetworkJob *job=dynamic_cast(sender()); if (!job) { return; } job->deleteLater(); if (jobs.contains(job)) { CategoryItem *cat=jobs[job]; if (!cat) { return; } jobs.remove(job); // ShoutCast and Dirble have two jobs when listing a category - child categories and station list. // So, only set as fetched if both are finished. bool haveOtherJob=false; foreach (CategoryItem *c, jobs.values()) { if (c==cat) { haveOtherJob=true; break; } } if (!haveOtherJob) { cat->state=CategoryItem::Fetched; } QModelIndex index=createIndex(cat->parent->children.indexOf(cat), 0, (void *)cat); if (job->ok()) { QList newItems; if (cat!=favourites && QLatin1String("http")==job->url().scheme()) { QString url=job->origUrl().toString(); if (constRadioTimeHost==job->origUrl().host()) { newItems=parseRadioTimeResponse(job->actualJob(), cat); } else if (constIceCastUrl==url) { newItems=parseIceCastResponse(job->actualJob(), cat); } else if (cat->isSoma()) { newItems=parseSomaFmResponse(job->actualJob(), cat); } else if (constDiChannelListHost==job->origUrl().host()) { newItems=parseDigitallyImportedResponse(job->actualJob(), cat); } else if (constShoutCastHost==job->origUrl().host()) { newItems=parseShoutCastResponse(job->actualJob(), cat); } else if (constDirbleHost==job->origUrl().host()) { newItems=parseDirbleResponse(job->actualJob(), cat, job->property(constOrigUrlProperty).toString()); } else if (cat->isListenLive()) { newItems=parseListenLiveResponse(job->actualJob(), cat); } } if (cat->parent==root && cat->supportsBookmarks) { QList bookmarks=cat->loadBookmarks(); if (bookmarks.count()) { CategoryItem *bookmarksCat=cat->getBookmarksCategory(); if (bookmarksCat) { QList newBookmarks; foreach (Item *bm, bookmarks) { foreach (Item *ex, bookmarksCat->children) { if (ex->url==bm->url) { delete bm; bm=0; break; } } if (bm) { newBookmarks.append(bm); bm->parent=bookmarksCat; } } if (newBookmarks.count()) { QModelIndex index=createIndex(bookmarksCat->parent->children.indexOf(bookmarksCat), 0, (void *)bookmarksCat); beginInsertRows(index, bookmarksCat->children.count(), (bookmarksCat->children.count()+newBookmarks.count())-1); bookmarksCat->children+=newBookmarks; endInsertRows(); } } else { bookmarksCat=cat->createBookmarksCategory(); foreach (Item *i, bookmarks) { i->parent=bookmarksCat; } bookmarksCat->children=bookmarks; newItems.append(bookmarksCat); } } } if (!newItems.isEmpty()) { beginInsertRows(index, cat->children.count(), (cat->children.count()+newItems.count())-1); cat->children+=newItems; endInsertRows(); if (cat!=favourites) { cat->saveCache(); } } } emit dataChanged(index, index); if (jobs.isEmpty()) { emit loaded(); } } } // Required due to icon missing for StdActions::searchAction for Mac/Unity... See note in constructor above. void StreamsModel::tooltipUpdated(QAction *act) { if (act!=searchAction && act==StdActions::self()->searchAction) { searchAction->setToolTip(StdActions::self()->searchAction->toolTip()); } } void StreamsModel::storedPlaylists(const QList &list) { QDateTime favouritesDateTime; foreach (const Playlist &p, list) { if (p.name==constPlayListName) { favouritesDateTime=p.lastModified; break; } } if (favouritesDateTime.isNull()) { favourites->state=CategoryItem::Fetched; if (!favourites->children.isEmpty()) { beginRemoveRows(favouritesIndex(), 0, favourites->children.count()-1); qDeleteAll(favourites->children); favourites->children.clear(); endRemoveRows(); } importOldFavourites(); } else if (favourites->lastModified.isNull() || favourites->lastModifiedlastModified.currentMSecsSinceEpoch())<3) { if (favourites->lastModified.isNull()) { favourites->state=CategoryItem::Fetching; } emit listFavouriteStreams(); } favourites->lastModified=favouritesDateTime; } void StreamsModel::savedFavouriteStream(const QString &url, const QString &name) { if (favouritesNameForUrl(url).isEmpty()) { beginInsertRows(favouritesIndex(), favourites->children.count(), favourites->children.count()); favourites->children.append(new Item(url, name, favourites)); endInsertRows(); emit addedToFavourites(name); } else { emit listFavouriteStreams(); } } void StreamsModel::removedFavouriteStreams(const QList &removed) { if (removed.isEmpty()) { return; } quint32 adjust=0; QModelIndex parent=favouritesIndex(); QList::ConstIterator it=removed.constBegin(); QList::ConstIterator end=removed.constEnd(); while(it!=end) { quint32 rowBegin=*it; quint32 rowEnd=*it; QList::ConstIterator next=it+1; while(next!=end) { if (*next!=(rowEnd+1)) { break; } else { it=next; rowEnd=*next; next++; } } beginRemoveRows(parent, rowBegin-adjust, rowEnd-adjust); for (quint32 i=rowBegin; i<=rowEnd; ++i) { delete favourites->children.takeAt(rowBegin-adjust); } adjust+=(rowEnd-rowBegin)+1; endRemoveRows(); it++; } emit dataChanged(parent, parent); } static StreamsModel::Item * getStream(StreamsModel::FavouritesCategoryItem *fav, const Stream &stream, int offset) { int i=0; foreach (StreamsModel::Item *s, fav->children) { if (++iname==stream.name && s->url==stream.url) { return s; } } return 0; } void StreamsModel::favouriteStreams(const QList &streams) { QModelIndex idx=favouritesIndex(); if (favourites->children.isEmpty()) { if (streams.isEmpty()) { if (CategoryItem::Fetched!=favourites->state) { favourites->state=CategoryItem::Fetched; emit dataChanged(idx, idx); } return; } beginInsertRows(idx, 0, streams.count()-1); foreach (const Stream &s, streams) { favourites->children.append(new Item(s.url, s.name, favourites)); } endInsertRows(); } else if (streams.isEmpty()) { beginRemoveRows(idx, 0, favourites->children.count()-1); qDeleteAll(favourites->children); favourites->children.clear(); endRemoveRows(); } else { for (qint32 i=0; ichildren.count() ? favourites->children.at(i) : 0; if (!si || si->name!=s.name || si->url!=s.url) { si=ichildren.count() ? getStream(favourites, s, i) : 0; if (!si) { beginInsertRows(idx, i, i); favourites->children.insert(i, new Item(s.url, s.name, favourites)); endInsertRows(); } else { int existing=favourites->children.indexOf(si); beginMoveRows(idx, existing, existing, idx, i>existing ? i+1 : i); Item *si=favourites->children.takeAt(existing); favourites->children.insert(i, si); endMoveRows(); } } } if (favourites->children.count()>streams.count()) { int toRemove=favourites->children.count()-streams.count(); beginRemoveRows(idx, favourites->children.count()-toRemove, favourites->children.count()-1); for (int i=0; ichildren.takeLast(); } endRemoveRows(); } } importOldFavourites(); if (CategoryItem::Fetched!=favourites->state) { favourites->state=CategoryItem::Fetched; emit dataChanged(idx, idx); emit favouritesLoaded(); } } QList StreamsModel::parseRadioTimeResponse(QIODevice *dev, CategoryItem *cat, bool parseSubText) { QList newItems; QXmlStreamReader doc(dev); while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("outline")==doc.name()) { Item *item = parseRadioTimeEntry(doc, cat, parseSubText); if (item) { newItems.append(item); } } } return newItems; } static QStringList fixGenres(const QString &genre) { QStringList allGenres=Song::capitalize(genre).split(' ', QString::SkipEmptyParts); QStringList fixed; foreach (const QString &genre, allGenres) { if (genre.length() < 2 || genre.contains("ÃÂ") || // Broken unicode. genre.contains(QRegExp("^#x[0-9a-f][0-9a-f]"))) { // Broken XML entities. continue; } // Convert 80 -> 80s. if (genre.contains(QRegExp("^[0-9]0$"))) { fixed << genre + 's'; } else { fixed << genre; } } if (fixed.isEmpty()) { fixed << i18n("Other"); } return fixed; } static void trimGenres(QMap > &genres) { QString other=i18n("Other"); QSet genreSet = genres.keys().toSet(); foreach (const QString &genre, genreSet) { if (other!=genre && genres[genre].count() < 2) { genres[other]+=genres[genre]; genres.remove(genre); } } } QList StreamsModel::parseIceCastResponse(QIODevice *dev, CategoryItem *cat) { QList newItems; QtIOCompressor compressor(dev); compressor.setStreamFormat(QtIOCompressor::GzipFormat); compressor.open(QIODevice::ReadOnly); QXmlStreamReader doc(&compressor); QSet names; QMap > genres; while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("entry")==doc.name()) { QString name; QString url; QStringList stationGenres; while (!doc.atEnd()) { doc.readNext(); if (QXmlStreamReader::StartElement==doc.tokenType()) { QStringRef elem = doc.name(); if (QLatin1String("server_name")==elem) { name=doc.readElementText().trimmed(); } else if (QLatin1String("listen_url")==elem) { url=doc.readElementText().trimmed(); } else if (QLatin1String("genre")==elem) { stationGenres=fixGenres(doc.readElementText().trimmed()); } } else if (doc.isEndElement() && QLatin1String("entry")==doc.name()) { break; } } if (!name.isEmpty() && !url.isEmpty() && !names.contains(name)) { names.insert(name); foreach (const QString &g, stationGenres) { genres[g].append(new Item(url, name, cat)); } } } } trimGenres(genres); QMap >::ConstIterator it(genres.constBegin()); QMap >::ConstIterator end(genres.constEnd()); for (; it!=end; ++it) { CategoryItem *genre=new CategoryItem(QString(), it.key(), cat); genre->state=CategoryItem::Fetched; foreach (Item *i, it.value()) { i->parent=genre; genre->children.append(i); } newItems.append(genre); } return newItems; } QList StreamsModel::parseSomaFmResponse(QIODevice *dev, CategoryItem *cat) { QList newItems; QXmlStreamReader doc(dev); while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("channel")==doc.name()) { Item *item = parseSomaFmEntry(doc, cat); if (item) { newItems.append(item); } } } return newItems; } QList StreamsModel::parseDigitallyImportedResponse(QIODevice *dev, CategoryItem *cat) { QList newItems; #if QT_VERSION >= 0x050000 QVariantMap data=QJsonDocument::fromJson(dev->readAll()).toVariant().toMap(); #else QVariantMap data = QJson::Parser().parse(dev->readAll()).toMap(); #endif QString listenHost=QLatin1String("listen.")+QUrl(cat->url).host().remove("www."); if (data.contains("channel_filters")) { QVariantList filters = data["channel_filters"].toList(); foreach (const QVariant &filter, filters) { // Find the filter called "All" QVariantMap filterMap = filter.toMap(); if (filterMap.value("name", QString()).toString() != "All") { continue; } // Add all its stations to the result QVariantList channels = filterMap.value("channels", QVariantList()).toList(); foreach (const QVariant &channel, channels) { QVariantMap channelMap = channel.toMap(); QString url=constDiStdUrl.arg(listenHost).arg(channelMap.value("key").toString()); newItems.append(new Item(url, channelMap.value("name").toString(), cat)); } break; } } return newItems; } struct ListenLiveStream { enum Format { Unknown, WMA, OGG, MP3, AAC }; ListenLiveStream() : format(Unknown), bitrate(0) { } bool operator<(const ListenLiveStream &o) const { return weight()>o.weight(); } int weight() const { return ((bitrate&0xff)<<4)+(format&0x0f); } void setFormat(const QString &f) { if (QLatin1String("mp3")==f.toLower()) { format=MP3; } else if (QLatin1String("aacplus")==f.toLower()) { format=AAC; } else if (QLatin1String("ogg vorbis")==f.toLower()) { format=OGG; } else if (QLatin1String("windows media")==f.toLower()) { format=WMA; } else { format=Unknown; } } QString url; Format format; unsigned int bitrate; }; struct ListenLiveStationEntry { ListenLiveStationEntry() { clear(); } void clear() { name=location=QString(); streams.clear(); } QString name; QString location; QList streams; }; static QString getString(QString &str, const QString &start, const QString &end) { QString rv; int b=str.indexOf(start); int e=-1==b ? -1 : str.indexOf(end, b+start.length()); if (-1!=e) { rv=str.mid(b+start.length(), e-(b+start.length())).trimmed(); str=str.mid(e+end.length()); } return rv; } QList StreamsModel::parseListenLiveResponse(QIODevice *dev, CategoryItem *cat) { QList newItems; if (dev) { ListenLiveStationEntry entry; while (!dev->atEnd()) { QString line=QString::fromUtf8(dev->readLine()).trimmed().replace("> <", "><").replace("", "").replace("
", "
") .replace(" ,", ","); if (""==line) { entry.clear(); } else if (line.startsWith("", ""); QString extra=getString(line, "", ""); if (!extra.isEmpty()) { entry.name+=" "+extra; } } else { // Station URLs... QString url; QString bitrate; int idx=0; do { url=getString(line, "href=\"", "\""); bitrate=getString(line, ">", " Kbps"); bool sameFormatAsLast=line.startsWith(","); if (!url.isEmpty() && !bitrate.isEmpty() && !url.startsWith(QLatin1String("javascript")) && idx")) { if (entry.location.isEmpty()) { entry.location=getString(line, "", ""); } } else if (""==line) { if (entry.streams.count()) { qSort(entry.streams); QString name; QString url=entry.streams.at(0).url; if (QLatin1String("National")==entry.location || entry.name.endsWith(QLatin1Char('(')+entry.location+QLatin1Char(')'))) { name=entry.name; } else if (entry.name.endsWith(QLatin1Char(')'))) { name=entry.name.left(entry.name.length()-1)+QLatin1String(", ")+entry.location+QLatin1Char(')'); } else { name=entry.name+QLatin1String(" (")+entry.location+QLatin1Char(')'); } if (!name.isEmpty()) { newItems.append(new Item(url, name, cat)); } } } } } return newItems; } QList StreamsModel::parseShoutCastSearchResponse(QIODevice *dev, CategoryItem *cat) { QList newItems; QXmlStreamReader doc(dev); while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("stationlist")==doc.name()) { newItems+=parseShoutCastStations(doc, cat); } } return newItems; } QList StreamsModel::parseShoutCastResponse(QIODevice *dev, CategoryItem *cat) { QList newItems; QXmlStreamReader doc(dev); while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("genrelist")==doc.name()) { newItems+=parseShoutCastLinks(doc, cat); } else if (doc.isStartElement() && QLatin1String("stationlist")==doc.name()) { newItems+=parseShoutCastStations(doc, cat); } } return newItems; } QList StreamsModel::parseShoutCastLinks(QXmlStreamReader &doc, CategoryItem *cat) { QList newItems; while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("genre")==doc.name()) { newItems.append(new ShoutCastCategoryItem(QLatin1String("http://")+constShoutCastHost+QLatin1String("/genre/secondary?parentid=")+ doc.attributes().value("id").toString()+QLatin1String("&f=xml&k=")+constShoutCastApiKey, doc.attributes().value("name").toString(), cat)); } else if (doc.isEndElement() && QLatin1String("genrelist")==doc.name()) { return newItems; } } return newItems; } QList StreamsModel::parseShoutCastStations(QXmlStreamReader &doc, CategoryItem *cat) { QList newItems; while (!doc.atEnd()) { doc.readNext(); if (doc.isStartElement() && QLatin1String("station")==doc.name()) { newItems.append(new Item(QLatin1String("http://yp.shoutcast.com/sbin/tunein-station.pls?id=")+ doc.attributes().value("id").toString(), doc.attributes().value("name").toString(), cat)); } else if (doc.isEndElement() && QLatin1String("stationlist")==doc.name()) { return newItems; } } return newItems; } QList StreamsModel::parseDirbleResponse(QIODevice *dev, CategoryItem *cat, const QString &origUrl) { if (origUrl.contains("/v1/stations/apikey/")) { return parseDirbleStations(dev, cat); } QList newItems; #if QT_VERSION >= 0x050000 QVariantList data=QJsonDocument::fromJson(dev->readAll()).toVariant().toList(); #else QVariantList data = QJson::Parser().parse(dev->readAll()).toList(); #endif // Get categories... if (!data.isEmpty()) { foreach (const QVariant &d, data) { QVariantMap map = d.toMap(); newItems.append(new DirbleCategoryItem(QLatin1String("http://")+constDirbleHost+QLatin1String("/v1/childCategories/apikey/")+constDirbleApiKey+ QLatin1String("/primaryid/")+map["id"].toString(), map["name"].toString(), cat)); } } return newItems; } QList StreamsModel::parseDirbleStations(QIODevice *dev, CategoryItem *cat) { QList newItems; #if QT_VERSION >= 0x050000 QVariantList data=QJsonDocument::fromJson(dev->readAll()).toVariant().toList(); #else QVariantList data = QJson::Parser().parse(dev->readAll()).toList(); #endif foreach (const QVariant &d, data) { QVariantMap map = d.toMap(); map["country"]; newItems.append(new Item(map["streamurl"].toString(), map["name"].toString(), cat, map["bitrate"].toString())); } return newItems; } StreamsModel::Item * StreamsModel::parseRadioTimeEntry(QXmlStreamReader &doc, CategoryItem *parent, bool parseSubText) { Item *item=0; CategoryItem *cat=0; while (!doc.atEnd()) { if (doc.isStartElement()) { QString text=doc.attributes().value("text").toString(); if (!text.isEmpty()) { QString url=doc.attributes().value("URL").toString(); QString subText=parseSubText ? doc.attributes().value("subtext").toString() : QString(); bool isStation=QLatin1String("audio")==doc.attributes().value("type").toString(); if (isStation) { item=new Item(url, text, parent, subText); } else { cat=new CategoryItem(url, text, parent); cat->canBookmark=!url.isEmpty(); item=cat; } } } doc.readNext(); if (doc.isStartElement() && QLatin1String("outline")==doc.name()) { Item *child = parseRadioTimeEntry(doc, cat); if (child) { if (cat) { cat->state=CategoryItem::Fetched; cat->children.append(child); } else { delete child; } } } else if (doc.isEndElement() && QLatin1String("outline")==doc.name()) { break; } } return item; } StreamsModel::Item * StreamsModel::parseSomaFmEntry(QXmlStreamReader &doc, CategoryItem *parent) { QString name; QString url; QString streamFormat; while (!doc.atEnd()) { doc.readNext(); if (QXmlStreamReader::StartElement==doc.tokenType()) { QStringRef elem = doc.name(); if (QLatin1String("title")==elem) { name=doc.readElementText().trimmed(); } else if (QLatin1String("fastpls")==elem) { if (streamFormat.isEmpty() || QLatin1String("mp3")!=streamFormat) { streamFormat=doc.attributes().value("format").toString(); url=doc.readElementText(); } } } else if (doc.isEndElement() && QLatin1String("channel")==doc.name()) { break; } } return name.isEmpty() || url.isEmpty() ? 0 : new Item(url, name, parent); } void StreamsModel::importOldFavourites() { if (!favourites->importedOld) { if (Settings::self()->version()storeStreamsInMpdDir() ? MPDConnection::self()->getDetails().dir : Utils::dataDir(QString(), false); if (!prevFile.isEmpty()) { prevFile+="streams.xml.gz"; if (QFile::exists(prevFile)) { importIntoFavourites(prevFile); } } } favourites->importedOld=true; } } void StreamsModel::loadInstalledProviders() { QSet added; QString dir=Utils::dataDir(StreamsModel::constSubDir); QStringList subDirs=QDir(dir).entryList(QStringList() << "*", QDir::Dirs|QDir::Readable|QDir::NoDot|QDir::NoDotDot); QStringList streamFiles=QStringList() << constCompressedXmlFile << constXmlFile << constSettingsFile; foreach (const QString &sub, subDirs) { if (!added.contains(sub)) { foreach (const QString &streamFile, streamFiles) { if (QFile::exists(dir+sub+Utils::constDirSep+streamFile)) { addInstalledProvider(sub, getExternalIcon(dir+sub), dir+sub+Utils::constDirSep+streamFile, false); added.insert(sub); break; } } } } } StreamsModel::CategoryItem * StreamsModel::addInstalledProvider(const QString &name, const QIcon &icon, const QString &streamsFileName, bool replace) { CategoryItem *cat=0; if (streamsFileName.endsWith(constSettingsFile)) { QFile file(streamsFileName); if (file.open(QIODevice::ReadOnly)) { #if QT_VERSION >= 0x050000 QVariantMap map=QJsonDocument::fromJson(file.readAll()).toVariant().toMap(); #else QVariantMap map = QJson::Parser().parse(file.readAll()).toMap(); #endif QString type=map["type"].toString(); QString url=map["url"].toString(); if (!url.isEmpty() && !type.isEmpty()) { QStringList toRemove=QStringList() << " " << "." << "/" << "\\" << "(" << ")"; QString cacheName=name; foreach (const QString &rem, toRemove) { cacheName=cacheName.replace(rem, ""); } cacheName=cacheName.toLower(); if (type=="di") { cat=new DiCategoryItem(url, name, root, icon, cacheName); } else if (type=="soma") { cat=new SomaCategoryItem(url, name, root, icon, cacheName, map["modName"].toBool()); } else if (type=="listenlive") { cat=new ListenLiveCategoryItem(url, name, root, icon, cacheName); } } } } else { cat=new XmlCategoryItem(name, root, icon, streamsFileName); } if (!cat) { return 0; } cat->configName="x-"+name; if (replace) { // Remove any existing entry... removeInstalledProvider(cat->configName); } if (replace) { beginInsertRows(QModelIndex(), root->children.count(), root->children.count()); root->children.append(cat); endInsertRows(); } else { root->children.append(cat); } return cat; } void StreamsModel::removeInstalledProvider(const QString &key) { foreach (Item *i, root->children) { if (key==static_cast(i)->configName) { int row=root->children.indexOf(i); if (row>=0) { static_cast(i)->removeCache(); beginRemoveRows(QModelIndex(), row, row); delete root->children.takeAt(row); endRemoveRows(); } break; } } } QModelIndex StreamsModel::categoryIndex(const CategoryItem *cat) const { int row=root->children.indexOf(const_cast(cat)); return -1==row ? QModelIndex() : createIndex(row, 0, (void *)cat); } #endif // ENABLE_STREAMS const QString StreamsModel::constPrefix=QLatin1String("cantata-stream-"); const QString StreamsModel::constPlayListName=QLatin1String("[Radio Streams]"); QString StreamsModel::modifyUrl(const QString &u, bool addPrefix, const QString &name) { return MPDParseUtils::addStreamName(!addPrefix || !u.startsWith("http:") ? u : (constPrefix+u), name); }