347 lines
20 KiB
C++
347 lines
20 KiB
C++
#include "./helper.h"
|
|
|
|
#include "../id3/id3genres.h"
|
|
#include "../tagvalue.h"
|
|
|
|
#include <c++utilities/chrono/format.h>
|
|
#include <c++utilities/conversion/conversionexception.h>
|
|
|
|
using namespace CppUtilities;
|
|
|
|
#include <cppunit/TestFixture.h>
|
|
#include <cppunit/extensions/HelperMacros.h>
|
|
|
|
using namespace std;
|
|
using namespace CppUtilities;
|
|
using namespace TagParser;
|
|
using namespace CPPUNIT_NS;
|
|
|
|
/*!
|
|
* \brief The TagValueTests class tests the TagParser::TagValue class.
|
|
*/
|
|
class TagValueTests : public TestFixture {
|
|
CPPUNIT_TEST_SUITE(TagValueTests);
|
|
CPPUNIT_TEST(testBasics);
|
|
CPPUNIT_TEST(testBinary);
|
|
CPPUNIT_TEST(testInteger);
|
|
CPPUNIT_TEST(testUnsignedInteger);
|
|
CPPUNIT_TEST(testPositionInSet);
|
|
CPPUNIT_TEST(testTimeSpan);
|
|
CPPUNIT_TEST(testDateTime);
|
|
CPPUNIT_TEST(testDateTimeExpression);
|
|
CPPUNIT_TEST(testPopularity);
|
|
CPPUNIT_TEST(testString);
|
|
CPPUNIT_TEST(testEqualityOperator);
|
|
CPPUNIT_TEST(testPopularityScaling);
|
|
CPPUNIT_TEST_SUITE_END();
|
|
|
|
public:
|
|
void setUp() override;
|
|
void tearDown() override;
|
|
|
|
void testBasics();
|
|
void testBinary();
|
|
void testInteger();
|
|
void testUnsignedInteger();
|
|
void testPositionInSet();
|
|
void testTimeSpan();
|
|
void testDateTime();
|
|
void testDateTimeExpression();
|
|
void testPopularity();
|
|
void testString();
|
|
void testEqualityOperator();
|
|
void testPopularityScaling();
|
|
};
|
|
|
|
CPPUNIT_TEST_SUITE_REGISTRATION(TagValueTests);
|
|
|
|
void TagValueTests::setUp()
|
|
{
|
|
}
|
|
|
|
void TagValueTests::tearDown()
|
|
{
|
|
}
|
|
|
|
void TagValueTests::testBasics()
|
|
{
|
|
CPPUNIT_ASSERT(TagValue::empty().isEmpty());
|
|
CPPUNIT_ASSERT_EQUAL(TagDataType::Undefined, TagValue().type());
|
|
}
|
|
|
|
void TagValueTests::testBinary()
|
|
{
|
|
const TagValue binary("123", 3, TagDataType::Binary);
|
|
CPPUNIT_ASSERT_EQUAL(TagDataType::Binary, binary.type());
|
|
CPPUNIT_ASSERT_EQUAL("123"s, string(binary.dataPointer(), binary.dataSize()));
|
|
CPPUNIT_ASSERT_THROW(binary.toString(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(binary.toInteger(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(binary.toPositionInSet(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(binary.toStandardGenreIndex(), ConversionException);
|
|
}
|
|
|
|
void TagValueTests::testInteger()
|
|
{
|
|
// positive number
|
|
auto integer = TagValue(42);
|
|
CPPUNIT_ASSERT(!integer.isEmpty());
|
|
CPPUNIT_ASSERT_EQUAL(TagDataType::Integer, integer.type());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::int32_t>(42), integer.toInteger());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::uint64_t>(42), integer.toUnsignedInteger());
|
|
CPPUNIT_ASSERT_EQUAL("42"s, integer.toString());
|
|
integer.assignInteger(2);
|
|
CPPUNIT_ASSERT_EQUAL("Country"s, string(Id3Genres::stringFromIndex(integer.toStandardGenreIndex())));
|
|
integer.assignInteger(Id3Genres::emptyGenreIndex());
|
|
CPPUNIT_ASSERT_EQUAL(Id3Genres::emptyGenreIndex(), integer.toStandardGenreIndex());
|
|
integer.clearData();
|
|
CPPUNIT_ASSERT_EQUAL(Id3Genres::emptyGenreIndex(), integer.toStandardGenreIndex());
|
|
|
|
// negative number
|
|
integer.assignInteger(-25);
|
|
CPPUNIT_ASSERT_EQUAL("-25"s, integer.toString());
|
|
CPPUNIT_ASSERT_EQUAL(PositionInSet(-25), integer.toPositionInSet());
|
|
CPPUNIT_ASSERT_THROW(integer.toStandardGenreIndex(), ConversionException);
|
|
|
|
// zero
|
|
integer.assignInteger(0);
|
|
CPPUNIT_ASSERT_MESSAGE("explicitly assigned zero not considered empty", !integer.isEmpty());
|
|
CPPUNIT_ASSERT_EQUAL("0"s, integer.toString());
|
|
CPPUNIT_ASSERT_EQUAL(DateTime(), integer.toDateTime());
|
|
CPPUNIT_ASSERT_EQUAL(TimeSpan(), integer.toTimeSpan());
|
|
|
|
// empty value treatet as zero when using to...() methods
|
|
integer.clearData();
|
|
CPPUNIT_ASSERT_MESSAGE("cleared vale considered empty", integer.isEmpty());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("only date (but not type) cleared"s, TagDataType::Integer, integer.type());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::int32_t>(0), integer.toInteger());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::uint64_t>(0), integer.toUnsignedInteger());
|
|
CPPUNIT_ASSERT_EQUAL(string(), integer.toString());
|
|
CPPUNIT_ASSERT_EQUAL(DateTime(), integer.toDateTime());
|
|
CPPUNIT_ASSERT_EQUAL(TimeSpan(), integer.toTimeSpan());
|
|
}
|
|
|
|
void TagValueTests::testUnsignedInteger()
|
|
{
|
|
auto unsignedInteger = TagValue(static_cast<std::uint64_t>(42ul));
|
|
CPPUNIT_ASSERT(!unsignedInteger.isEmpty());
|
|
CPPUNIT_ASSERT_EQUAL(TagDataType::UnsignedInteger, unsignedInteger.type());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::int32_t>(42), unsignedInteger.toInteger());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::uint64_t>(42), unsignedInteger.toUnsignedInteger());
|
|
CPPUNIT_ASSERT_EQUAL("42"s, unsignedInteger.toString());
|
|
unsignedInteger.assignUnsignedInteger(2);
|
|
CPPUNIT_ASSERT_EQUAL("Country"s, string(Id3Genres::stringFromIndex(unsignedInteger.toStandardGenreIndex())));
|
|
unsignedInteger.assignInteger(Id3Genres::emptyGenreIndex());
|
|
CPPUNIT_ASSERT_EQUAL(Id3Genres::emptyGenreIndex(), unsignedInteger.toStandardGenreIndex());
|
|
unsignedInteger.clearData();
|
|
CPPUNIT_ASSERT_EQUAL(Id3Genres::emptyGenreIndex(), unsignedInteger.toStandardGenreIndex());
|
|
|
|
// zero
|
|
unsignedInteger.assignInteger(0);
|
|
CPPUNIT_ASSERT_MESSAGE("explicitly assigned zero not considered empty", !unsignedInteger.isEmpty());
|
|
CPPUNIT_ASSERT_EQUAL("0"s, unsignedInteger.toString());
|
|
CPPUNIT_ASSERT_EQUAL(DateTime(), unsignedInteger.toDateTime());
|
|
CPPUNIT_ASSERT_EQUAL(TimeSpan(), unsignedInteger.toTimeSpan());
|
|
}
|
|
|
|
void TagValueTests::testPositionInSet()
|
|
{
|
|
const TagValue test(PositionInSet(4, 23));
|
|
CPPUNIT_ASSERT_EQUAL(PositionInSet(4, 23), test.toPositionInSet());
|
|
CPPUNIT_ASSERT_EQUAL(4, test.toInteger());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::uint64_t>(4), test.toUnsignedInteger());
|
|
CPPUNIT_ASSERT_EQUAL("4/23"s, test.toString());
|
|
CPPUNIT_ASSERT_THROW(test.toStandardGenreIndex(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(test.toDateTime(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(test.toTimeSpan(), ConversionException);
|
|
}
|
|
|
|
void TagValueTests::testTimeSpan()
|
|
{
|
|
const TimeSpan fiveMinutes(TimeSpan::fromMinutes(5));
|
|
TagValue timeSpan;
|
|
timeSpan.assignTimeSpan(fiveMinutes);
|
|
CPPUNIT_ASSERT_EQUAL(timeSpan, TagValue(timeSpan));
|
|
CPPUNIT_ASSERT_EQUAL(fiveMinutes, timeSpan.toTimeSpan());
|
|
CPPUNIT_ASSERT_EQUAL(fiveMinutes.toString(), timeSpan.toString());
|
|
CPPUNIT_ASSERT_THROW(timeSpan.toInteger(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(timeSpan.toDateTime(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(timeSpan.toPositionInSet(), ConversionException);
|
|
}
|
|
|
|
void TagValueTests::testDateTime()
|
|
{
|
|
const auto now = DateTime::now();
|
|
auto value = TagValue();
|
|
value.assignDateTime(now);
|
|
CPPUNIT_ASSERT_EQUAL(value, TagValue(value));
|
|
CPPUNIT_ASSERT_EQUAL(now, value.toDateTime());
|
|
CPPUNIT_ASSERT_EQUAL(now.toIsoString(), value.toString());
|
|
CPPUNIT_ASSERT_THROW(value.toInteger(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(value.toTimeSpan(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(value.toPositionInSet(), ConversionException);
|
|
}
|
|
|
|
void TagValueTests::testDateTimeExpression()
|
|
{
|
|
auto expr = DateTimeExpression::fromIsoString("2007");
|
|
auto value = TagValue();
|
|
value.assignDateTimeExpression(expr);
|
|
CPPUNIT_ASSERT_EQUAL(value, TagValue(expr));
|
|
CPPUNIT_ASSERT_EQUAL(expr.value, value.toDateTime());
|
|
CPPUNIT_ASSERT_EQUAL(expr, value.toDateTimeExpression());
|
|
CPPUNIT_ASSERT_EQUAL(expr.toIsoString(), value.toString());
|
|
CPPUNIT_ASSERT_THROW(value.toInteger(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(value.toTimeSpan(), ConversionException);
|
|
CPPUNIT_ASSERT_THROW(value.toPositionInSet(), ConversionException);
|
|
}
|
|
|
|
void TagValueTests::testPopularity()
|
|
{
|
|
const auto tagValue = TagValue(Popularity{ .user = "foo", .rating = 40.0, .playCounter = 123, .scale = TagType::VorbisComment });
|
|
const auto popularity = tagValue.toPopularity();
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to popularity (user)", "foo"s, popularity.user);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to popularity (rating)", 40.0, popularity.rating);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to popularity (play counter)", std::uint64_t(123), popularity.playCounter);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to popularity (scale)", TagType::VorbisComment, popularity.scale);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to string", "foo|40|123"s, tagValue.toString());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to string (only rating)", "43"s, TagValue(Popularity{ .rating = 43 }).toString());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to integer", 40, tagValue.toInteger());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to unsigned integer", static_cast<std::uint64_t>(40), tagValue.toUnsignedInteger());
|
|
CPPUNIT_ASSERT_THROW_MESSAGE(
|
|
"failing conversion to other type", TagValue("foo|bar"sv, TagTextEncoding::Latin1).toPopularity(), ConversionException);
|
|
const auto scaledPopularity = tagValue.toScaledPopularity();
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("rating scaled to generic scale", 2.0, scaledPopularity.rating);
|
|
CPPUNIT_ASSERT_THROW_MESSAGE(
|
|
"failed to scale if no scaling for specified format defined", tagValue.toScaledPopularity(TagType::Mp4Tag), ConversionException);
|
|
}
|
|
|
|
void TagValueTests::testString()
|
|
{
|
|
CPPUNIT_ASSERT_EQUAL("15\xe4"s, TagValue("15ä", 4, TagTextEncoding::Utf8).toString(TagTextEncoding::Latin1));
|
|
CPPUNIT_ASSERT_EQUAL("15\xe4"s, TagValue("15ä", TagTextEncoding::Utf8, TagTextEncoding::Latin1).toString());
|
|
CPPUNIT_ASSERT_EQUAL("15ä"s, TagValue("15ä", 4, TagTextEncoding::Utf8).toString(TagTextEncoding::Utf8));
|
|
CPPUNIT_ASSERT_EQUAL("\x31\0\x35\0"s, TagValue(15).toString(TagTextEncoding::Utf16LittleEndian));
|
|
CPPUNIT_ASSERT_EQUAL("\0\x31\0\x35"s, TagValue(15).toString(TagTextEncoding::Utf16BigEndian));
|
|
CPPUNIT_ASSERT_EQUAL(15, TagValue("\0\x31\0\x35"s, TagTextEncoding::Utf16BigEndian).toInteger());
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<std::uint64_t>(15), TagValue("\0\x31\0\x35"s, TagTextEncoding::Utf16BigEndian).toUnsignedInteger());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE(
|
|
"original encoding preserved", "15ä"s, TagValue("15ä", 4, TagTextEncoding::Utf8).toString(TagTextEncoding::Unspecified));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("original encoding preserved", "\0\x31\0\x35"s,
|
|
TagValue("\0\x31\0\x35", 4, TagTextEncoding::Utf16BigEndian).toString(TagTextEncoding::Unspecified));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE(
|
|
"UTF-8 BOM truncated", "täst"s, TagValue("\xef\xbb\xbftäst", 8, TagTextEncoding::Utf8).toString(TagTextEncoding::Unspecified));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("UTF-16 LE BOM truncated", "\0t\0\xe4\0s\0t"s,
|
|
TagValue("\xff\xfe\0t\0\xe4\0s\0t", 10, TagTextEncoding::Utf16LittleEndian).toString(TagTextEncoding::Unspecified));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("UTF-16 BE BOM truncated", "t\0\xe4\0s\0t\0"s,
|
|
TagValue("\xfe\xfft\0\xe4\0s\0t\0", 10, TagTextEncoding::Utf16BigEndian).toString(TagTextEncoding::Unspecified));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion via c'tor", "15\xe4"s,
|
|
TagValue("\xef\xbb\xbf\x31\x35ä", 7, TagTextEncoding::Utf8, TagTextEncoding::Latin1).toString(TagTextEncoding::Unspecified));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to int", -15, TagValue(" - 156", 5, TagTextEncoding::Utf8).toInteger());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to int", 15, TagValue("\0\x31\0\x35", 4, TagTextEncoding::Utf16BigEndian).toInteger());
|
|
CPPUNIT_ASSERT_THROW_MESSAGE("failing conversion to int", TagValue("15ä", 4, TagTextEncoding::Utf8).toInteger(), ConversionException);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to pos", PositionInSet(4, 15), TagValue("4 / 15", 6, TagTextEncoding::Utf8).toPositionInSet());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE(
|
|
"conversion to pos", PositionInSet(15), TagValue("\0\x31\0\x35", 4, TagTextEncoding::Utf16BigEndian).toPositionInSet());
|
|
CPPUNIT_ASSERT_THROW_MESSAGE("failing conversion pos", TagValue("a4 / 15", 7, TagTextEncoding::Utf8).toPositionInSet(), ConversionException);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE(
|
|
"conversion to date time", DateTime::fromDate(2004, 4, 15), TagValue("2004-04-15", 10, TagTextEncoding::Utf8).toDateTime());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to date time expression", DateTimeExpression::fromIsoString("2004-04"),
|
|
TagValue("2004-04-15", 7, TagTextEncoding::Utf8).toDateTimeExpression());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to date from UTF-16", DateTime::fromDate(2015, 4, 15),
|
|
TagValue("\0\x32\0\x30\0\x31\0\x35\0\x2d\0\x30\0\x34\0\x2d\0\x31\0\x35", 20, TagTextEncoding::Utf16BigEndian).toDateTime());
|
|
CPPUNIT_ASSERT_THROW_MESSAGE("failing conversion to date", TagValue("_", 1, TagTextEncoding::Utf8).toDateTime(), ConversionException);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to time span", TimeSpan::fromHours(1.5), TagValue("01:30:00", 10, TagTextEncoding::Utf8).toTimeSpan());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to time span from UTF-16", TimeSpan::fromHours(1.5),
|
|
TagValue("\0\x31\0\x3a\0\x33\0\x30\0\x3a\0\x30\0\x30", 14, TagTextEncoding::Utf16BigEndian).toTimeSpan());
|
|
CPPUNIT_ASSERT_THROW_MESSAGE("failing conversion to time span", TagValue("_", 1, TagTextEncoding::Utf8).toTimeSpan(), ConversionException);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE(
|
|
"conversion to genre from index", 15, TagValue("\0\x31\0\x35", 4, TagTextEncoding::Utf16BigEndian).toStandardGenreIndex());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to genre from name", 2, TagValue("Country", 7, TagTextEncoding::Latin1).toStandardGenreIndex());
|
|
CPPUNIT_ASSERT_THROW_MESSAGE(
|
|
"failing conversion to genre", TagValue("Kountry", 7, TagTextEncoding::Latin1).toStandardGenreIndex(), ConversionException);
|
|
const auto popularity = TagValue("foo|42|123"sv).toPopularity();
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to popularity (user)", "foo"s, popularity.user);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to popularity (rating)", 42.0, popularity.rating);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("conversion to popularity (play counter)", std::uint64_t(123), popularity.playCounter);
|
|
CPPUNIT_ASSERT_THROW_MESSAGE("failing conversion to popularity", TagValue("foo|bar"sv).toPopularity(), ConversionException);
|
|
}
|
|
|
|
void TagValueTests::testEqualityOperator()
|
|
{
|
|
CPPUNIT_ASSERT_MESSAGE("equality requires identical types or identical string representation"s, TagValue(0) != TagValue::empty());
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("comparison of equal types"s, TagValue(15), TagValue(15));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("types might differ"s, TagValue("15", 2, TagTextEncoding::Latin1), TagValue(15));
|
|
CPPUNIT_ASSERT_MESSAGE("but some types shall never be considered equal"s, TagValue(DateTime(0)) != TagValue(TimeSpan(0)));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("comparison of equal UTF-16 strings"s, TagValue("\x31\0\x32\0", 4, TagTextEncoding::Utf16LittleEndian),
|
|
TagValue("\x31\0\x32\0", 4, TagTextEncoding::Utf16LittleEndian));
|
|
CPPUNIT_ASSERT_MESSAGE("comparison of different UTF-16 strings"s,
|
|
TagValue("\x31\0\x33\0", 4, TagTextEncoding::Utf16LittleEndian) != TagValue("\x31\0\x32\0", 4, TagTextEncoding::Utf16LittleEndian));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE(
|
|
"comparison of equal binary data"s, TagValue("\x31\0\x32\0", 4, TagDataType::Binary), TagValue("\x31\0\x32\0", 4, TagDataType::Binary));
|
|
CPPUNIT_ASSERT_MESSAGE(
|
|
"comparison of different binary data"s, TagValue("\x31\0\x33\0", 4, TagDataType::Binary) != TagValue("\x31\0\x32\0", 4, TagDataType::Binary));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("different encodings are converted if neccassary"s, TagValue("\0\x31\0\x35", 4, TagTextEncoding::Utf16BigEndian),
|
|
TagValue("15", 2, TagTextEncoding::Latin1));
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE(
|
|
"encoding is ignored when not relevant for types"s, TagValue("\0\x31\0\x35", 4, TagTextEncoding::Utf16BigEndian), TagValue(15));
|
|
const TagValue fooTagValue("foo", 3, TagDataType::Text), fOoTagValue("fOo", 3, TagDataType::Text);
|
|
CPPUNIT_ASSERT_MESSAGE("string comparison case-sensitive by default"s, fooTagValue != fOoTagValue);
|
|
CPPUNIT_ASSERT_MESSAGE("case-insensitive string comparison"s, fooTagValue.compareTo(fOoTagValue, TagValueComparisionFlags::CaseInsensitive));
|
|
const auto popularity = Popularity{ .user = "some user", .rating = 200, .playCounter = 0 };
|
|
const auto first = TagValue(popularity), second = TagValue(popularity);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("comparison of equal popularity (string and binary representation)"s, TagValue("some user|200.0"sv), first);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("comparison of equal popularity (only binary representation)"s, first, second);
|
|
CPPUNIT_ASSERT_MESSAGE("default-popularity not equal to empty tag value"s, TagValue(Popularity()) != TagValue());
|
|
CPPUNIT_ASSERT_MESSAGE("popularity not equal"s, first != TagValue(Popularity({ .rating = 200 })));
|
|
|
|
// meta-data
|
|
TagValue withDescription(15);
|
|
withDescription.setDescription("test");
|
|
CPPUNIT_ASSERT_MESSAGE("meta-data must be equal"s, withDescription != TagValue(15));
|
|
CPPUNIT_ASSERT_MESSAGE("different meta-data ignored"s, withDescription.compareTo(TagValue(15), TagValueComparisionFlags::IgnoreMetaData));
|
|
TagValue withDescription2(withDescription);
|
|
CPPUNIT_ASSERT_EQUAL(withDescription, withDescription2);
|
|
withDescription2.setMimeType("foo/bar");
|
|
CPPUNIT_ASSERT(withDescription != withDescription2);
|
|
withDescription.setMimeType(withDescription2.mimeType());
|
|
CPPUNIT_ASSERT_EQUAL(withDescription, withDescription2);
|
|
withDescription2.setDescription("Test");
|
|
CPPUNIT_ASSERT_MESSAGE("meta-data case must match by default"s, withDescription != withDescription2);
|
|
CPPUNIT_ASSERT_MESSAGE("meta-data case ignored"s, withDescription.compareTo(withDescription2, TagValueComparisionFlags::CaseInsensitive));
|
|
}
|
|
|
|
void TagValueTests::testPopularityScaling()
|
|
{
|
|
const auto genericZero = Popularity{ .rating = 0.0, .scale = TagType::Unspecified };
|
|
const auto genericMin = Popularity{ .rating = 1.0, .scale = TagType::Unspecified };
|
|
const auto genericMax = Popularity{ .rating = 5.0, .scale = TagType::Unspecified };
|
|
const auto genericMiddle = Popularity{ .rating = 3.0, .scale = TagType::Unspecified };
|
|
const auto id3zero = Popularity{ .rating = 0.0, .scale = TagType::Id3v2Tag };
|
|
const auto id3min = Popularity{ .rating = 1.0, .scale = TagType::Id3v2Tag };
|
|
const auto id3max = Popularity{ .rating = 255.0, .scale = TagType::Id3v2Tag };
|
|
const auto id3middle = Popularity{ .rating = 128.0, .scale = TagType::Id3v2Tag };
|
|
const auto vorbisZero = Popularity{ .rating = 0.0, .scale = TagType::VorbisComment };
|
|
const auto vorbisMin = Popularity{ .rating = 20.0, .scale = TagType::VorbisComment };
|
|
const auto vorbisMax = Popularity{ .rating = 100.0, .scale = TagType::OggVorbisComment };
|
|
const auto vorbisMiddle = Popularity{ .rating = 60.0, .scale = TagType::OggVorbisComment };
|
|
const auto mkvMin = Popularity{ .rating = 0.0, .scale = TagType::MatroskaTag };
|
|
const auto mkvMax = Popularity{ .rating = 5.0, .scale = TagType::MatroskaTag };
|
|
const auto mkvMiddle = Popularity{ .rating = 2.5, .scale = TagType::MatroskaTag };
|
|
for (const auto &rawZero : { id3zero, vorbisZero }) {
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("zero: raw to generic", genericZero.rating, rawZero.scaled(TagType::Unspecified).rating);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("zero: generic to raw ", rawZero.rating, genericZero.scaled(rawZero.scale).rating);
|
|
}
|
|
for (const auto &rawMin : { id3min, vorbisMin, mkvMin }) {
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("min: raw to generic", genericMin.rating, rawMin.scaled(TagType::Unspecified).rating);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("min: generic to raw ", rawMin.rating, genericMin.scaled(rawMin.scale).rating);
|
|
}
|
|
for (const auto &rawMax : { id3max, vorbisMax, mkvMax }) {
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("max: raw to generic", genericMax.rating, rawMax.scaled(TagType::Unspecified).rating);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("max: generic to raw ", rawMax.rating, genericMax.scaled(rawMax.scale).rating);
|
|
}
|
|
for (const auto &rawMiddle : { id3middle, vorbisMiddle, mkvMiddle }) {
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("middle: raw to generic", genericMiddle.rating, rawMiddle.scaled(TagType::Unspecified).rating);
|
|
CPPUNIT_ASSERT_EQUAL_MESSAGE("middle: generic to raw ", rawMiddle.rating, genericMiddle.scaled(rawMiddle.scale).rating);
|
|
}
|
|
}
|