tagparser/tagvalue.h

772 lines
27 KiB
C++

#ifndef TAG_PARSER_TAGVALUE_H
#define TAG_PARSER_TAGVALUE_H
#include "./localehelper.h"
#include "./positioninset.h"
#include "./tagtype.h"
#include <c++utilities/chrono/datetime.h>
#include <c++utilities/chrono/timespan.h>
#include <c++utilities/conversion/binaryconversion.h>
#include <c++utilities/misc/flagenumclass.h>
#include <c++utilities/misc/traits.h>
#include <cstdint>
#include <cstring>
#include <iosfwd>
#include <memory>
#include <string>
#include <unordered_map>
namespace TagParser {
class Tag;
class Id3v2Frame;
/*!
* \brief Specifies the text encoding.
*/
enum class TagTextEncoding : unsigned int {
Latin1, /**< ISO/IEC 8859-1 aka "Latin 1" */
Utf8, /**< UTF-8 */
Utf16LittleEndian, /**< UTF-16 (little endian) */
Utf16BigEndian, /**< UTF-16 (big endian) */
Unspecified /**< unspecified encoding */
};
/*!
* \brief Specifies additional flags about the tag value.
* \remarks It depends on the tag format whether these flags can be present. When setting a flag
* which is not supported by the tag format, the tag implementation for that tag format
* is supposed to ignore these flags.
*/
enum class TagValueFlags : std::uint64_t {
None, /**< no flags present */
ReadOnly, /**< the tag value is labeled as read-only */
};
} // namespace TagParser
CPP_UTILITIES_MARK_FLAG_ENUM_CLASS(TagParser, TagParser::TagValueFlags)
namespace TagParser {
/*!
* \brief Returns the size of one character for the specified \a encoding in bytes.
* \remarks For variable-width encoding the minimum size is returned.
*/
constexpr int characterSize(TagTextEncoding encoding)
{
switch (encoding) {
case TagTextEncoding::Latin1:
case TagTextEncoding::Utf8:
return 1;
case TagTextEncoding::Utf16LittleEndian:
case TagTextEncoding::Utf16BigEndian:
return 2;
default:
return 0;
}
}
struct TAG_PARSER_EXPORT Popularity {
/// \brief The user who gave the rating / played the file, e.g. identified by e-mail address.
std::string user;
/// \brief The rating on a tag type specific scale.
double rating = 0.0;
/// \brief Play counter specific to the user.
std::uint64_t playCounter = 0;
/// \brief Specifies the scale used for \a rating by the tag defining that scale.
/// \remarks The value TagType::Unspecified is used to denote a *generic* scale from 1.0 to
/// 5.0 where 5.0 is the best and the special value 0.0 stands for "not rated".
TagType scale = TagType::Unspecified;
bool scaleTo(TagType targetScale);
Popularity scaled(TagType targetScale) const;
std::string toString() const;
static Popularity fromString(std::string_view str);
static Popularity fromString(std::string_view str, TagType scale);
/// \brief Returns whether the Popularity is empty. The \a scale and zero-values don't count.
bool isEmpty() const
{
return user.empty() && rating == 0.0 && !playCounter;
}
/// \brief Returns whether two instances are equal.
/// \remarks Currently they must match exactly but in the future conversions between different
/// scales might be implemented and two instances would be considered equal if the ratings are
/// considered equal (even specified using different scales).
bool operator==(const Popularity &other) const
{
return playCounter == other.playCounter && rating == other.rating && user == other.user && scale == other.scale;
}
};
/*!
* \brief Same as Popularity::scaleTo() but returns a new object.
*/
inline Popularity Popularity::scaled(TagType targetScale) const
{
auto copy = *this;
copy.scaleTo(targetScale);
return copy;
}
/*!
* \brief Specifies the data type.
*/
enum class TagDataType : unsigned int {
Text, /**< text/string */
Integer, /**< integer */
PositionInSet, /**< position in set, see TagParser::PositionInSet */
StandardGenreIndex, /**< pre-defined genre name denoted by numerical code */
TimeSpan, /**< time span, see CppUtilities::TimeSpan */
DateTime, /**< date time, see CppUtilities::DateTime */
Picture, /**< picture file */
Binary, /**< unspecified binary data */
Undefined, /**< undefined/invalid data type */
Popularity, /**< rating with user info and play counter (as in ID3v2's "Popularimeter") */
UnsignedInteger, /**< unsigned integer */
DateTimeExpression, /**< date time expression, see CppUtilities::DateTimeExpression */
};
TAG_PARSER_EXPORT std::string_view tagDataTypeString(TagDataType dataType);
/*!
* \brief The TagValueComparisionOption enum specifies options for TagValue::compareTo().
*/
enum class TagValueComparisionFlags : unsigned int {
None, /**< no special behavior */
CaseInsensitive = 0x1, /**< string-comparisons are case-insensitive (does *not* affect non-string comparisons) */
IgnoreMetaData = 0x2, /**< do *not* take meta-data like description and MIME-types into account */
};
struct TagValuePrivate;
class TAG_PARSER_EXPORT TagValue {
public:
// constructor, destructor
explicit TagValue();
explicit TagValue(const char *text, std::size_t textSize, TagTextEncoding textEncoding = TagTextEncoding::Latin1,
TagTextEncoding convertTo = TagTextEncoding::Unspecified);
explicit TagValue(
const char *text, TagTextEncoding textEncoding = TagTextEncoding::Latin1, TagTextEncoding convertTo = TagTextEncoding::Unspecified);
explicit TagValue(
const std::string &text, TagTextEncoding textEncoding = TagTextEncoding::Latin1, TagTextEncoding convertTo = TagTextEncoding::Unspecified);
explicit TagValue(
std::string_view text, TagTextEncoding textEncoding = TagTextEncoding::Latin1, TagTextEncoding convertTo = TagTextEncoding::Unspecified);
explicit TagValue(int value);
explicit TagValue(std::uint64_t value);
explicit TagValue(
const char *data, std::size_t length, TagDataType type = TagDataType::Undefined, TagTextEncoding encoding = TagTextEncoding::Latin1);
explicit TagValue(std::unique_ptr<char[]> &&data, std::size_t length, TagDataType type = TagDataType::Binary,
TagTextEncoding encoding = TagTextEncoding::Latin1);
explicit TagValue(PositionInSet value);
explicit TagValue(CppUtilities::DateTime value);
explicit TagValue(const CppUtilities::DateTimeExpression &value);
explicit TagValue(CppUtilities::TimeSpan value);
explicit TagValue(const Popularity &value);
TagValue(const TagValue &other);
TagValue(TagValue &&other);
~TagValue();
// operators
TagValue &operator=(const TagValue &other);
TagValue &operator=(TagValue &&other);
bool operator==(const TagValue &other) const;
bool operator!=(const TagValue &other) const;
operator bool() const;
// methods
bool isNull() const;
bool isEmpty() const;
void clearData();
void clearMetadata();
void clearDataAndMetadata();
TagDataType type() const;
std::string toString(TagTextEncoding encoding = TagTextEncoding::Unspecified) const;
std::string toDisplayString() const;
void toString(std::string &result, TagTextEncoding encoding = TagTextEncoding::Unspecified) const;
std::u16string toWString(TagTextEncoding encoding = TagTextEncoding::Unspecified) const;
void toWString(std::u16string &result, TagTextEncoding encoding = TagTextEncoding::Unspecified) const;
std::int32_t toInteger() const;
std::uint64_t toUnsignedInteger() const;
int toStandardGenreIndex() const;
PositionInSet toPositionInSet() const;
CppUtilities::TimeSpan toTimeSpan() const;
CppUtilities::DateTime toDateTime() const;
CppUtilities::DateTimeExpression toDateTimeExpression() const;
Popularity toPopularity() const;
Popularity toScaledPopularity(TagType scale = TagType::Unspecified) const;
std::size_t dataSize() const;
char *dataPointer();
const char *dataPointer() const;
std::string_view data() const;
const std::string &description() const;
void setDescription(std::string_view value, TagTextEncoding encoding = TagTextEncoding::Latin1);
const std::string &mimeType() const;
void setMimeType(std::string_view mimeType);
const Locale &locale() const;
Locale &locale();
void setLocale(const Locale &locale);
TagValueFlags flags() const;
void setFlags(TagValueFlags flags);
bool isLabeledAsReadonly() const;
void setReadonly(bool readOnly);
const std::unordered_map<std::string, std::string> &nativeData() const;
std::unordered_map<std::string, std::string> &nativeData();
TagTextEncoding dataEncoding() const;
void convertDataEncoding(TagTextEncoding encoding);
void convertDataEncodingForTag(const Tag *tag);
TagTextEncoding descriptionEncoding() const;
void convertDescriptionEncoding(TagTextEncoding encoding);
static const TagValue &empty();
void assignText(const char *text, std::size_t textSize, TagTextEncoding textEncoding = TagTextEncoding::Latin1,
TagTextEncoding convertTo = TagTextEncoding::Unspecified);
void assignText(
const std::string &text, TagTextEncoding textEncoding = TagTextEncoding::Latin1, TagTextEncoding convertTo = TagTextEncoding::Unspecified);
void assignText(
std::string_view text, TagTextEncoding textEncoding = TagTextEncoding::Latin1, TagTextEncoding convertTo = TagTextEncoding::Unspecified);
void assignInteger(int value);
void assignUnsignedInteger(std::uint64_t value);
void assignStandardGenreIndex(int index);
void assignData(const char *data, std::size_t length, TagDataType type = TagDataType::Binary, TagTextEncoding encoding = TagTextEncoding::Latin1);
void assignData(std::unique_ptr<char[]> &&data, std::size_t length, TagDataType type = TagDataType::Binary,
TagTextEncoding encoding = TagTextEncoding::Latin1);
void assignPosition(PositionInSet value);
void assignTimeSpan(CppUtilities::TimeSpan value);
void assignDateTime(CppUtilities::DateTime value);
void assignDateTimeExpression(const CppUtilities::DateTimeExpression &value);
void assignPopularity(const Popularity &value);
static void stripBom(const char *&text, std::size_t &length, TagTextEncoding encoding);
static void ensureHostByteOrder(std::u16string &u16str, TagTextEncoding currentEncoding);
template <typename ContainerType,
CppUtilities::Traits::EnableIf<CppUtilities::Traits::IsIteratable<ContainerType>,
std::is_same<typename std::add_const<typename std::remove_pointer<typename ContainerType::value_type>::type>::type, const TagValue>>
* = nullptr>
static std::vector<std::string> toStrings(const ContainerType &values, TagTextEncoding encoding = TagTextEncoding::Utf8);
bool compareTo(const TagValue &other, TagValueComparisionFlags options = TagValueComparisionFlags::None) const;
bool compareData(const TagValue &other, bool ignoreCase = false) const;
static bool compareData(const std::string &data1, const std::string &data2, bool ignoreCase = false);
static bool compareData(const char *data1, std::size_t size1, const char *data2, std::size_t size2, bool ignoreCase = false);
private:
std::unique_ptr<char[]> m_ptr;
std::size_t m_size;
std::string m_desc;
std::string m_mimeType;
Locale m_locale;
std::unordered_map<std::string, std::string> m_nativeData;
TagDataType m_type;
TagTextEncoding m_encoding;
TagTextEncoding m_descEncoding;
TagValueFlags m_flags;
std::unique_ptr<TagValuePrivate> m_p;
};
/*!
* \brief Constructs a new TagValue holding the given integer \a value.
*/
inline TagValue::TagValue(int value)
: TagValue(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::Integer)
{
}
/*!
* \brief Constructs a new TagValue holding the given unsigned integer \a value.
*/
inline TagParser::TagValue::TagValue(std::uint64_t value)
: TagValue(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::UnsignedInteger)
{
}
/*!
* \brief Constructs a new TagValue holding a copy of the given PositionInSet \a value.
*/
inline TagValue::TagValue(PositionInSet value)
: TagValue(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::PositionInSet)
{
}
/*!
* \brief Constructs a new TagValue holding a copy of the given DateTime \a value.
*/
inline TagValue::TagValue(CppUtilities::DateTime value)
: TagValue(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::DateTime)
{
}
/*!
* \brief Constructs a new TagValue holding a copy of the given DateTimeExpression \a value.
*/
inline TagValue::TagValue(const CppUtilities::DateTimeExpression &value)
: TagValue(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::DateTimeExpression)
{
}
/*!
* \brief Constructs a new TagValue holding a copy of the given TimeSpan \a value.
*/
inline TagValue::TagValue(CppUtilities::TimeSpan value)
: TagValue(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::TimeSpan)
{
}
/*!
* \brief Constructs a new TagValue holding a copy of the given Popularity \a value.
*/
inline TagValue::TagValue(const Popularity &value)
: TagValue()
{
assignPopularity(value);
}
/*!
* \brief Returns whether both instances are equal.
* \sa The same as TagValue::compareTo() with TagValueComparisionOption::None so see TagValue::compareTo() for details.
*/
inline bool TagValue::operator==(const TagValue &other) const
{
return compareTo(other, TagValueComparisionFlags::None);
}
/*!
* \brief Returns whether both instances are not equal.
* \sa The negation of TagValue::compareTo() with TagValueComparisionOption::None so see TagValue::compareTo() for details.
*/
inline bool TagValue::operator!=(const TagValue &other) const
{
return !compareTo(other, TagValueComparisionFlags::None);
}
/*!
* \brief Returns whether the value is not empty.
* \sa See TagValue::isEmpty() for a definition on what is considered empty.
*/
inline TagParser::TagValue::operator bool() const
{
return !isEmpty();
}
/*!
* \brief Assigns a copy of the given \a text.
* \param text Specifies the text to be assigned.
* \param textEncoding Specifies the encoding of the given \a text.
* \param convertTo Specifies the encoding to convert \a text to; set to TagTextEncoding::Unspecified to
* use \a textEncoding without any character set conversions.
* \throws Throws a ConversionException if the conversion the specified character set fails.
* \remarks Strips the BOM of the specified \a text.
*/
inline void TagValue::assignText(const std::string &text, TagTextEncoding textEncoding, TagTextEncoding convertTo)
{
assignText(text.data(), text.size(), textEncoding, convertTo);
}
/*!
* \brief Assigns a copy of the given \a text.
* \param text Specifies the text to be assigned.
* \param textEncoding Specifies the encoding of the given \a text.
* \param convertTo Specifies the encoding to convert \a text to; set to TagTextEncoding::Unspecified to
* use \a textEncoding without any character set conversions.
* \throws Throws a ConversionException if the conversion the specified character set fails.
* \remarks Strips the BOM of the specified \a text.
*/
inline void TagValue::assignText(std::string_view text, TagTextEncoding textEncoding, TagTextEncoding convertTo)
{
assignText(text.data(), text.size(), textEncoding, convertTo);
}
/*!
* \brief Assigns the given PositionInSet \a value.
*/
inline void TagValue::assignPosition(PositionInSet value)
{
if (value.isNull()) {
m_type = TagDataType::PositionInSet;
clearData();
} else {
assignData(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::PositionInSet);
}
}
/*!
* \brief Assigns the given TimeSpan \a value.
*/
inline void TagValue::assignTimeSpan(CppUtilities::TimeSpan value)
{
assignData(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::TimeSpan);
}
/*!
* \brief Assigns the given DateTime \a value.
*/
inline void TagValue::assignDateTime(CppUtilities::DateTime value)
{
assignData(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::DateTime);
}
/*!
* \brief Assigns the given DateTimeExpression \a value.
*/
inline void TagParser::TagValue::assignDateTimeExpression(const CppUtilities::DateTimeExpression &value)
{
assignData(reinterpret_cast<const char *>(&value), sizeof(value), TagDataType::DateTimeExpression);
}
/*!
* \brief Assigns the given standard genre \a index to be assigned.
* \param index Specifies the index to be assigned.
* \sa <a href="http://en.wikipedia.org/wiki/ID3#List_of_genres">List of genres - Wikipedia</a>
*/
inline void TagValue::assignStandardGenreIndex(int index)
{
assignInteger(index);
m_type = TagDataType::StandardGenreIndex;
}
/*!
* \brief Returns the type of the assigned value.
*/
inline TagDataType TagValue::type() const
{
return m_type;
}
/*!
* \brief Converts the value of the current TagValue object to its equivalent
* std::string representation.
* \param result Specifies the string to store the result.
* \param encoding Specifies the encoding to to be used; set to TagTextEncoding::Unspecified to use the
* present encoding without any character set conversion.
* \remarks
* - Not all types can be converted to a string, eg. TagDataType::Picture, TagDataType::Binary and
* TagDataType::Unspecified will always fail to convert.
* - If UTF-16 is the desired output \a encoding, it makes sense to use the toWString() method instead.
* \throws Throws ConversionException on failure.
*/
inline std::string TagValue::toString(TagTextEncoding encoding) const
{
std::string res;
toString(res, encoding);
return res;
}
/*!
* \brief Converts the value of the current TagValue object to its equivalent
* std::wstring representation.
* \throws Throws ConversionException on failure.
* \remarks Use this only, if \a encoding is an UTF-16 encoding.
*/
inline std::u16string TagValue::toWString(TagTextEncoding encoding) const
{
std::u16string res;
toWString(res, encoding);
return res;
}
/*!
* \brief Returns whether no value is assigned at all.
* \remarks
* - Returns only true for default constructed instances or cleared instances (using TagValue::clearData()).
* So for empty strings, the integer 0, a TimeSpan of zero length, ... this function returns false.
* - Meta-data such as description and MIME-type is not considered as an assigned value.
*/
inline bool TagValue::isNull() const
{
return m_ptr == nullptr;
}
/*!
* \brief Returns whether no or an empty value is assigned.
* \remarks
* - An empty string and empty binary or picture data counts as empty so this function will return
* true for those. However, the integer 0, a TimeSpan of zero length, ... are not considered empty
* and this function will return false.
* - Meta-data such as description and MIME-type is not considered as an assigned value.
*/
inline bool TagValue::isEmpty() const
{
return m_ptr == nullptr || m_size == 0;
}
/*!
* \brief Clears the assigned data.
* \remarks Meta data such as description and MIME type remains unaffected.
* \sa clearMetadata()
* \sa clearDataAndMetadata()
*/
inline void TagValue::clearData()
{
m_size = 0;
m_ptr.reset();
}
/*!
* \brief Wipes assigned data including meta data.
* \sa clearData()
* \sa clearMetadata()
*/
inline void TagValue::clearDataAndMetadata()
{
clearData();
clearMetadata();
}
/*!
* \brief Returns the size of the assigned value in bytes.
* \remarks Meta data such as description and MIME type is not considered as part of the assigned value.
*/
inline std::size_t TagValue::dataSize() const
{
return m_size;
}
/*!
* \brief Returns a pointer to the raw data assigned to the current instance.
* \remarks The instance keeps ownership over the data which will be invalidated when the
* TagValue gets destroyed or another value is assigned.
* \remarks The raw data is not null terminated. See dataSize().
*/
inline char *TagValue::dataPointer()
{
return m_ptr.get();
}
inline const char *TagValue::dataPointer() const
{
return m_ptr.get();
}
/*!
* \brief Returns the currently assigned raw data.
*/
inline std::string_view TagValue::data() const
{
return std::string_view(m_ptr.get(), m_size);
}
/*!
* \brief Returns the description.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* \sa
* - descriptionEncoding() for the encoding of the returned string
* - convertDescriptionEncoding() to change the encoding of the description
* - setDescription() for setting the description
*/
inline const std::string &TagValue::description() const
{
return m_desc;
}
/*!
* \brief Sets the description.
* \param value Specifies the description.
* \param encoding Specifies the encoding used to provide the description.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* \sa
* - description() and descriptionEncoding()
* - convertDescriptionEncoding() to change the description encoding after assignment
*/
inline void TagValue::setDescription(std::string_view value, TagTextEncoding encoding)
{
m_desc = value;
m_descEncoding = encoding;
}
/*!
* \brief Returns the MIME type.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* \sa setMimeType()
*/
inline const std::string &TagValue::mimeType() const
{
return m_mimeType;
}
/*!
* \brief Sets the MIME type.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* \sa mimeType()
*/
inline void TagValue::setMimeType(std::string_view mimeType)
{
m_mimeType = mimeType;
}
/*!
* \brief Returns the locale.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* - The format of the locale is specific to the tag format. The implementation of the tag format might
* store the value without further validation.
* \sa setLocale()
*/
inline const Locale &TagValue::locale() const
{
return m_locale;
}
/*!
* \brief Returns the locale.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* - The format of the locale is specific to the tag format. The implementation of the tag format might
* store the value without further validation.
* \sa setLocale()
*/
inline Locale &TagValue::locale()
{
return m_locale;
}
/*!
* \brief Sets the setLocale.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* - The format of the locale is specific to the tag format. The implementation of the tag format might
* store the value without further validation.
* \sa locale()
*/
inline void TagValue::setLocale(const Locale &locale)
{
m_locale = locale;
}
/*!
* \brief Returns the flags.
* \sa TagValueFlags
*/
inline TagValueFlags TagValue::flags() const
{
return m_flags;
}
/*!
* \brief Sets the flags.
* \sa TagValueFlags
*/
inline void TagValue::setFlags(TagValueFlags flags)
{
m_flags = flags;
}
/*!
* \brief Returns an indication whether the value is labeled as read-only.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* - This is just an additional information. It has no effect on the behavior of the TagValue thus
* assignments can still be performed (to prohibit assignments simply use the "const" keyword).
* \sa setReadonly()
*/
inline bool TagValue::isLabeledAsReadonly() const
{
return m_flags & TagValueFlags::ReadOnly;
}
/*!
* \brief Sets whether the TagValue is labeled as read-only.
* \remarks
* - Whether this additional meta-data is available and can be used depends on the tag format. It will
* be ignored by the implementation of the tag format if not supported.
* - This is just an additional information. It has no effect on the behavior of the TagValue thus
* assignments can still be performed (to prohibit assignments simply use the "const" keyword).
* \sa isLabeledAsReadonly()
*/
inline void TagValue::setReadonly(bool readOnly)
{
CppUtilities::modFlagEnum(m_flags, TagValueFlags::ReadOnly, readOnly);
}
/*!
* \brief Holds tag format specific meta-data for that field which does not fit into any of the other
* meta-data properties.
*/
inline const std::unordered_map<std::string, std::string> &TagValue::nativeData() const
{
return m_nativeData;
}
/*!
* \brief Holds tag format specific meta-data for that field which does not fit into any of the other
* meta-data properties.
*/
inline std::unordered_map<std::string, std::string> &TagValue::nativeData()
{
return m_nativeData;
}
/*!
* \brief Returns the data encoding.
* \remarks This value is only relevant if type() equals TagDataType::Text.
* \sa assignText()
*/
inline TagTextEncoding TagValue::dataEncoding() const
{
return m_encoding;
}
/*!
* \brief Returns the description encoding.
* \remarks This value is only relevant if a description is assigned.
* \sa description(), setDescription()
*/
inline TagTextEncoding TagValue::descriptionEncoding() const
{
return m_descEncoding;
}
/*!
* \brief Converts the specified \a values to string using the specified \a encoding.
* \throws Throws ConversionException on failure.
* \sa toString()
*/
template <typename ContainerType,
CppUtilities::Traits::EnableIf<CppUtilities::Traits::IsIteratable<ContainerType>,
std::is_same<typename std::add_const<typename std::remove_pointer<typename ContainerType::value_type>::type>::type, const TagValue>> *>
std::vector<std::string> TagValue::toStrings(const ContainerType &values, TagTextEncoding encoding)
{
std::vector<std::string> res;
res.reserve(values.size());
for (const auto &value : values) {
res.emplace_back(CppUtilities::Traits::dereferenceMaybe(value).toString(encoding));
}
return res;
}
/*!
* \brief Returns whether the raw data of the current instance equals the raw data of \a other.
*/
inline bool TagValue::compareData(const TagValue &other, bool ignoreCase) const
{
return compareData(m_ptr.get(), m_size, other.m_ptr.get(), other.m_size, ignoreCase);
}
/*!
* \brief Returns whether 2 data buffers are equal.
*/
inline bool TagValue::compareData(const std::string &data1, const std::string &data2, bool ignoreCase)
{
return compareData(data1.data(), data1.size(), data2.data(), data2.size(), ignoreCase);
}
} // namespace TagParser
CPP_UTILITIES_MARK_FLAG_ENUM_CLASS(TagParser, TagParser::TagValueComparisionFlags)
#endif // TAG_PARSER_TAGVALUE_H