8#include <c++utilities/conversion/stringconversion.h>
9#include <c++utilities/io/binarywriter.h>
64 if (!
value.isEmpty()) {
74 if (!
value.isEmpty()) {
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();
184 return LongDescription;
209 case PreDefinedGenre:
260 case LongDescription:
277 if (!field.value().isEmpty()) {
278 values.emplace_back(&field.value());
280 for (
const auto &
value : field.additionalData()) {
281 if (!
value.value.isEmpty()) {
291 switch (
value.type()) {
314 auto valuesIterator =
values.cbegin();
316 for (; valuesIterator !=
values.cend() && range.first != range.second;) {
317 if (!valuesIterator->isEmpty()) {
318 auto &extendedField = range.first->second;
319 if (extendedId.matches(extendedField) && (!extendedId.updateOnly || !extendedField.value().isEmpty())) {
320 extendedField.clearValue();
321 extendedField.setValue(*valuesIterator);
332 for (; valuesIterator !=
values.cend(); ++valuesIterator) {
333 if (valuesIterator->isEmpty()) {
335 std::forward_as_tuple(extendedId.mean, extendedId.name, *valuesIterator));
338 for (; range.first != range.second; ++range.first) {
339 range.first->second.clearValue();
355 for (
auto i = range.first; i != range.second; ++i) {
356 if (i->second.mean() == mean && i->second.name() == name) {
357 i->second.setValue(
value);
384 static const string context(
"parsing MP4 tag");
386 istream &stream = metaAtom.
container().stream();
387 BinaryReader &reader = metaAtom.
container().reader();
388 if (metaAtom.
totalSize() > numeric_limits<std::uint32_t>::max()) {
396 diag.emplace_back(
DiagLevel::Critical,
"Unable to parse child atoms of meta atom (stores hdlr and ilst atoms).", context);
400 int versionByte = reader.readByte();
401 if (versionByte != 0) {
404 if (reader.readUInt24BE()) {
407 if (reader.readInt32BE()) {
408 diag.emplace_back(
DiagLevel::Warning,
"Predefined 32-bit integer (hdlr atom) isn't set to 0.", context);
410 std::uint64_t handlerType = reader.readUInt64BE();
411 if ( (handlerType != 0x6d6469726170706c)) {
412 diag.emplace_back(
DiagLevel::Warning,
"Handler type (value in hdlr atom) is unknown. Trying to parse meta information anyhow.", context);
421 diag.emplace_back(
DiagLevel::Critical,
"Unable to parse child atoms of meta atom (stores hdlr and ilst atoms).", context);
424 diag.emplace_back(
DiagLevel::Warning,
"No ilst atom found (stores attached meta information).", context);
431 tagField.
reparse(*child, diag);
432 fields().emplace(child->id(), std::move(tagField));
487 m_maker.reserve(m_tag.fields().size());
488 for (
auto &field : m_tag.fields()) {
489 if (!field.second.value().isEmpty() && (!m_omitPreDefinedGenre || field.first != Mp4TagAtomIds::PreDefinedGenre)) {
491 m_ilstSize += m_maker.emplace_back(field.second.prepareMaking(diag)).requiredSize();
492 } catch (const Failure &) {
496 if (m_ilstSize != 8) {
497 m_metaSize += m_ilstSize;
499 if (m_metaSize >= numeric_limits<std::uint32_t>::max()) {
500 diag.emplace_back(
DiagLevel::Critical,
"Making such big tags is not implemented.",
"making MP4 tag");
515 BinaryWriter writer(&stream);
516 writer.writeUInt32BE(
static_cast<std::uint32_t
>(m_metaSize));
519 static const std::uint8_t hdlrData[37] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x21, 0x68, 0x64, 0x6C, 0x72, 0x00, 0x00, 0x00, 0x00, 0x00,
520 0x00, 0x00, 0x00, 0x6D, 0x64, 0x69, 0x72, 0x61, 0x70, 0x70, 0x6C, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
521 stream.write(
reinterpret_cast<const char *
>(hdlrData),
sizeof(hdlrData));
522 if (m_ilstSize != 8) {
524 writer.writeUInt32BE(
static_cast<std::uint32_t
>(m_ilstSize));
527 for (
auto &maker : m_maker) {
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.
static const TagValue & empty()
Returns a default-constructed TagValue where TagValue::isNull() and TagValue::isEmpty() both return t...
Encapsulates IDs of MP4 atoms holding tag information.
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