| // Copyright (c) 2016 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/logging.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "ui/gfx/color_space.h" |
| #include "ui/gfx/color_transform.h" |
| #include "ui/gfx/icc_profile.h" |
| #include "ui/gfx/test/icc_profiles.h" |
| #include "ui/gfx/transform.h" |
| |
| namespace gfx { |
| |
| // Internal functions, exposted for testing. |
| GFX_EXPORT Transform GetPrimaryMatrix(ColorSpace::PrimaryID id); |
| GFX_EXPORT Transform GetTransferMatrix(ColorSpace::MatrixID id); |
| GFX_EXPORT float ToLinear(ColorSpace::TransferID id, float v); |
| GFX_EXPORT float FromLinear(ColorSpace::TransferID id, float v); |
| |
| ColorSpace::PrimaryID all_primaries[] = { |
| ColorSpace::PrimaryID::BT709, ColorSpace::PrimaryID::BT470M, |
| ColorSpace::PrimaryID::BT470BG, ColorSpace::PrimaryID::SMPTE170M, |
| ColorSpace::PrimaryID::SMPTE240M, ColorSpace::PrimaryID::FILM, |
| ColorSpace::PrimaryID::BT2020, ColorSpace::PrimaryID::SMPTEST428_1, |
| ColorSpace::PrimaryID::SMPTEST431_2, ColorSpace::PrimaryID::SMPTEST432_1, |
| }; |
| |
| ColorSpace::TransferID all_transfers[] = { |
| ColorSpace::TransferID::BT709, ColorSpace::TransferID::GAMMA22, |
| ColorSpace::TransferID::GAMMA28, ColorSpace::TransferID::SMPTE170M, |
| ColorSpace::TransferID::SMPTE240M, ColorSpace::TransferID::LINEAR, |
| ColorSpace::TransferID::LOG, ColorSpace::TransferID::LOG_SQRT, |
| ColorSpace::TransferID::IEC61966_2_4, ColorSpace::TransferID::BT1361_ECG, |
| ColorSpace::TransferID::IEC61966_2_1, ColorSpace::TransferID::BT2020_10, |
| ColorSpace::TransferID::BT2020_12, ColorSpace::TransferID::SMPTEST2084, |
| ColorSpace::TransferID::ARIB_STD_B67, |
| // This one is weird as the non-linear numbers are not between 0 and 1. |
| // TODO(hubbe): Test this separately. |
| // ColorSpace::TransferID::SMPTEST428_1, |
| }; |
| |
| ColorSpace::MatrixID all_matrices[] = { |
| ColorSpace::MatrixID::RGB, ColorSpace::MatrixID::BT709, |
| ColorSpace::MatrixID::UNSPECIFIED, ColorSpace::MatrixID::RESERVED, |
| ColorSpace::MatrixID::FCC, ColorSpace::MatrixID::BT470BG, |
| ColorSpace::MatrixID::SMPTE170M, ColorSpace::MatrixID::SMPTE240M, |
| |
| // YCOCG produces lots of negative values which isn't compatible with many |
| // transfer functions. |
| // TODO(hubbe): Test this separately. |
| // ColorSpace::MatrixID::YCOCG, |
| ColorSpace::MatrixID::BT2020_NCL, ColorSpace::MatrixID::BT2020_CL, |
| ColorSpace::MatrixID::YDZDX, |
| }; |
| |
| ColorSpace::RangeID all_ranges[] = {ColorSpace::RangeID::UNSPECIFIED, |
| ColorSpace::RangeID::FULL, |
| ColorSpace::RangeID::LIMITED, |
| ColorSpace::RangeID::DERIVED}; |
| |
| TEST(SimpleColorSpace, BT709toSRGB) { |
| ColorSpace bt709 = ColorSpace::CreateREC709(); |
| ColorSpace sRGB = ColorSpace::CreateSRGB(); |
| std::unique_ptr<ColorTransform> t(ColorTransform::NewColorTransform( |
| bt709, sRGB, ColorTransform::Intent::INTENT_ABSOLUTE)); |
| |
| ColorTransform::TriStim tmp(16.0f / 255.0f, 0.5f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 0.0f, 0.001f); |
| |
| tmp = ColorTransform::TriStim(235.0f / 255.0f, 0.5f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 1.0f, 0.001f); |
| |
| // Test a blue color |
| tmp = ColorTransform::TriStim(128.0f / 255.0f, 240.0f / 255.0f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_GT(tmp.z(), tmp.x()); |
| EXPECT_GT(tmp.z(), tmp.y()); |
| } |
| |
| TEST(SimpleColorSpace, BT709toSRGBICC) { |
| ICCProfile srgb_icc = ICCProfileForTestingSRGB(); |
| ColorSpace bt709 = ColorSpace::CreateREC709(); |
| ColorSpace sRGB = srgb_icc.GetColorSpace(); |
| std::unique_ptr<ColorTransform> t(ColorTransform::NewColorTransform( |
| bt709, sRGB, ColorTransform::Intent::INTENT_ABSOLUTE)); |
| |
| ColorTransform::TriStim tmp(16.0f / 255.0f, 0.5f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 0.0f, 0.001f); |
| |
| tmp = ColorTransform::TriStim(235.0f / 255.0f, 0.5f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 1.0f, 0.001f); |
| |
| // Test a blue color |
| tmp = ColorTransform::TriStim(128.0f / 255.0f, 240.0f / 255.0f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_GT(tmp.z(), tmp.x()); |
| EXPECT_GT(tmp.z(), tmp.y()); |
| } |
| |
| TEST(SimpleColorSpace, GetColorSpace) { |
| ICCProfile srgb_icc = ICCProfileForTestingSRGB(); |
| ColorSpace sRGB = srgb_icc.GetColorSpace(); |
| ColorSpace sRGB2 = sRGB; |
| |
| // Prevent sRGB2 from using a cached ICC profile. |
| sRGB2.icc_profile_id_ = 0; |
| |
| std::unique_ptr<ColorTransform> t(ColorTransform::NewColorTransform( |
| sRGB, sRGB2, ColorTransform::Intent::INTENT_ABSOLUTE)); |
| |
| ColorTransform::TriStim tmp(1.0f, 1.0f, 1.0f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 1.0f, 0.001f); |
| |
| tmp = ColorTransform::TriStim(1.0f, 0.0f, 0.0f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 0.0f, 0.001f); |
| |
| tmp = ColorTransform::TriStim(0.0f, 1.0f, 0.0f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 0.0f, 0.001f); |
| |
| tmp = ColorTransform::TriStim(0.0f, 0.0f, 1.0f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 1.0f, 0.001f); |
| } |
| |
| TEST(SimpleColorSpace, UnknownToSRGB) { |
| ColorSpace unknown; |
| ColorSpace sRGB = ColorSpace::CreateSRGB(); |
| std::unique_ptr<ColorTransform> t(ColorTransform::NewColorTransform( |
| unknown, sRGB, ColorTransform::Intent::INTENT_PERCEPTUAL)); |
| |
| ColorTransform::TriStim tmp(16.0f / 255.0f, 0.5f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 0.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 0.0f, 0.001f); |
| |
| tmp = ColorTransform::TriStim(235.0f / 255.0f, 0.5f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_NEAR(tmp.x(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.y(), 1.0f, 0.001f); |
| EXPECT_NEAR(tmp.z(), 1.0f, 0.001f); |
| |
| // Test a blue color |
| tmp = ColorTransform::TriStim(128.0f / 255.0f, 240.0f / 255.0f, 0.5f); |
| t->transform(&tmp, 1); |
| EXPECT_GT(tmp.z(), tmp.x()); |
| EXPECT_GT(tmp.z(), tmp.y()); |
| } |
| |
| class PrimaryTest : public testing::TestWithParam<ColorSpace::PrimaryID> {}; |
| |
| TEST_P(PrimaryTest, checkInvertible) { |
| EXPECT_EQ(GetPrimaryMatrix(GetParam()).matrix().get(3, 3), 1.0f); |
| // Check that all primary matrices are invertable. |
| EXPECT_TRUE(GetPrimaryMatrix(GetParam()).IsInvertible()); |
| } |
| |
| INSTANTIATE_TEST_CASE_P(ColorSpace, |
| PrimaryTest, |
| testing::ValuesIn(all_primaries)); |
| |
| class MatrixTest : public testing::TestWithParam<ColorSpace::MatrixID> {}; |
| |
| TEST_P(MatrixTest, checkInvertible) { |
| EXPECT_EQ(GetTransferMatrix(GetParam()).matrix().get(3, 3), 1.0f); |
| // Check that all transfer matrices are invertable. |
| EXPECT_TRUE(GetTransferMatrix(GetParam()).IsInvertible()); |
| }; |
| |
| INSTANTIATE_TEST_CASE_P(ColorSpace, |
| MatrixTest, |
| testing::ValuesIn(all_matrices)); |
| |
| class TransferTest : public testing::TestWithParam<ColorSpace::TransferID> {}; |
| |
| TEST_P(TransferTest, basicTest) { |
| for (float x = 0.0f; x <= 1.0f; x += 1.0f / 128.0f) { |
| float linear = ToLinear(GetParam(), x); |
| float x2 = FromLinear(GetParam(), linear); |
| EXPECT_NEAR(x, x2, 0.001f); |
| } |
| } |
| |
| INSTANTIATE_TEST_CASE_P(ColorSpace, |
| TransferTest, |
| testing::ValuesIn(all_transfers)); |
| |
| typedef std::tr1::tuple<ColorSpace::PrimaryID, |
| ColorSpace::TransferID, |
| ColorSpace::MatrixID, |
| ColorSpace::RangeID> |
| ColorSpaceTestData; |
| |
| class ColorSpaceTest : public testing::TestWithParam<ColorSpaceTestData> { |
| public: |
| ColorSpaceTest() |
| : color_space_(std::tr1::get<0>(GetParam()), |
| std::tr1::get<1>(GetParam()), |
| std::tr1::get<2>(GetParam()), |
| std::tr1::get<3>(GetParam())) {} |
| |
| protected: |
| ColorSpace color_space_; |
| }; |
| |
| TEST_P(ColorSpaceTest, testNullTransform) { |
| std::unique_ptr<ColorTransform> t(ColorTransform::NewColorTransform( |
| color_space_, color_space_, ColorTransform::Intent::INTENT_ABSOLUTE)); |
| ColorTransform::TriStim tristim(0.4f, 0.5f, 0.6f); |
| t->transform(&tristim, 1); |
| EXPECT_NEAR(tristim.x(), 0.4f, 0.001f); |
| EXPECT_NEAR(tristim.y(), 0.5f, 0.001f); |
| EXPECT_NEAR(tristim.z(), 0.6f, 0.001f); |
| } |
| |
| TEST_P(ColorSpaceTest, toXYZandBack) { |
| std::unique_ptr<ColorTransform> t1(ColorTransform::NewColorTransform( |
| color_space_, ColorSpace::CreateXYZD50(), |
| ColorTransform::Intent::INTENT_ABSOLUTE)); |
| std::unique_ptr<ColorTransform> t2(ColorTransform::NewColorTransform( |
| ColorSpace::CreateXYZD50(), color_space_, |
| ColorTransform::Intent::INTENT_ABSOLUTE)); |
| ColorTransform::TriStim tristim(0.4f, 0.5f, 0.6f); |
| t1->transform(&tristim, 1); |
| t2->transform(&tristim, 1); |
| EXPECT_NEAR(tristim.x(), 0.4f, 0.001f); |
| EXPECT_NEAR(tristim.y(), 0.5f, 0.001f); |
| EXPECT_NEAR(tristim.z(), 0.6f, 0.001f); |
| } |
| |
| INSTANTIATE_TEST_CASE_P( |
| A, |
| ColorSpaceTest, |
| testing::Combine(testing::ValuesIn(all_primaries), |
| testing::ValuesIn(all_transfers), |
| testing::Values(ColorSpace::MatrixID::BT709), |
| testing::Values(ColorSpace::RangeID::LIMITED))); |
| |
| INSTANTIATE_TEST_CASE_P( |
| B, |
| ColorSpaceTest, |
| testing::Combine(testing::Values(ColorSpace::PrimaryID::BT709), |
| testing::ValuesIn(all_transfers), |
| testing::ValuesIn(all_matrices), |
| testing::ValuesIn(all_ranges))); |
| |
| INSTANTIATE_TEST_CASE_P( |
| C, |
| ColorSpaceTest, |
| testing::Combine(testing::ValuesIn(all_primaries), |
| testing::Values(ColorSpace::TransferID::BT709), |
| testing::ValuesIn(all_matrices), |
| testing::ValuesIn(all_ranges))); |
| } // namespace |