2010-03-24 00:11:46 +01:00
|
|
|
/* This file is part of Clementine.
|
2010-11-20 14:27:10 +01:00
|
|
|
Copyright 2010, David Sansome <me@davidsansome.com>
|
2010-03-24 00:11:46 +01:00
|
|
|
|
|
|
|
Clementine 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.
|
|
|
|
|
|
|
|
Clementine 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 Clementine. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2010-03-01 16:40:12 +01:00
|
|
|
#include "m3uparser.h"
|
2011-04-22 18:50:29 +02:00
|
|
|
#include "core/logging.h"
|
2011-11-28 14:51:35 +01:00
|
|
|
#include "core/timeconstants.h"
|
2010-03-01 16:40:12 +01:00
|
|
|
|
2010-09-13 18:31:42 +02:00
|
|
|
#include <QBuffer>
|
2010-03-01 16:40:12 +01:00
|
|
|
#include <QtDebug>
|
|
|
|
|
2010-12-11 11:35:07 +01:00
|
|
|
M3UParser::M3UParser(LibraryBackendInterface* library, QObject* parent)
|
2014-02-07 16:34:20 +01:00
|
|
|
: ParserBase(library, parent) {}
|
2010-03-01 16:40:12 +01:00
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
SongList M3UParser::Load(QIODevice* device, const QString& playlist_path,
|
|
|
|
const QDir& dir) const {
|
2010-05-22 22:06:19 +02:00
|
|
|
SongList ret;
|
|
|
|
|
|
|
|
M3UType type = STANDARD;
|
|
|
|
Metadata current_metadata;
|
|
|
|
|
2010-09-13 18:31:42 +02:00
|
|
|
QString data = QString::fromUtf8(device->readAll());
|
|
|
|
data.replace('\r', '\n');
|
2010-10-11 17:20:53 +02:00
|
|
|
data.replace("\n\n", "\n");
|
2010-09-13 18:31:42 +02:00
|
|
|
QByteArray bytes = data.toUtf8();
|
|
|
|
QBuffer buffer(&bytes);
|
|
|
|
buffer.open(QIODevice::ReadOnly);
|
|
|
|
|
|
|
|
QString line = QString::fromUtf8(buffer.readLine()).trimmed();
|
2010-03-05 12:57:06 +01:00
|
|
|
if (line.startsWith("#EXTM3U")) {
|
2010-03-01 16:40:12 +01:00
|
|
|
// This is in extended M3U format.
|
2010-05-22 22:06:19 +02:00
|
|
|
type = EXTENDED;
|
2010-09-13 18:31:42 +02:00
|
|
|
line = QString::fromUtf8(buffer.readLine()).trimmed();
|
2010-03-01 16:40:12 +01:00
|
|
|
}
|
|
|
|
|
2010-03-05 12:57:06 +01:00
|
|
|
forever {
|
2010-03-01 16:40:12 +01:00
|
|
|
if (line.startsWith('#')) {
|
|
|
|
// Extended info or comment.
|
2010-05-22 22:06:19 +02:00
|
|
|
if (type == EXTENDED && line.startsWith("#EXT")) {
|
|
|
|
if (!ParseMetadata(line, ¤t_metadata)) {
|
2011-04-22 18:50:29 +02:00
|
|
|
qLog(Warning) << "Failed to parse metadata: " << line;
|
2010-03-01 16:40:12 +01:00
|
|
|
}
|
|
|
|
}
|
2011-03-17 17:05:08 +01:00
|
|
|
} else if (!line.isEmpty()) {
|
2011-04-28 14:27:53 +02:00
|
|
|
Song song = LoadSong(line, 0, dir);
|
2013-06-01 04:34:21 +02:00
|
|
|
if (!current_metadata.title.isEmpty()) {
|
|
|
|
song.set_title(current_metadata.title);
|
|
|
|
}
|
|
|
|
if (!current_metadata.artist.isEmpty()) {
|
|
|
|
song.set_artist(current_metadata.artist);
|
|
|
|
}
|
|
|
|
if (current_metadata.length > 0) {
|
|
|
|
song.set_length_nanosec(current_metadata.length);
|
|
|
|
}
|
2011-04-28 14:27:53 +02:00
|
|
|
ret << song;
|
2010-12-11 11:35:07 +01:00
|
|
|
|
2011-04-28 14:27:53 +02:00
|
|
|
current_metadata = Metadata();
|
2010-03-05 12:57:06 +01:00
|
|
|
}
|
2010-09-13 18:31:42 +02:00
|
|
|
if (buffer.atEnd()) {
|
2010-03-05 12:57:06 +01:00
|
|
|
break;
|
2010-03-01 16:40:12 +01:00
|
|
|
}
|
2010-09-13 18:31:42 +02:00
|
|
|
line = QString::fromUtf8(buffer.readLine()).trimmed();
|
2010-03-05 12:57:06 +01:00
|
|
|
}
|
2010-03-01 16:40:12 +01:00
|
|
|
|
2010-05-22 22:06:19 +02:00
|
|
|
return ret;
|
2010-03-01 16:40:12 +01:00
|
|
|
}
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
bool M3UParser::ParseMetadata(const QString& line,
|
|
|
|
M3UParser::Metadata* metadata) const {
|
2010-03-01 16:40:12 +01:00
|
|
|
// Extended info, eg.
|
|
|
|
// #EXTINF:123,Sample Artist - Sample title
|
|
|
|
QString info = line.section(':', 1);
|
|
|
|
QString l = info.section(',', 0, 0);
|
|
|
|
bool ok = false;
|
|
|
|
int length = l.toInt(&ok);
|
|
|
|
if (!ok) {
|
|
|
|
return false;
|
|
|
|
}
|
2011-02-14 20:34:37 +01:00
|
|
|
metadata->length = length * kNsecPerSec;
|
2010-05-23 17:35:28 +02:00
|
|
|
|
2010-03-01 16:40:12 +01:00
|
|
|
QString track_info = info.section(',', 1);
|
|
|
|
QStringList list = track_info.split('-');
|
|
|
|
if (list.size() <= 1) {
|
2010-05-23 17:35:28 +02:00
|
|
|
metadata->title = track_info;
|
|
|
|
return true;
|
2010-03-01 16:40:12 +01:00
|
|
|
}
|
|
|
|
metadata->artist = list[0].trimmed();
|
|
|
|
metadata->title = list[1].trimmed();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
void M3UParser::Save(const SongList& songs, QIODevice* device,
|
|
|
|
const QDir& dir) const {
|
2010-05-23 19:00:45 +02:00
|
|
|
device->write("#EXTM3U\n");
|
2014-02-07 16:34:20 +01:00
|
|
|
foreach(const Song & song, songs) {
|
2011-04-28 14:27:53 +02:00
|
|
|
if (song.url().isEmpty()) {
|
2010-05-23 19:00:45 +02:00
|
|
|
continue;
|
|
|
|
}
|
2011-02-13 19:34:30 +01:00
|
|
|
QString meta = QString("#EXTINF:%1,%2 - %3\n")
|
2014-02-07 16:34:20 +01:00
|
|
|
.arg(song.length_nanosec() / kNsecPerSec)
|
|
|
|
.arg(song.artist())
|
|
|
|
.arg(song.title());
|
2010-06-02 23:26:12 +02:00
|
|
|
device->write(meta.toUtf8());
|
2011-04-28 14:27:53 +02:00
|
|
|
device->write(URLOrRelativeFilename(song.url(), dir).toUtf8());
|
2010-05-23 19:00:45 +02:00
|
|
|
device->write("\n");
|
|
|
|
}
|
2010-05-22 22:06:19 +02:00
|
|
|
}
|
2010-06-15 15:24:17 +02:00
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
bool M3UParser::TryMagic(const QByteArray& data) const {
|
2010-06-15 15:24:17 +02:00
|
|
|
return data.contains("#EXTM3U") || data.contains("#EXTINF");
|
|
|
|
}
|