strawberry-audio-player-win.../3rdparty/taglib/mpeg/id3v2/frames/chapterframe.cpp

294 lines
8.4 KiB
C++
Raw Normal View History

2018-05-10 15:29:28 +02:00
/***************************************************************************
copyright : (C) 2013 by Lukas Krejci
email : krejclu6@fel.cvut.cz
***************************************************************************/
/***************************************************************************
* This library is free software; you can redistribute it and/or modify *
* it under the terms of the GNU Lesser General Public License version *
* 2.1 as published by the Free Software Foundation. *
* *
* This library 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 *
* Lesser General Public License for more details. *
* *
* You should have received a copy of the GNU Lesser General Public *
* License along with this library; if not, write to the Free Software *
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA *
* 02110-1301 USA *
* *
* Alternatively, this file is available under the Mozilla Public *
* License Version 1.1. You may obtain a copy of the License at *
* http://www.mozilla.org/MPL/ *
***************************************************************************/
2020-06-14 23:54:18 +02:00
#include <cstdio>
2020-06-26 23:30:30 +02:00
#include "tbytevectorlist.h"
#include "tpropertymap.h"
#include "tdebug.h"
2018-05-10 15:29:28 +02:00
#include "chapterframe.h"
using namespace Strawberry_TagLib::TagLib;
2018-05-10 15:29:28 +02:00
using namespace ID3v2;
2020-06-13 19:02:42 +02:00
class ChapterFrame::ChapterFramePrivate {
public:
2020-06-26 23:30:30 +02:00
explicit ChapterFramePrivate() : tagHeader(nullptr),
startTime(0),
endTime(0),
startOffset(0),
endOffset(0) {
2018-05-10 15:29:28 +02:00
embeddedFrameList.setAutoDelete(true);
}
const ID3v2::Header *tagHeader;
ByteVector elementID;
unsigned int startTime;
unsigned int endTime;
unsigned int startOffset;
unsigned int endOffset;
FrameListMap embeddedFrameListMap;
FrameList embeddedFrameList;
};
////////////////////////////////////////////////////////////////////////////////
// public methods
////////////////////////////////////////////////////////////////////////////////
ChapterFrame::ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data) : ID3v2::Frame(data), d(new ChapterFramePrivate()) {
2018-05-10 15:29:28 +02:00
d->tagHeader = tagHeader;
setData(data);
}
ChapterFrame::ChapterFrame(const ByteVector &elementID, unsigned int startTime, unsigned int endTime, unsigned int startOffset, unsigned int endOffset, const FrameList &embeddedFrames) : ID3v2::Frame("CHAP"), d(new ChapterFramePrivate()) {
// setElementID has a workaround for a previously silly API where you had to specifically include the null byte.
2018-05-10 15:29:28 +02:00
setElementID(elementID);
d->startTime = startTime;
d->endTime = endTime;
d->startOffset = startOffset;
d->endOffset = endOffset;
2020-06-13 19:02:42 +02:00
for (FrameList::ConstIterator it = embeddedFrames.begin();
it != embeddedFrames.end();
++it)
2018-05-10 15:29:28 +02:00
addEmbeddedFrame(*it);
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
ChapterFrame::~ChapterFrame() {
2018-05-10 15:29:28 +02:00
delete d;
}
2020-06-13 19:02:42 +02:00
ByteVector ChapterFrame::elementID() const {
2018-05-10 15:29:28 +02:00
return d->elementID;
}
2020-06-13 19:02:42 +02:00
unsigned int ChapterFrame::startTime() const {
2018-05-10 15:29:28 +02:00
return d->startTime;
}
2020-06-13 19:02:42 +02:00
unsigned int ChapterFrame::endTime() const {
2018-05-10 15:29:28 +02:00
return d->endTime;
}
2020-06-13 19:02:42 +02:00
unsigned int ChapterFrame::startOffset() const {
2018-05-10 15:29:28 +02:00
return d->startOffset;
}
2020-06-13 19:02:42 +02:00
unsigned int ChapterFrame::endOffset() const {
2018-05-10 15:29:28 +02:00
return d->endOffset;
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::setElementID(const ByteVector &eID) {
2018-05-10 15:29:28 +02:00
d->elementID = eID;
2020-06-13 19:02:42 +02:00
if (d->elementID.endsWith(char(0)))
2018-05-10 15:29:28 +02:00
d->elementID = d->elementID.mid(0, d->elementID.size() - 1);
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::setStartTime(const unsigned int &sT) {
2018-05-10 15:29:28 +02:00
d->startTime = sT;
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::setEndTime(const unsigned int &eT) {
2018-05-10 15:29:28 +02:00
d->endTime = eT;
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::setStartOffset(const unsigned int &sO) {
2018-05-10 15:29:28 +02:00
d->startOffset = sO;
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::setEndOffset(const unsigned int &eO) {
2018-05-10 15:29:28 +02:00
d->endOffset = eO;
}
2020-06-13 19:02:42 +02:00
const FrameListMap &ChapterFrame::embeddedFrameListMap() const {
2018-05-10 15:29:28 +02:00
return d->embeddedFrameListMap;
}
2020-06-13 19:02:42 +02:00
const FrameList &ChapterFrame::embeddedFrameList() const {
2018-05-10 15:29:28 +02:00
return d->embeddedFrameList;
}
2020-06-13 19:02:42 +02:00
const FrameList &ChapterFrame::embeddedFrameList(const ByteVector &frameID) const {
2018-05-10 15:29:28 +02:00
return d->embeddedFrameListMap[frameID];
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::addEmbeddedFrame(Frame *frame) {
2018-05-10 15:29:28 +02:00
d->embeddedFrameList.append(frame);
d->embeddedFrameListMap[frame->frameID()].append(frame);
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::removeEmbeddedFrame(Frame *frame, bool del) {
2018-05-10 15:29:28 +02:00
// remove the frame from the frame list
FrameList::Iterator it = d->embeddedFrameList.find(frame);
d->embeddedFrameList.erase(it);
// ...and from the frame list map
it = d->embeddedFrameListMap[frame->frameID()].find(frame);
d->embeddedFrameListMap[frame->frameID()].erase(it);
// ...and delete as desired
2020-06-13 19:02:42 +02:00
if (del)
2018-05-10 15:29:28 +02:00
delete frame;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::removeEmbeddedFrames(const ByteVector &id) {
2018-05-10 15:29:28 +02:00
FrameList l = d->embeddedFrameListMap[id];
2020-06-13 19:02:42 +02:00
for (FrameList::ConstIterator it = l.begin(); it != l.end(); ++it)
2018-05-10 15:29:28 +02:00
removeEmbeddedFrame(*it, true);
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
String ChapterFrame::toString() const {
2018-05-10 15:29:28 +02:00
String s = String(d->elementID) +
2020-06-13 19:02:42 +02:00
": start time: " + String::number(d->startTime) +
", end time: " + String::number(d->endTime);
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
if (d->startOffset != 0xFFFFFFFF)
2018-05-10 15:29:28 +02:00
s += ", start offset: " + String::number(d->startOffset);
2020-06-13 19:02:42 +02:00
if (d->endOffset != 0xFFFFFFFF)
2018-05-10 15:29:28 +02:00
s += ", end offset: " + String::number(d->endOffset);
2020-06-13 19:02:42 +02:00
if (!d->embeddedFrameList.isEmpty()) {
2018-05-10 15:29:28 +02:00
StringList frameIDs;
2020-06-13 19:02:42 +02:00
for (FrameList::ConstIterator it = d->embeddedFrameList.begin();
it != d->embeddedFrameList.end();
++it)
2018-05-10 15:29:28 +02:00
frameIDs.append((*it)->frameID());
s += ", sub-frames: [ " + frameIDs.toString(", ") + " ]";
}
return s;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
PropertyMap ChapterFrame::asProperties() const {
2018-05-10 15:29:28 +02:00
PropertyMap map;
map.unsupportedData().append(frameID() + String("/") + d->elementID);
return map;
2018-05-10 15:29:28 +02:00
}
ChapterFrame *ChapterFrame::findByElementID(const ID3v2::Tag *tag, const ByteVector &eID) { // static
2018-05-10 15:29:28 +02:00
ID3v2::FrameList comments = tag->frameList("CHAP");
2020-06-13 19:02:42 +02:00
for (ID3v2::FrameList::ConstIterator it = comments.begin();
it != comments.end();
++it) {
2018-05-10 15:29:28 +02:00
ChapterFrame *frame = dynamic_cast<ChapterFrame *>(*it);
2020-06-13 19:02:42 +02:00
if (frame && frame->elementID() == eID)
2018-05-10 15:29:28 +02:00
return frame;
}
2019-04-21 21:39:11 +02:00
return nullptr;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
void ChapterFrame::parseFields(const ByteVector &data) {
2020-06-26 23:30:30 +02:00
size_t size = data.size();
2020-06-13 19:02:42 +02:00
if (size < 18) {
2018-05-10 15:29:28 +02:00
debug("A CHAP frame must contain at least 18 bytes (1 byte element ID "
"terminated by null and 4x4 bytes for start and end time and offset).");
return;
}
2020-06-26 23:30:30 +02:00
size_t pos = 0;
size_t embPos = 0;
d->elementID = readStringField(data, String::Latin1, pos).data(String::Latin1);
d->startTime = data.toUInt32BE(pos);
2018-05-10 15:29:28 +02:00
pos += 4;
2020-06-26 23:30:30 +02:00
d->endTime = data.toUInt32BE(pos);
2018-05-10 15:29:28 +02:00
pos += 4;
2020-06-26 23:30:30 +02:00
d->startOffset = data.toUInt32BE(pos);
2018-05-10 15:29:28 +02:00
pos += 4;
2020-06-26 23:30:30 +02:00
d->endOffset = data.toUInt32BE(pos);
2018-05-10 15:29:28 +02:00
pos += 4;
size -= pos;
// Embedded frames are optional
2020-06-13 19:02:42 +02:00
if (size < header()->size())
2018-05-10 15:29:28 +02:00
return;
2020-06-13 19:02:42 +02:00
while (embPos < size - header()->size()) {
2019-09-20 18:07:13 +02:00
Frame *frame = FrameFactory::instance()->createFrame(data.mid(pos + embPos), d->tagHeader);
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
if (!frame)
2018-05-10 15:29:28 +02:00
return;
// Checks to make sure that frame parsed correctly.
2020-06-13 19:02:42 +02:00
if (frame->size() <= 0) {
2018-05-10 15:29:28 +02:00
delete frame;
return;
}
embPos += frame->size() + header()->size();
addEmbeddedFrame(frame);
}
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
ByteVector ChapterFrame::renderFields() const {
2018-05-10 15:29:28 +02:00
ByteVector data;
data.append(d->elementID);
data.append('\0');
2020-06-26 23:30:30 +02:00
data.append(ByteVector::fromUInt32BE(d->startTime));
data.append(ByteVector::fromUInt32BE(d->endTime));
data.append(ByteVector::fromUInt32BE(d->startOffset));
data.append(ByteVector::fromUInt32BE(d->endOffset));
2018-05-10 15:29:28 +02:00
FrameList l = d->embeddedFrameList;
2020-06-13 19:02:42 +02:00
for (FrameList::ConstIterator it = l.begin(); it != l.end(); ++it)
2018-05-10 15:29:28 +02:00
data.append((*it)->render());
return data;
2018-05-10 15:29:28 +02:00
}
ChapterFrame::ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data, Header *h) : Frame(h), d(new ChapterFramePrivate()) {
2018-05-10 15:29:28 +02:00
d->tagHeader = tagHeader;
parseFields(fieldData(data));
2018-05-10 15:29:28 +02:00
}