8#include <c++utilities/conversion/stringconversion.h>
9#include <c++utilities/io/binarywriter.h>
90 for (
auto i = range.first; i != range.second; ++i) {
91 const auto &extendedField = i->second;
92 if (extendedId.matches(extendedField)) {
93 values.emplace_back(&extendedField.value());
94 for (
const auto &additionalData : extendedField.additionalData()) {
95 values.emplace_back(&additionalData.value);
111 for (
auto i = range.first; i != range.second; ++i) {
112 if (i->second.mean() == mean && i->second.name() == name) {
113 return i->second.value();
121 using namespace Mp4TagAtomIds;
195 using namespace Mp4TagAtomIds;
207 case PreDefinedGenre:
273 if (!field.value().isEmpty()) {
274 values.emplace_back(&field.value());
276 for (
const auto &
value : field.additionalData()) {
310 auto valuesIterator =
values.cbegin();
312 for (; valuesIterator !=
values.cend() && range.first != range.second;) {
313 if (!valuesIterator->isEmpty()) {
314 auto &extendedField = range.first->second;
315 if (extendedId.matches(extendedField) && (!extendedId.updateOnly || !extendedField.value().isEmpty())) {
316 extendedField.clearValue();
317 extendedField.setValue(*valuesIterator);
328 for (; valuesIterator !=
values.cend(); ++valuesIterator) {
329 if (valuesIterator->isEmpty()) {
331 std::forward_as_tuple(extendedId.mean, extendedId.name, *valuesIterator));
334 for (; range.first != range.second; ++range.first) {
335 range.first->second.clearValue();
351 for (
auto i = range.first; i != range.second; ++i) {
352 if (i->second.mean() == mean && i->second.name() == name) {
353 i->second.setValue(
value);
380 static const string context(
"parsing MP4 tag");
382 istream &stream = metaAtom.
container().stream();
383 BinaryReader &reader = metaAtom.
container().reader();
384 if (metaAtom.
totalSize() > numeric_limits<std::uint32_t>::max()) {
392 diag.emplace_back(
DiagLevel::Critical,
"Unable to parse child atoms of meta atom (stores hdlr and ilst atoms).", context);
396 int versionByte = reader.readByte();
397 if (versionByte != 0) {
400 if (reader.readUInt24BE()) {
403 if (reader.readInt32BE()) {
404 diag.emplace_back(
DiagLevel::Warning,
"Predefined 32-bit integer (hdlr atom) isn't set to 0.", context);
406 std::uint64_t handlerType = reader.readUInt64BE();
407 if ( (handlerType != 0x6d6469726170706c)) {
408 diag.emplace_back(
DiagLevel::Warning,
"Handler type (value in hdlr atom) is unknown. Trying to parse meta information anyhow.", context);
417 diag.emplace_back(
DiagLevel::Critical,
"Unable to parse child atoms of meta atom (stores hdlr and ilst atoms).", context);
420 diag.emplace_back(
DiagLevel::Warning,
"No ilst atom found (stores attached meta information).", context);
427 tagField.
reparse(*child, diag);
428 fields().emplace(child->id(), std::move(tagField));
481 m_omitPreDefinedGenre(m_tag.fields().count(m_tag.hasField(Mp4TagAtomIds::
Genre)))
483 m_maker.reserve(m_tag.fields().size());
484 for (
auto &field : m_tag.fields()) {
485 if (!field.second.value().isEmpty() && (!m_omitPreDefinedGenre || field.first != Mp4TagAtomIds::PreDefinedGenre)) {
487 m_ilstSize += m_maker.emplace_back(field.second.prepareMaking(diag)).requiredSize();
488 } catch (const Failure &) {
492 if (m_ilstSize != 8) {
493 m_metaSize += m_ilstSize;
495 if (m_metaSize >= numeric_limits<std::uint32_t>::max()) {
496 diag.emplace_back(
DiagLevel::Critical,
"Making such big tags is not implemented.",
"making MP4 tag");
497 throw NotImplementedException();
511 BinaryWriter writer(&stream);
512 writer.writeUInt32BE(
static_cast<std::uint32_t
>(m_metaSize));
513 writer.writeUInt32BE(Mp4AtomIds::Meta);
515 static const std::uint8_t hdlrData[37] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x21, 0x68, 0x64, 0x6C, 0x72, 0x00, 0x00, 0x00, 0x00, 0x00,
516 0x00, 0x00, 0x00, 0x6D, 0x64, 0x69, 0x72, 0x61, 0x70, 0x70, 0x6C, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
517 stream.write(
reinterpret_cast<const char *
>(hdlrData),
sizeof(hdlrData));
518 if (m_ilstSize != 8) {
520 writer.writeUInt32BE(
static_cast<std::uint32_t
>(m_ilstSize));
521 writer.writeUInt32BE(Mp4AtomIds::ItunesList);
523 for (
auto &maker : m_maker) {
528 diag.emplace_back(DiagLevel::Warning,
"Tag is empty.",
"making MP4 tag");
The Diagnostics class is a container for DiagMessage.
The class inherits from std::exception and serves as base class for exceptions thrown by the elements...
bool setValue(const IdentifierType &id, const TagValue &value)
Assigns the given value to the field with the specified id.
bool hasField(KnownField field) const
Returns an indication whether the specified field is present.
typename FieldMapBasedTagTraits< Mp4Tag >::FieldType::IdentifierType IdentifierType
const TagValue & value(const IdentifierType &id) const
Returns the value of the field with the specified id.
const std::multimap< IdentifierType, FieldType, Compare > & fields() const
typename FieldMapBasedTagTraits< Mp4Tag >::FieldType FieldType
std::vector< const TagValue * > values(const IdentifierType &id) const
Returns the values of the field with the specified id.
bool setValues(const IdentifierType &id, const std::vector< TagValue > &values)
Assigns the given values to the field with the specified id.
std::uint64_t startOffset() const
Returns the start offset in the related stream.
std::uint32_t headerSize() const
Returns the header size of the element in byte.
ImplementationType * childById(const IdentifierType &id, Diagnostics &diag)
Returns the first child with the specified id.
ImplementationType * nextSibling()
Returns the next sibling of the element.
ImplementationType * firstChild()
Returns the first child of the element.
std::uint64_t totalSize() const
Returns the total size of the element.
ContainerType & container()
Returns the related container.
The Mp4Atom class helps to parse MP4 files.
The Mp4TagField class is used by Mp4Tag to store the fields.
void reparse(Mp4Atom &ilstChild, Diagnostics &diag)
Parses field information from the specified Mp4Atom.
The Mp4TagMaker class helps writing MP4 tags.
void make(std::ostream &stream, Diagnostics &diag)
Saves the tag (specified when constructing the object) to the specified stream.
Implementation of TagParser::Tag for the MP4 container.
std::vector< const TagValue * > values(KnownField field) const override
Returns the values of the specified field.
const TagValue & value(KnownField value) const override
Returns the value of the specified field.
bool hasField(KnownField value) const override
Returns an indication whether the specified field is present.
bool setValue(KnownField field, const TagValue &value) override
Assigns the given value to the specified field.
IdentifierType internallyGetFieldId(KnownField field) const
Mp4TagMaker prepareMaking(Diagnostics &diag)
Prepares making.
bool setValues(KnownField field, const std::vector< TagValue > &values) override
Assigns the given values to the field with the specified id.
void parse(Mp4Atom &metaAtom, Diagnostics &diag)
Parses tag information from the specified metaAtom.
void make(std::ostream &stream, Diagnostics &diag)
Writes tag information to the specified stream.
bool canEncodingBeUsed(TagTextEncoding encoding) const override
Returns an indication whether the specified encoding can be used to provide string values for the tag...
KnownField internallyGetKnownField(const IdentifierType &id) const
void internallyGetValuesFromField(const FieldType &field, std::vector< const TagValue * > &values) const
Adds values from additional data atoms as well.
The exception that is thrown when the data to be parsed holds no parsable information (e....
This exception is thrown when the an operation is invoked that has not been implemented yet.
The TagValue class wraps values of different types.
TagDataType type() const
Returns the type of the assigned value.
static const TagValue & empty()
Returns a default-constructed TagValue where TagValue::isNull() and TagValue::isEmpty() both return t...
bool isEmpty() const
Returns whether no or an empty value is assigned.
Contains all classes and functions of the TagInfo library.
KnownField
Specifies the field.
TagTextEncoding
Specifies the text encoding.
The Mp4ExtendedFieldId specifies parameter for an extended field denoted via Mp4TagAtomIds::Extended.
bool updateOnly
Whether only existing fields should be updated but no new extended field should be created.
std::string_view mean
mean parameter, usually Mp4TagExtendedMeanIds::iTunes
Mp4ExtendedFieldId(std::string_view mean, std::string_view name, bool updateOnly=false)
Constructs a new instance with the specified parameter.
std::string_view name
name parameter