summaryrefslogtreecommitdiffstats
path: root/third_party/jpeg-xl/lib/jxl/alpha_test.cc
blob: 6e26e77be6a7c294809bb017a5eec8683f0f6ebe (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// Copyright (c) the JPEG XL Project 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 "lib/jxl/alpha.h"

#include <array>

#include "lib/jxl/base/common.h"
#include "lib/jxl/testing.h"

namespace jxl {
namespace {

AlphaBlendingInputLayer makeAbil(const Color& rgb, const float& a) {
  const float* data = rgb.data();
  return {data, data + 1, data + 2, &a};
}

AlphaBlendingOutput makeAbo(Color& rgb, float& a) {
  float* data = rgb.data();
  return {data, data + 1, data + 2, &a};
}

TEST(AlphaTest, BlendingWithNonPremultiplied) {
  const Color bg_rgb{100, 110, 120};
  const float bg_a = 180.f / 255;
  const Color fg_rgb{25, 21, 23};
  const float fg_a = 15420.f / 65535;
  const float fg_a2 = 2.0f;
  Color out_rgb;
  float out_a;
  PerformAlphaBlending(
      /*bg=*/makeAbil(bg_rgb, bg_a),
      /*fg=*/makeAbil(fg_rgb, fg_a),
      /*out=*/makeAbo(out_rgb, out_a), 1,
      /*alpha_is_premultiplied=*/false, /*clamp=*/false);
  EXPECT_ARRAY_NEAR(out_rgb, (Color{77.2f, 83.0f, 90.6f}), 0.05f);
  EXPECT_NEAR(out_a, 3174.f / 4095, 1e-5);
  PerformAlphaBlending(
      /*bg=*/makeAbil(bg_rgb, bg_a),
      /*fg=*/makeAbil(fg_rgb, fg_a2),
      /*out=*/makeAbo(out_rgb, out_a), 1,
      /*alpha_is_premultiplied=*/false, /*clamp=*/true);
  EXPECT_ARRAY_NEAR(out_rgb, fg_rgb, 0.05f);
  EXPECT_NEAR(out_a, 1.0f, 1e-5);
}

TEST(AlphaTest, BlendingWithPremultiplied) {
  const Color bg_rgb{100, 110, 120};
  const float bg_a = 180.f / 255;
  const Color fg_rgb{25, 21, 23};
  const float fg_a = 15420.f / 65535;
  const float fg_a2 = 2.0f;
  Color out_rgb;
  float out_a;
  PerformAlphaBlending(
      /*bg=*/makeAbil(bg_rgb, bg_a),
      /*fg=*/makeAbil(fg_rgb, fg_a),
      /*out=*/makeAbo(out_rgb, out_a), 1,
      /*alpha_is_premultiplied=*/true, /*clamp=*/false);
  EXPECT_ARRAY_NEAR(out_rgb, (Color{101.5f, 105.1f, 114.8f}), 0.05f);
  EXPECT_NEAR(out_a, 3174.f / 4095, 1e-5);
  PerformAlphaBlending(
      /*bg=*/makeAbil(bg_rgb, bg_a),
      /*fg=*/makeAbil(fg_rgb, fg_a2),
      /*out=*/makeAbo(out_rgb, out_a), 1,
      /*alpha_is_premultiplied=*/true, /*clamp=*/true);
  EXPECT_ARRAY_NEAR(out_rgb, fg_rgb, 0.05f);
  EXPECT_NEAR(out_a, 1.0f, 1e-5);
}

TEST(AlphaTest, Mul) {
  const float bg = 100;
  const float fg = 25;
  float out;
  PerformMulBlending(&bg, &fg, &out, 1, /*clamp=*/false);
  EXPECT_NEAR(out, fg * bg, .05f);
  PerformMulBlending(&bg, &fg, &out, 1, /*clamp=*/true);
  EXPECT_NEAR(out, bg, .05f);
}

TEST(AlphaTest, PremultiplyAndUnpremultiply) {
  using F4 = std::array<float, 4>;
  const F4 alpha{0.f, 63.f / 255, 127.f / 255, 1.f};
  F4 r{120, 130, 140, 150};
  F4 g{124, 134, 144, 154};
  F4 b{127, 137, 147, 157};

  PremultiplyAlpha(r.data(), g.data(), b.data(), alpha.data(), alpha.size());
  EXPECT_ARRAY_NEAR(r, (F4{0.0f, 130 * 63.f / 255, 140 * 127.f / 255, 150}),
                    1e-5f);
  EXPECT_ARRAY_NEAR(g, (F4{0.0f, 134 * 63.f / 255, 144 * 127.f / 255, 154}),
                    1e-5f);
  EXPECT_ARRAY_NEAR(b, (F4{0.0f, 137 * 63.f / 255, 147 * 127.f / 255, 157}),
                    1e-5f);

  UnpremultiplyAlpha(r.data(), g.data(), b.data(), alpha.data(), alpha.size());
  EXPECT_ARRAY_NEAR(r, (F4{120, 130, 140, 150}), 1e-4f);
  EXPECT_ARRAY_NEAR(g, (F4{124, 134, 144, 154}), 1e-4f);
  EXPECT_ARRAY_NEAR(b, (F4{127, 137, 147, 157}), 1e-4f);
}

TEST(AlphaTest, UnpremultiplyAndPremultiply) {
  using F4 = std::array<float, 4>;
  const F4 alpha{0.f, 63.f / 255, 127.f / 255, 1.f};
  F4 r{50, 60, 70, 80};
  F4 g{54, 64, 74, 84};
  F4 b{57, 67, 77, 87};

  UnpremultiplyAlpha(r.data(), g.data(), b.data(), alpha.data(), alpha.size());
  EXPECT_ARRAY_NEAR(
      r, (F4{50.0f * (1 << 26), 60 * 255.f / 63, 70 * 255.f / 127, 80}), 1e-4f);
  EXPECT_ARRAY_NEAR(
      g, (F4{54.0f * (1 << 26), 64 * 255.f / 63, 74 * 255.f / 127, 84}), 1e-4f);
  EXPECT_ARRAY_NEAR(
      b, (F4{57.0f * (1 << 26), 67 * 255.f / 63, 77 * 255.f / 127, 87}), 1e-4f);

  PremultiplyAlpha(r.data(), g.data(), b.data(), alpha.data(), alpha.size());
  EXPECT_ARRAY_NEAR(r, (F4{50, 60, 70, 80}), 1e-4);
  EXPECT_ARRAY_NEAR(g, (F4{54, 64, 74, 84}), 1e-4);
  EXPECT_ARRAY_NEAR(b, (F4{57, 67, 77, 87}), 1e-4);
}

}  // namespace
}  // namespace jxl