ColorSpace.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546
  1. /*
  2. * Copyright (c) 2021-2022, Matthew Olsson <mattco@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibGfx/ICC/WellKnownProfiles.h>
  7. #include <LibPDF/ColorSpace.h>
  8. #include <LibPDF/CommonNames.h>
  9. #include <LibPDF/Document.h>
  10. #include <LibPDF/ObjectDerivatives.h>
  11. namespace PDF {
  12. RefPtr<Gfx::ICC::Profile> ICCBasedColorSpace::s_srgb_profile;
  13. #define ENUMERATE(name, may_be_specified_directly) \
  14. ColorSpaceFamily ColorSpaceFamily::name { #name, may_be_specified_directly };
  15. ENUMERATE_COLOR_SPACE_FAMILIES(ENUMERATE);
  16. #undef ENUMERATE
  17. PDFErrorOr<ColorSpaceFamily> ColorSpaceFamily::get(DeprecatedFlyString const& family_name)
  18. {
  19. #define ENUMERATE(f_name, may_be_specified_directly) \
  20. if (family_name == f_name.name()) { \
  21. return ColorSpaceFamily::f_name; \
  22. }
  23. ENUMERATE_COLOR_SPACE_FAMILIES(ENUMERATE)
  24. #undef ENUMERATE
  25. dbgln_if(PDF_DEBUG, "Unknown ColorSpace family: {}", family_name);
  26. return Error(Error::Type::MalformedPDF, "Unknown ColorSpace family"_string);
  27. }
  28. PDFErrorOr<NonnullRefPtr<ColorSpace>> ColorSpace::create(DeprecatedFlyString const& name)
  29. {
  30. // Simple color spaces with no parameters, which can be specified directly
  31. if (name == CommonNames::DeviceGray)
  32. return DeviceGrayColorSpace::the();
  33. if (name == CommonNames::DeviceRGB)
  34. return DeviceRGBColorSpace::the();
  35. if (name == CommonNames::DeviceCMYK)
  36. return DeviceCMYKColorSpace::the();
  37. if (name == CommonNames::Pattern)
  38. return Error::rendering_unsupported_error("Pattern color spaces not yet implemented");
  39. VERIFY_NOT_REACHED();
  40. }
  41. PDFErrorOr<NonnullRefPtr<ColorSpace>> ColorSpace::create(Document* document, NonnullRefPtr<ArrayObject> color_space_array)
  42. {
  43. auto color_space_name = TRY(color_space_array->get_name_at(document, 0))->name();
  44. Vector<Value> parameters;
  45. parameters.ensure_capacity(color_space_array->size() - 1);
  46. for (size_t i = 1; i < color_space_array->size(); i++)
  47. parameters.unchecked_append(color_space_array->at(i));
  48. if (color_space_name == CommonNames::CalGray)
  49. return TRY(CalGrayColorSpace::create(document, move(parameters)));
  50. if (color_space_name == CommonNames::CalRGB)
  51. return TRY(CalRGBColorSpace::create(document, move(parameters)));
  52. if (color_space_name == CommonNames::DeviceN)
  53. return TRY(DeviceNColorSpace::create(document, move(parameters)));
  54. if (color_space_name == CommonNames::ICCBased)
  55. return TRY(ICCBasedColorSpace::create(document, move(parameters)));
  56. if (color_space_name == CommonNames::Indexed)
  57. return Error::rendering_unsupported_error("Indexed color spaces not yet implemented");
  58. if (color_space_name == CommonNames::Lab)
  59. return TRY(LabColorSpace::create(document, move(parameters)));
  60. if (color_space_name == CommonNames::Pattern)
  61. return Error::rendering_unsupported_error("Pattern color spaces not yet implemented");
  62. if (color_space_name == CommonNames::Separation)
  63. return TRY(SeparationColorSpace::create(document, move(parameters)));
  64. dbgln("Unknown color space: {}", color_space_name);
  65. return Error::rendering_unsupported_error("unknown color space");
  66. }
  67. NonnullRefPtr<DeviceGrayColorSpace> DeviceGrayColorSpace::the()
  68. {
  69. static auto instance = adopt_ref(*new DeviceGrayColorSpace());
  70. return instance;
  71. }
  72. PDFErrorOr<Color> DeviceGrayColorSpace::color(ReadonlySpan<Value> arguments) const
  73. {
  74. VERIFY(arguments.size() == 1);
  75. auto gray = static_cast<u8>(arguments[0].to_float() * 255.0f);
  76. return Color(gray, gray, gray);
  77. }
  78. Vector<float> DeviceGrayColorSpace::default_decode() const
  79. {
  80. return { 0.0f, 1.0f };
  81. }
  82. NonnullRefPtr<DeviceRGBColorSpace> DeviceRGBColorSpace::the()
  83. {
  84. static auto instance = adopt_ref(*new DeviceRGBColorSpace());
  85. return instance;
  86. }
  87. PDFErrorOr<Color> DeviceRGBColorSpace::color(ReadonlySpan<Value> arguments) const
  88. {
  89. VERIFY(arguments.size() == 3);
  90. auto r = static_cast<u8>(arguments[0].to_float() * 255.0f);
  91. auto g = static_cast<u8>(arguments[1].to_float() * 255.0f);
  92. auto b = static_cast<u8>(arguments[2].to_float() * 255.0f);
  93. return Color(r, g, b);
  94. }
  95. Vector<float> DeviceRGBColorSpace::default_decode() const
  96. {
  97. return { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f };
  98. }
  99. NonnullRefPtr<DeviceCMYKColorSpace> DeviceCMYKColorSpace::the()
  100. {
  101. static auto instance = adopt_ref(*new DeviceCMYKColorSpace());
  102. return instance;
  103. }
  104. PDFErrorOr<Color> DeviceCMYKColorSpace::color(ReadonlySpan<Value> arguments) const
  105. {
  106. VERIFY(arguments.size() == 4);
  107. auto c = arguments[0].to_float();
  108. auto m = arguments[1].to_float();
  109. auto y = arguments[2].to_float();
  110. auto k = arguments[3].to_float();
  111. return Color::from_cmyk(c, m, y, k);
  112. }
  113. Vector<float> DeviceCMYKColorSpace::default_decode() const
  114. {
  115. return { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f };
  116. }
  117. PDFErrorOr<NonnullRefPtr<DeviceNColorSpace>> DeviceNColorSpace::create(Document*, Vector<Value>&& parameters)
  118. {
  119. // "[ /DeviceN names alternateSpace tintTransform ]
  120. // or
  121. // [ /DeviceN names alternateSpace tintTransform attributes ]"
  122. if (parameters.size() != 4 && parameters.size() != 5)
  123. return Error { Error::Type::MalformedPDF, "DevicN color space expects 4 or 5 parameters" };
  124. // "The names parameter is an array of name objects specifying the individual color components.
  125. // The length of the array determines the number of components in the DeviceN color space"
  126. auto names = parameters[0].get<NonnullRefPtr<Object>>()->cast<ArrayObject>();
  127. // "The alternateSpace parameter is an array or name object that can be any device or CIE-based color space
  128. // but not another special color space (Pattern, Indexed, Separation, or DeviceN)."
  129. // FIXME: Implement.
  130. return adopt_ref(*new DeviceNColorSpace(names->size()));
  131. }
  132. PDFErrorOr<Color> DeviceNColorSpace::color(ReadonlySpan<Value>) const
  133. {
  134. return Error::rendering_unsupported_error("DeviceN color spaces not yet implemented");
  135. }
  136. int DeviceNColorSpace::number_of_components() const
  137. {
  138. return m_number_of_components;
  139. }
  140. Vector<float> DeviceNColorSpace::default_decode() const
  141. {
  142. Vector<float> decoding_ranges;
  143. for (u8 i = 0; i < number_of_components(); i++) {
  144. decoding_ranges.append(0.0);
  145. decoding_ranges.append(1.0);
  146. }
  147. return decoding_ranges;
  148. }
  149. DeviceNColorSpace::DeviceNColorSpace(size_t number_of_components)
  150. : m_number_of_components(number_of_components)
  151. {
  152. }
  153. constexpr Array<float, 3> matrix_multiply(Array<float, 9> a, Array<float, 3> b)
  154. {
  155. return Array<float, 3> {
  156. a[0] * b[0] + a[1] * b[1] + a[2] * b[2],
  157. a[3] * b[0] + a[4] * b[1] + a[5] * b[2],
  158. a[6] * b[0] + a[7] * b[1] + a[8] * b[2]
  159. };
  160. }
  161. // Converts to a flat XYZ space with white point = (1, 1, 1)
  162. // Step 2 of https://www.color.org/adobebpc.pdf
  163. constexpr Array<float, 3> flatten_and_normalize_whitepoint(Array<float, 3> whitepoint, Array<float, 3> xyz)
  164. {
  165. VERIFY(whitepoint[1] == 1.0f);
  166. return {
  167. (1.0f / whitepoint[0]) * xyz[0],
  168. xyz[1],
  169. (1.0f / whitepoint[2]) * xyz[2],
  170. };
  171. }
  172. constexpr float decode_l(float input)
  173. {
  174. constexpr float decode_l_scaling_constant = 0.00110705646f; // (((8 + 16) / 116) ^ 3) / 8
  175. if (input < 0.0f)
  176. return -decode_l(-input);
  177. if (input >= 0.0f && input <= 8.0f)
  178. return input * decode_l_scaling_constant;
  179. return powf(((input + 16.0f) / 116.0f), 3.0f);
  180. }
  181. constexpr Array<float, 3> scale_black_point(Array<float, 3> blackpoint, Array<float, 3> xyz)
  182. {
  183. auto y_dst = decode_l(0); // DestinationBlackPoint is just [0, 0, 0]
  184. auto y_src = decode_l(blackpoint[0]);
  185. auto scale = (1 - y_dst) / (1 - y_src);
  186. auto offset = 1 - scale;
  187. return {
  188. xyz[0] * scale + offset,
  189. xyz[1] * scale + offset,
  190. xyz[2] * scale + offset,
  191. };
  192. }
  193. // https://en.wikipedia.org/wiki/Illuminant_D65
  194. constexpr Array<float, 3> convert_to_d65(Array<float, 3> xyz)
  195. {
  196. constexpr float d65x = 0.95047f;
  197. constexpr float d65y = 1.0f;
  198. constexpr float d65z = 1.08883f;
  199. return { xyz[0] * d65x, xyz[1] * d65y, xyz[2] * d65z };
  200. }
  201. // https://en.wikipedia.org/wiki/SRGB
  202. constexpr Array<float, 3> convert_to_srgb(Array<float, 3> xyz)
  203. {
  204. // See the sRGB D65 [M]^-1 matrix in the following page
  205. // http://www.brucelindbloom.com/index.html?Eqn_RGB_XYZ_Matrix.html
  206. constexpr Array<float, 9> conversion_matrix = {
  207. 3.2404542,
  208. -1.5371385,
  209. -0.4985314,
  210. -0.969266,
  211. 1.8760108,
  212. 0.0415560,
  213. 0.0556434,
  214. -0.2040259,
  215. 1.0572252,
  216. };
  217. auto linear_srgb = matrix_multiply(conversion_matrix, xyz);
  218. // FIXME: Use the real sRGB curve by replacing this function with Gfx::ICC::sRGB().from_pcs().
  219. return { pow(linear_srgb[0], 1.0f / 2.2f), pow(linear_srgb[1], 1.0f / 2.2f), pow(linear_srgb[2], 1.0f / 2.2f) };
  220. }
  221. PDFErrorOr<NonnullRefPtr<CalGrayColorSpace>> CalGrayColorSpace::create(Document* document, Vector<Value>&& parameters)
  222. {
  223. if (parameters.size() != 1)
  224. return Error { Error::Type::MalformedPDF, "Gray color space expects one parameter" };
  225. auto param = parameters[0];
  226. if (!param.has<NonnullRefPtr<Object>>() || !param.get<NonnullRefPtr<Object>>()->is<DictObject>())
  227. return Error { Error::Type::MalformedPDF, "Gray color space expects a dict parameter" };
  228. auto dict = param.get<NonnullRefPtr<Object>>()->cast<DictObject>();
  229. if (!dict->contains(CommonNames::WhitePoint))
  230. return Error { Error::Type::MalformedPDF, "Gray color space expects a Whitepoint key" };
  231. auto white_point_array = TRY(dict->get_array(document, CommonNames::WhitePoint));
  232. if (white_point_array->size() != 3)
  233. return Error { Error::Type::MalformedPDF, "Gray color space expects 3 Whitepoint parameters" };
  234. auto color_space = adopt_ref(*new CalGrayColorSpace());
  235. color_space->m_whitepoint[0] = white_point_array->at(0).to_float();
  236. color_space->m_whitepoint[1] = white_point_array->at(1).to_float();
  237. color_space->m_whitepoint[2] = white_point_array->at(2).to_float();
  238. if (color_space->m_whitepoint[1] != 1.0f)
  239. return Error { Error::Type::MalformedPDF, "Gray color space expects 2nd Whitepoint to be 1.0" };
  240. if (dict->contains(CommonNames::BlackPoint)) {
  241. auto black_point_array = TRY(dict->get_array(document, CommonNames::BlackPoint));
  242. if (black_point_array->size() == 3) {
  243. color_space->m_blackpoint[0] = black_point_array->at(0).to_float();
  244. color_space->m_blackpoint[1] = black_point_array->at(1).to_float();
  245. color_space->m_blackpoint[2] = black_point_array->at(2).to_float();
  246. }
  247. }
  248. if (dict->contains(CommonNames::Gamma)) {
  249. color_space->m_gamma = TRY(document->resolve(dict->get_value(CommonNames::Gamma))).to_float();
  250. }
  251. return color_space;
  252. }
  253. PDFErrorOr<Color> CalGrayColorSpace::color(ReadonlySpan<Value> arguments) const
  254. {
  255. VERIFY(arguments.size() == 1);
  256. auto a = clamp(arguments[0].to_float(), 0.0f, 1.0f);
  257. auto ag = powf(a, m_gamma);
  258. auto x = m_whitepoint[0] * ag;
  259. auto y = m_whitepoint[1] * ag;
  260. auto z = m_whitepoint[2] * ag;
  261. auto flattened_xyz = flatten_and_normalize_whitepoint(m_whitepoint, { x, y, z });
  262. auto scaled_black_point_xyz = scale_black_point(m_blackpoint, flattened_xyz);
  263. auto d65_normalized = convert_to_d65(scaled_black_point_xyz);
  264. auto srgb = convert_to_srgb(d65_normalized);
  265. auto red = static_cast<u8>(clamp(srgb[0], 0.0f, 1.0f) * 255.0f);
  266. auto green = static_cast<u8>(clamp(srgb[1], 0.0f, 1.0f) * 255.0f);
  267. auto blue = static_cast<u8>(clamp(srgb[2], 0.0f, 1.0f) * 255.0f);
  268. return Color(red, green, blue);
  269. }
  270. Vector<float> CalGrayColorSpace::default_decode() const
  271. {
  272. return { 0.0f, 1.0f };
  273. }
  274. PDFErrorOr<NonnullRefPtr<CalRGBColorSpace>> CalRGBColorSpace::create(Document* document, Vector<Value>&& parameters)
  275. {
  276. if (parameters.size() != 1)
  277. return Error { Error::Type::MalformedPDF, "RGB color space expects one parameter" };
  278. auto param = parameters[0];
  279. if (!param.has<NonnullRefPtr<Object>>() || !param.get<NonnullRefPtr<Object>>()->is<DictObject>())
  280. return Error { Error::Type::MalformedPDF, "RGB color space expects a dict parameter" };
  281. auto dict = param.get<NonnullRefPtr<Object>>()->cast<DictObject>();
  282. if (!dict->contains(CommonNames::WhitePoint))
  283. return Error { Error::Type::MalformedPDF, "RGB color space expects a Whitepoint key" };
  284. auto white_point_array = TRY(dict->get_array(document, CommonNames::WhitePoint));
  285. if (white_point_array->size() != 3)
  286. return Error { Error::Type::MalformedPDF, "RGB color space expects 3 Whitepoint parameters" };
  287. auto color_space = adopt_ref(*new CalRGBColorSpace());
  288. color_space->m_whitepoint[0] = white_point_array->at(0).to_float();
  289. color_space->m_whitepoint[1] = white_point_array->at(1).to_float();
  290. color_space->m_whitepoint[2] = white_point_array->at(2).to_float();
  291. if (color_space->m_whitepoint[1] != 1.0f)
  292. return Error { Error::Type::MalformedPDF, "RGB color space expects 2nd Whitepoint to be 1.0" };
  293. if (dict->contains(CommonNames::BlackPoint)) {
  294. auto black_point_array = TRY(dict->get_array(document, CommonNames::BlackPoint));
  295. if (black_point_array->size() == 3) {
  296. color_space->m_blackpoint[0] = black_point_array->at(0).to_float();
  297. color_space->m_blackpoint[1] = black_point_array->at(1).to_float();
  298. color_space->m_blackpoint[2] = black_point_array->at(2).to_float();
  299. }
  300. }
  301. if (dict->contains(CommonNames::Gamma)) {
  302. auto gamma_array = TRY(dict->get_array(document, CommonNames::Gamma));
  303. if (gamma_array->size() == 3) {
  304. color_space->m_gamma[0] = gamma_array->at(0).to_float();
  305. color_space->m_gamma[1] = gamma_array->at(1).to_float();
  306. color_space->m_gamma[2] = gamma_array->at(2).to_float();
  307. }
  308. }
  309. if (dict->contains(CommonNames::Matrix)) {
  310. auto matrix_array = TRY(dict->get_array(document, CommonNames::Matrix));
  311. if (matrix_array->size() == 9) {
  312. color_space->m_matrix[0] = matrix_array->at(0).to_float();
  313. color_space->m_matrix[1] = matrix_array->at(1).to_float();
  314. color_space->m_matrix[2] = matrix_array->at(2).to_float();
  315. color_space->m_matrix[3] = matrix_array->at(3).to_float();
  316. color_space->m_matrix[4] = matrix_array->at(4).to_float();
  317. color_space->m_matrix[5] = matrix_array->at(5).to_float();
  318. color_space->m_matrix[6] = matrix_array->at(6).to_float();
  319. color_space->m_matrix[7] = matrix_array->at(7).to_float();
  320. color_space->m_matrix[8] = matrix_array->at(8).to_float();
  321. }
  322. }
  323. return color_space;
  324. }
  325. PDFErrorOr<Color> CalRGBColorSpace::color(ReadonlySpan<Value> arguments) const
  326. {
  327. VERIFY(arguments.size() == 3);
  328. auto a = clamp(arguments[0].to_float(), 0.0f, 1.0f);
  329. auto b = clamp(arguments[1].to_float(), 0.0f, 1.0f);
  330. auto c = clamp(arguments[2].to_float(), 0.0f, 1.0f);
  331. auto agr = powf(a, m_gamma[0]);
  332. auto bgg = powf(b, m_gamma[1]);
  333. auto cgb = powf(c, m_gamma[2]);
  334. auto x = m_matrix[0] * agr + m_matrix[3] * bgg + m_matrix[6] * cgb;
  335. auto y = m_matrix[1] * agr + m_matrix[4] * bgg + m_matrix[7] * cgb;
  336. auto z = m_matrix[2] * agr + m_matrix[5] * bgg + m_matrix[8] * cgb;
  337. auto flattened_xyz = flatten_and_normalize_whitepoint(m_whitepoint, { x, y, z });
  338. auto scaled_black_point_xyz = scale_black_point(m_blackpoint, flattened_xyz);
  339. auto d65_normalized = convert_to_d65(scaled_black_point_xyz);
  340. auto srgb = convert_to_srgb(d65_normalized);
  341. auto red = static_cast<u8>(clamp(srgb[0], 0.0f, 1.0f) * 255.0f);
  342. auto green = static_cast<u8>(clamp(srgb[1], 0.0f, 1.0f) * 255.0f);
  343. auto blue = static_cast<u8>(clamp(srgb[2], 0.0f, 1.0f) * 255.0f);
  344. return Color(red, green, blue);
  345. }
  346. Vector<float> CalRGBColorSpace::default_decode() const
  347. {
  348. return { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f };
  349. }
  350. PDFErrorOr<NonnullRefPtr<ColorSpace>> ICCBasedColorSpace::create(Document* document, Vector<Value>&& parameters)
  351. {
  352. if (parameters.is_empty())
  353. return Error { Error::Type::MalformedPDF, "ICCBased color space expected one parameter" };
  354. auto param = TRY(document->resolve(parameters[0]));
  355. if (!param.has<NonnullRefPtr<Object>>() || !param.get<NonnullRefPtr<Object>>()->is<StreamObject>())
  356. return Error { Error::Type::MalformedPDF, "ICCBased color space expects a stream parameter" };
  357. auto stream = param.get<NonnullRefPtr<Object>>()->cast<StreamObject>();
  358. auto dict = stream->dict();
  359. auto maybe_profile = Gfx::ICC::Profile::try_load_from_externally_owned_memory(stream->bytes());
  360. if (!maybe_profile.is_error())
  361. return adopt_ref(*new ICCBasedColorSpace(maybe_profile.release_value()));
  362. if (dict->contains(CommonNames::Alternate)) {
  363. auto alternate_color_space_object = MUST(dict->get_object(document, CommonNames::Alternate));
  364. if (alternate_color_space_object->is<NameObject>())
  365. return ColorSpace::create(alternate_color_space_object->cast<NameObject>()->name());
  366. return Error { Error::Type::Internal, "Alternate color spaces in array format are not supported" };
  367. }
  368. return Error { Error::Type::MalformedPDF, "Failed to load ICC color space with malformed profile and no alternate" };
  369. }
  370. ICCBasedColorSpace::ICCBasedColorSpace(NonnullRefPtr<Gfx::ICC::Profile> profile)
  371. : m_profile(profile)
  372. {
  373. }
  374. PDFErrorOr<Color> ICCBasedColorSpace::color(ReadonlySpan<Value> arguments) const
  375. {
  376. if (!s_srgb_profile)
  377. s_srgb_profile = TRY(Gfx::ICC::sRGB());
  378. Vector<u8> bytes;
  379. for (auto const& arg : arguments) {
  380. VERIFY(arg.has_number());
  381. bytes.append(static_cast<u8>(arg.to_float() * 255.0f));
  382. }
  383. auto pcs = TRY(m_profile->to_pcs(bytes));
  384. Array<u8, 3> output;
  385. TRY(s_srgb_profile->from_pcs(pcs, output.span()));
  386. return Color(output[0], output[1], output[2]);
  387. }
  388. int ICCBasedColorSpace::number_of_components() const
  389. {
  390. return Gfx::ICC::number_of_components_in_color_space(m_profile->data_color_space());
  391. }
  392. Vector<float> ICCBasedColorSpace::default_decode() const
  393. {
  394. auto color_space = m_profile->data_color_space();
  395. switch (color_space) {
  396. case Gfx::ICC::ColorSpace::Gray:
  397. return { 0.0, 1.0 };
  398. case Gfx::ICC::ColorSpace::RGB:
  399. return { 0.0, 1.0, 0.0, 1.0, 0.0, 1.0 };
  400. case Gfx::ICC::ColorSpace::CMYK:
  401. return { 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0 };
  402. default:
  403. warnln("PDF: Unknown default_decode params for color space {}", Gfx::ICC::data_color_space_name(color_space));
  404. Vector<float> decoding_ranges;
  405. for (u8 i = 0; i < Gfx::ICC::number_of_components_in_color_space(color_space); i++) {
  406. decoding_ranges.append(0.0);
  407. decoding_ranges.append(1.0);
  408. }
  409. return decoding_ranges;
  410. }
  411. }
  412. PDFErrorOr<NonnullRefPtr<LabColorSpace>> LabColorSpace::create(Document*, Vector<Value>&& parameters)
  413. {
  414. if (parameters.size() != 1)
  415. return Error { Error::Type::MalformedPDF, "Lab color space expects one parameter" };
  416. auto color_space = adopt_ref(*new LabColorSpace());
  417. // FIXME: Implement.
  418. return color_space;
  419. }
  420. PDFErrorOr<Color> LabColorSpace::color(ReadonlySpan<Value>) const
  421. {
  422. return Error::rendering_unsupported_error("Lab color spaces not yet implemented");
  423. }
  424. Vector<float> LabColorSpace::default_decode() const
  425. {
  426. return { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f };
  427. }
  428. PDFErrorOr<NonnullRefPtr<SeparationColorSpace>> SeparationColorSpace::create(Document*, Vector<Value>&&)
  429. {
  430. auto color_space = adopt_ref(*new SeparationColorSpace());
  431. // FIXME: Implement.
  432. return color_space;
  433. }
  434. PDFErrorOr<Color> SeparationColorSpace::color(ReadonlySpan<Value>) const
  435. {
  436. return Error::rendering_unsupported_error("Separation color spaces not yet implemented");
  437. }
  438. Vector<float> SeparationColorSpace::default_decode() const
  439. {
  440. return { 0.0f, 1.0f };
  441. }
  442. }