/* * Strawberry Music Player * This file was part of Clementine. * Copyright 2010, David Sansome * * Strawberry 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. * * Strawberry 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 Strawberry. If not, see . * */ #include #include #include #include #include #include #include #include #include #include #include "includes/shared_ptr.h" #include "core/settings.h" #include "utilities/xmlutils.h" #include "constants/timeconstants.h" #include "constants/playlistsettings.h" #include "xmlparser.h" #include "xspfparser.h" using namespace Qt::Literals::StringLiterals; class CollectionBackendInterface; XSPFParser::XSPFParser(const SharedPtr tagreader_client, const SharedPtr collection_backend, QObject *parent) : XMLParser(tagreader_client, collection_backend, parent) {} ParserBase::LoadResult XSPFParser::Load(QIODevice *device, const QString &playlist_path, const QDir &dir, const bool collection_lookup) const { Q_UNUSED(playlist_path); QString playlist_name; { QXmlStreamReader reader(device); if (Utilities::ParseUntilElement(&reader, u"playlist"_s) && Utilities::ParseUntilElement(&reader, u"title"_s)) { playlist_name = reader.readElementText(); } } device->seek(0); QXmlStreamReader reader(device); if (!Utilities::ParseUntilElement(&reader, u"playlist"_s)) { return LoadResult(); } if (!Utilities::ParseUntilElement(&reader, u"trackList"_s)) { return LoadResult(); } SongList songs; while (!reader.atEnd() && Utilities::ParseUntilElement(&reader, u"track"_s)) { const Song song = ParseTrack(&reader, dir, collection_lookup); if (song.is_valid()) { songs << song; } } return LoadResult(songs, playlist_name); } Song XSPFParser::ParseTrack(QXmlStreamReader *reader, const QDir &dir, const bool collection_lookup) const { QString title, artist, album, location, art; qint64 nanosec = -1; int track_num = -1; while (!reader->atEnd()) { QXmlStreamReader::TokenType type = reader->readNext(); QString name = reader->name().toString(); switch (type) { case QXmlStreamReader::StartElement:{ if (name == "location"_L1) { location = QUrl::fromPercentEncoding(reader->readElementText().toUtf8()); } else if (name == "title"_L1) { title = reader->readElementText(); } else if (name == "creator"_L1) { artist = reader->readElementText(); } else if (name == "album"_L1) { album = reader->readElementText(); } else if (name == "image"_L1) { art = QUrl::fromPercentEncoding(reader->readElementText().toUtf8()); } else if (name == "duration"_L1) { // in milliseconds. const QString duration = reader->readElementText(); bool ok = false; nanosec = duration.toInt(&ok) * kNsecPerMsec; if (!ok) { nanosec = -1; } } else if (name == "trackNum"_L1) { const QString track_num_str = reader->readElementText(); bool ok = false; track_num = track_num_str.toInt(&ok); if (!ok || track_num < 1) { track_num = -1; } } else if (name == "info"_L1) { // TODO: Do something with extra info? } break; } case QXmlStreamReader::EndElement:{ if (name == "track"_L1) { goto return_song; } } default: break; } } return_song: Song song = LoadSong(location, 0, track_num, dir, collection_lookup); // Override metadata with what was in the playlist if (song.source() != Song::Source::Collection) { if (!title.isEmpty()) song.set_title(title); if (!artist.isEmpty()) song.set_artist(artist); if (!album.isEmpty()) song.set_album(album); if (!art.isEmpty()) song.set_art_manual(QUrl(art)); if (nanosec > 0) song.set_length_nanosec(nanosec); if (track_num > 0) song.set_track(track_num); } return song; } void XSPFParser::Save(const QString &playlist_name, const SongList &songs, QIODevice *device, const QDir &dir, const PlaylistSettings::PathType path_type) const { QXmlStreamWriter writer(device); writer.setAutoFormatting(true); writer.setAutoFormattingIndent(2); writer.writeStartDocument(); StreamElement playlist(u"playlist"_s, &writer); writer.writeAttribute("version"_L1, "1"_L1); writer.writeDefaultNamespace("http://xspf.org/ns/0/"_L1); writer.writeTextElement("title"_L1, playlist_name); Settings s; s.beginGroup(PlaylistSettings::kSettingsGroup); bool write_metadata = s.value(PlaylistSettings::kWriteMetadata, true).toBool(); s.endGroup(); StreamElement tracklist(u"trackList"_s, &writer); for (const Song &song : songs) { QString filename_or_url = QString::fromLatin1(QUrl::toPercentEncoding(URLOrFilename(song.url(), dir, path_type), "/ ")); StreamElement track(u"track"_s, &writer); writer.writeTextElement("location"_L1, filename_or_url); if (write_metadata || (song.is_stream() && !song.is_radio())) { writer.writeTextElement("title"_L1, song.title()); if (!song.artist().isEmpty()) { writer.writeTextElement("creator"_L1, song.artist()); } if (!song.album().isEmpty()) { writer.writeTextElement("album"_L1, song.album()); } if (song.length_nanosec() != -1) { writer.writeTextElement("duration"_L1, QString::number(song.length_nanosec() / kNsecPerMsec)); } } if ((write_metadata || song.has_cue() || (song.is_stream() && !song.is_radio())) && song.track() > 0) { writer.writeTextElement("trackNum"_L1, QString::number(song.track())); } if (write_metadata || (song.is_stream() && !song.is_radio())) { const QUrl cover_url = song.art_manual().isEmpty() || !song.art_manual().isValid() ? song.art_automatic() : song.art_manual(); // Ignore images that are in our resource bundle. if (!cover_url.isEmpty() && cover_url.isValid()) { const QString cover_filename = QString::fromLatin1(QUrl::toPercentEncoding(URLOrFilename(cover_url, dir, path_type), "/ ")); writer.writeTextElement("image"_L1, cover_filename); } } } writer.writeEndDocument(); } bool XSPFParser::TryMagic(const QByteArray &data) const { return data.contains("