/***************************************************************************
* Copyright (C) 2009-2018 by Ilya Kotov *
* forkotov02@ya.ru *
* *
* 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; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
#include <QtGlobal>
#include <QBuffer>
#include <taglib/tag.h>
#include <taglib/fileref.h>
#include <taglib/vorbisfile.h>
#include <taglib/xiphcomment.h>
#include <taglib/tmap.h>
#include <taglib/flacpicture.h>
#include "vorbismetadatamodel.h"
VorbisMetaDataModel::VorbisMetaDataModel(const QString &path, bool readOnly)
#ifdef HAS_PICTURE_LIST
: MetaDataModel(readOnly, MetaDataModel::IS_COVER_EDITABLE)
#else
: MetaDataModel(readOnly)
#endif
{
m_path = path;
m_stream = new TagLib::FileStream(QStringToFileName(path), readOnly);
m_file = new TagLib::Ogg::Vorbis::File(m_stream);
m_tag = m_file->tag();
m_tags << new VorbisCommentModel(this);
}
VorbisMetaDataModel::~VorbisMetaDataModel()
{
while(!m_tags.isEmpty())
delete m_tags.takeFirst();
delete m_file;
delete m_stream;
}
QList<TagModel* > VorbisMetaDataModel::tags() const
{
return m_tags;
}
QPixmap VorbisMetaDataModel::cover() const
{
if(!m_tag || m_tag->isEmpty())
return QPixmap();
#ifdef HAS_PICTURE_LIST
TagLib::List<TagLib::FLAC::Picture *> list = m_tag->pictureList();
for(uint i = 0; i < list.size(); ++i)
{
if(list[i]->type() == TagLib::FLAC::Picture::FrontCover)
{
QPixmap cover;
cover.loadFromData(QByteArray(list[i]->data().data(), list[i]->data().size())); //read binary picture data
return cover;
}
}
#else
TagLib::StringList list = m_tag->fieldListMap()["METADATA_BLOCK_PICTURE"];
if(list.isEmpty())
return QPixmap();
for(uint i = 0; i < list.size(); ++i)
{
TagLib::FLAC::Picture pict;
TagLib::String value = list[i];
QByteArray block = QByteArray::fromBase64(TStringToQString(value).toLatin1());
pict.parse(TagLib::ByteVector(block.constData(), block.size()));
QPixmap cover;
cover.loadFromData(QByteArray(pict.data().data(), pict.data().size())); //read binary picture data
return cover;
}
#endif
return QPixmap();
}
#ifdef HAS_PICTURE_LIST
void VorbisMetaDataModel::setCover(const QPixmap &pix)
{
removeCover();
if(m_tag)
{
TagLib::FLAC::Picture *picture = new TagLib::FLAC::Picture();
picture->setType(TagLib::FLAC::Picture::FrontCover);
QByteArray data;
QBuffer buffer(&data);
buffer.open(QIODevice::WriteOnly);
pix.save(&buffer, "JPEG");
picture->setMimeType("image/jpeg");
picture->setData(TagLib::ByteVector(data.constData(), data.size()));
m_tag->addPicture(picture);
m_file->save();
}
}
void VorbisMetaDataModel::removeCover()
{
if(m_tag && !m_tag->isEmpty())
{
bool save = false;
TagLib::List<TagLib::FLAC::Picture *> list = m_tag->pictureList();
for(uint i = 0; i < list.size(); ++i)
{
if(list[i]->type() == TagLib::FLAC::Picture::FrontCover)
{
m_tag->removePicture(list[i], false);
save = true;
}
}
if(save)
{
m_file->save();
}
}
}
#endif
VorbisCommentModel::VorbisCommentModel(VorbisMetaDataModel *model) : TagModel(TagModel::Save)
{
m_model = model;
}
VorbisCommentModel::~VorbisCommentModel()
{}
QString VorbisCommentModel::name() const
{
return "Vorbis Comment";
}
QString VorbisCommentModel::value(Qmmp::MetaData key) const
{
if(!m_model->m_tag || m_model->m_tag->isEmpty())
return QString();
TagLib::Ogg::XiphComment *tag = m_model->m_tag;
switch((int) key)
{
case Qmmp::TITLE:
return TStringToQString(tag->title());
case Qmmp::ARTIST:
return TStringToQString(tag->artist());
case Qmmp::ALBUMARTIST:
if(tag->fieldListMap()["ALBUMARTIST"].isEmpty())
return QString();
else
return TStringToQString(tag->fieldListMap()["ALBUMARTIST"].front());
case Qmmp::ALBUM:
return TStringToQString(tag->album());
case Qmmp::COMMENT:
return TStringToQString(tag->comment());
case Qmmp::GENRE:
return TStringToQString(tag->genre());
case Qmmp::COMPOSER:
if(tag->fieldListMap()["COMPOSER"].isEmpty())
return QString();
else
return TStringToQString(tag->fieldListMap()["COMPOSER"].front());
case Qmmp::YEAR:
return QString::number(tag->year());
case Qmmp::TRACK:
return QString::number(tag->track());
case Qmmp::DISCNUMBER:
if(tag->fieldListMap()["DISCNUMBER"].isEmpty())
return QString();
else
return TStringToQString(tag->fieldListMap()["DISCNUMBER"].front());
}
return QString();
}
void VorbisCommentModel::setValue(Qmmp::MetaData key, const QString &value)
{
if(!m_model->m_tag || m_model->m_tag->isEmpty())
return;
TagLib::Ogg::XiphComment *tag = m_model->m_tag;
TagLib::String str = QStringToTString(value);
switch((int) key)
{
case Qmmp::TITLE:
tag->setTitle(str);
return;
case Qmmp::ARTIST:
tag->setArtist(str);
return;
case Qmmp::ALBUM:
tag->setAlbum(str);
return;
case Qmmp::ALBUMARTIST:
tag->addField("ALBUMARTIST", str, true);
return;
case Qmmp::COMMENT:
tag->setComment(str);
return;
case Qmmp::GENRE:
tag->setGenre(str);
return;
case Qmmp::COMPOSER:
tag->addField("COMPOSER", str, true);
return;
case Qmmp::TRACK:
tag->setTrack(value.toInt());
return;
case Qmmp::YEAR:
tag->setYear(value.toInt());
return;
case Qmmp::DISCNUMBER:
value == "0" ?
tag->removeField("DISCNUMBER"):
tag->addField("DISCNUMBER", str, true);
}
}
void VorbisCommentModel::save()
{
if(m_model->m_tag)
m_model->m_file->save();
#if ((TAGLIB_MAJOR_VERSION == 1) && (TAGLIB_MINOR_VERSION <= 10))
//taglib bug workarround
delete m_model->m_file;
m_model->m_file = new TagLib::Ogg::Vorbis::File(QStringToFileName(m_model->m_path));
m_model->m_tag = m_model->m_file->tag();
#endif
}