ColorSpace.cpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762
  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(Document* document, NonnullRefPtr<Object> color_space_object)
  29. {
  30. // "A color space is defined by an array object whose first element is a name object identifying the color space family.
  31. // The remaining array elements, if any, are parameters that further characterize the color space;
  32. // their number and types vary according to the particular family.
  33. // For families that do not require parameters, the color space can be specified simply by the family name itself instead of an array."
  34. if (color_space_object->is<NameObject>())
  35. return ColorSpace::create(color_space_object->cast<NameObject>()->name());
  36. if (color_space_object->is<ArrayObject>())
  37. return ColorSpace::create(document, color_space_object->cast<ArrayObject>());
  38. return Error { Error::Type::MalformedPDF, "Color space must be name or array" };
  39. }
  40. PDFErrorOr<NonnullRefPtr<ColorSpace>> ColorSpace::create(DeprecatedFlyString const& name)
  41. {
  42. // Simple color spaces with no parameters, which can be specified directly
  43. if (name == CommonNames::DeviceGray)
  44. return DeviceGrayColorSpace::the();
  45. if (name == CommonNames::DeviceRGB)
  46. return DeviceRGBColorSpace::the();
  47. if (name == CommonNames::DeviceCMYK)
  48. return DeviceCMYKColorSpace::the();
  49. if (name == CommonNames::Pattern)
  50. return Error::rendering_unsupported_error("Pattern color spaces not yet implemented");
  51. VERIFY_NOT_REACHED();
  52. }
  53. PDFErrorOr<NonnullRefPtr<ColorSpace>> ColorSpace::create(Document* document, NonnullRefPtr<ArrayObject> color_space_array)
  54. {
  55. auto color_space_name = TRY(color_space_array->get_name_at(document, 0))->name();
  56. Vector<Value> parameters;
  57. parameters.ensure_capacity(color_space_array->size() - 1);
  58. for (size_t i = 1; i < color_space_array->size(); i++)
  59. parameters.unchecked_append(color_space_array->at(i));
  60. if (color_space_name == CommonNames::CalGray)
  61. return TRY(CalGrayColorSpace::create(document, move(parameters)));
  62. if (color_space_name == CommonNames::CalRGB)
  63. return TRY(CalRGBColorSpace::create(document, move(parameters)));
  64. if (color_space_name == CommonNames::DeviceN)
  65. return TRY(DeviceNColorSpace::create(document, move(parameters)));
  66. if (color_space_name == CommonNames::ICCBased)
  67. return TRY(ICCBasedColorSpace::create(document, move(parameters)));
  68. if (color_space_name == CommonNames::Indexed)
  69. return TRY(IndexedColorSpace::create(document, move(parameters)));
  70. if (color_space_name == CommonNames::Lab)
  71. return TRY(LabColorSpace::create(document, move(parameters)));
  72. if (color_space_name == CommonNames::Pattern)
  73. return Error::rendering_unsupported_error("Pattern color spaces not yet implemented");
  74. if (color_space_name == CommonNames::Separation)
  75. return TRY(SeparationColorSpace::create(document, move(parameters)));
  76. dbgln("Unknown color space: {}", color_space_name);
  77. return Error::rendering_unsupported_error("unknown color space");
  78. }
  79. NonnullRefPtr<DeviceGrayColorSpace> DeviceGrayColorSpace::the()
  80. {
  81. static auto instance = adopt_ref(*new DeviceGrayColorSpace());
  82. return instance;
  83. }
  84. PDFErrorOr<Color> DeviceGrayColorSpace::color(ReadonlySpan<Value> arguments) const
  85. {
  86. VERIFY(arguments.size() == 1);
  87. auto gray = static_cast<u8>(arguments[0].to_float() * 255.0f);
  88. return Color(gray, gray, gray);
  89. }
  90. Vector<float> DeviceGrayColorSpace::default_decode() const
  91. {
  92. return { 0.0f, 1.0f };
  93. }
  94. NonnullRefPtr<DeviceRGBColorSpace> DeviceRGBColorSpace::the()
  95. {
  96. static auto instance = adopt_ref(*new DeviceRGBColorSpace());
  97. return instance;
  98. }
  99. PDFErrorOr<Color> DeviceRGBColorSpace::color(ReadonlySpan<Value> arguments) const
  100. {
  101. VERIFY(arguments.size() == 3);
  102. auto r = static_cast<u8>(arguments[0].to_float() * 255.0f);
  103. auto g = static_cast<u8>(arguments[1].to_float() * 255.0f);
  104. auto b = static_cast<u8>(arguments[2].to_float() * 255.0f);
  105. return Color(r, g, b);
  106. }
  107. Vector<float> DeviceRGBColorSpace::default_decode() const
  108. {
  109. return { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f };
  110. }
  111. NonnullRefPtr<DeviceCMYKColorSpace> DeviceCMYKColorSpace::the()
  112. {
  113. static auto instance = adopt_ref(*new DeviceCMYKColorSpace());
  114. return instance;
  115. }
  116. PDFErrorOr<Color> DeviceCMYKColorSpace::color(ReadonlySpan<Value> arguments) const
  117. {
  118. VERIFY(arguments.size() == 4);
  119. auto c = arguments[0].to_float();
  120. auto m = arguments[1].to_float();
  121. auto y = arguments[2].to_float();
  122. auto k = arguments[3].to_float();
  123. return Color::from_cmyk(c, m, y, k);
  124. }
  125. Vector<float> DeviceCMYKColorSpace::default_decode() const
  126. {
  127. return { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f };
  128. }
  129. PDFErrorOr<NonnullRefPtr<DeviceNColorSpace>> DeviceNColorSpace::create(Document*, Vector<Value>&& parameters)
  130. {
  131. // "[ /DeviceN names alternateSpace tintTransform ]
  132. // or
  133. // [ /DeviceN names alternateSpace tintTransform attributes ]"
  134. // (`/DeviceN` is already stripped from the array by the time we get here.)
  135. if (parameters.size() != 3 && parameters.size() != 4)
  136. return Error { Error::Type::MalformedPDF, "DeviceN color space expects 4 or 5 parameters" };
  137. // "The names parameter is an array of name objects specifying the individual color components.
  138. // The length of the array determines the number of components in the DeviceN color space"
  139. auto names = parameters[0].get<NonnullRefPtr<Object>>()->cast<ArrayObject>();
  140. // "The alternateSpace parameter is an array or name object that can be any device or CIE-based color space
  141. // but not another special color space (Pattern, Indexed, Separation, or DeviceN)."
  142. // FIXME: Implement.
  143. return adopt_ref(*new DeviceNColorSpace(names->size()));
  144. }
  145. PDFErrorOr<Color> DeviceNColorSpace::color(ReadonlySpan<Value>) const
  146. {
  147. return Error::rendering_unsupported_error("DeviceN color spaces not yet implemented");
  148. }
  149. int DeviceNColorSpace::number_of_components() const
  150. {
  151. return m_number_of_components;
  152. }
  153. Vector<float> DeviceNColorSpace::default_decode() const
  154. {
  155. Vector<float> decoding_ranges;
  156. for (u8 i = 0; i < number_of_components(); i++) {
  157. decoding_ranges.append(0.0);
  158. decoding_ranges.append(1.0);
  159. }
  160. return decoding_ranges;
  161. }
  162. DeviceNColorSpace::DeviceNColorSpace(size_t number_of_components)
  163. : m_number_of_components(number_of_components)
  164. {
  165. }
  166. constexpr Array<float, 3> matrix_multiply(Array<float, 9> a, Array<float, 3> b)
  167. {
  168. return Array<float, 3> {
  169. a[0] * b[0] + a[1] * b[1] + a[2] * b[2],
  170. a[3] * b[0] + a[4] * b[1] + a[5] * b[2],
  171. a[6] * b[0] + a[7] * b[1] + a[8] * b[2]
  172. };
  173. }
  174. // Converts to a flat XYZ space with white point = (1, 1, 1)
  175. // Step 2 of https://www.color.org/adobebpc.pdf
  176. constexpr Array<float, 3> flatten_and_normalize_whitepoint(Array<float, 3> whitepoint, Array<float, 3> xyz)
  177. {
  178. VERIFY(whitepoint[1] == 1.0f);
  179. return {
  180. (1.0f / whitepoint[0]) * xyz[0],
  181. xyz[1],
  182. (1.0f / whitepoint[2]) * xyz[2],
  183. };
  184. }
  185. constexpr float decode_l(float input)
  186. {
  187. constexpr float decode_l_scaling_constant = 0.00110705646f; // (((8 + 16) / 116) ^ 3) / 8
  188. if (input < 0.0f)
  189. return -decode_l(-input);
  190. if (input >= 0.0f && input <= 8.0f)
  191. return input * decode_l_scaling_constant;
  192. return powf(((input + 16.0f) / 116.0f), 3.0f);
  193. }
  194. constexpr Array<float, 3> scale_black_point(Array<float, 3> blackpoint, Array<float, 3> xyz)
  195. {
  196. auto y_dst = decode_l(0); // DestinationBlackPoint is just [0, 0, 0]
  197. auto y_src = decode_l(blackpoint[0]);
  198. auto scale = (1 - y_dst) / (1 - y_src);
  199. auto offset = 1 - scale;
  200. return {
  201. xyz[0] * scale + offset,
  202. xyz[1] * scale + offset,
  203. xyz[2] * scale + offset,
  204. };
  205. }
  206. // https://en.wikipedia.org/wiki/Illuminant_D65
  207. constexpr Array<float, 3> convert_to_d65(Array<float, 3> xyz)
  208. {
  209. constexpr float d65x = 0.95047f;
  210. constexpr float d65y = 1.0f;
  211. constexpr float d65z = 1.08883f;
  212. return { xyz[0] * d65x, xyz[1] * d65y, xyz[2] * d65z };
  213. }
  214. // https://en.wikipedia.org/wiki/SRGB
  215. constexpr Array<float, 3> convert_to_srgb(Array<float, 3> xyz)
  216. {
  217. // See the sRGB D65 [M]^-1 matrix in the following page
  218. // http://www.brucelindbloom.com/index.html?Eqn_RGB_XYZ_Matrix.html
  219. constexpr Array<float, 9> conversion_matrix = {
  220. 3.2404542,
  221. -1.5371385,
  222. -0.4985314,
  223. -0.969266,
  224. 1.8760108,
  225. 0.0415560,
  226. 0.0556434,
  227. -0.2040259,
  228. 1.0572252,
  229. };
  230. auto linear_srgb = matrix_multiply(conversion_matrix, xyz);
  231. // FIXME: Use the real sRGB curve by replacing this function with Gfx::ICC::sRGB().from_pcs().
  232. 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) };
  233. }
  234. PDFErrorOr<NonnullRefPtr<CalGrayColorSpace>> CalGrayColorSpace::create(Document* document, Vector<Value>&& parameters)
  235. {
  236. if (parameters.size() != 1)
  237. return Error { Error::Type::MalformedPDF, "Gray color space expects one parameter" };
  238. auto param = parameters[0];
  239. if (!param.has<NonnullRefPtr<Object>>() || !param.get<NonnullRefPtr<Object>>()->is<DictObject>())
  240. return Error { Error::Type::MalformedPDF, "Gray color space expects a dict parameter" };
  241. auto dict = param.get<NonnullRefPtr<Object>>()->cast<DictObject>();
  242. if (!dict->contains(CommonNames::WhitePoint))
  243. return Error { Error::Type::MalformedPDF, "Gray color space expects a Whitepoint key" };
  244. auto white_point_array = TRY(dict->get_array(document, CommonNames::WhitePoint));
  245. if (white_point_array->size() != 3)
  246. return Error { Error::Type::MalformedPDF, "Gray color space expects 3 Whitepoint parameters" };
  247. auto color_space = adopt_ref(*new CalGrayColorSpace());
  248. color_space->m_whitepoint[0] = white_point_array->at(0).to_float();
  249. color_space->m_whitepoint[1] = white_point_array->at(1).to_float();
  250. color_space->m_whitepoint[2] = white_point_array->at(2).to_float();
  251. if (color_space->m_whitepoint[1] != 1.0f)
  252. return Error { Error::Type::MalformedPDF, "Gray color space expects 2nd Whitepoint to be 1.0" };
  253. if (dict->contains(CommonNames::BlackPoint)) {
  254. auto black_point_array = TRY(dict->get_array(document, CommonNames::BlackPoint));
  255. if (black_point_array->size() == 3) {
  256. color_space->m_blackpoint[0] = black_point_array->at(0).to_float();
  257. color_space->m_blackpoint[1] = black_point_array->at(1).to_float();
  258. color_space->m_blackpoint[2] = black_point_array->at(2).to_float();
  259. }
  260. }
  261. if (dict->contains(CommonNames::Gamma)) {
  262. color_space->m_gamma = TRY(document->resolve(dict->get_value(CommonNames::Gamma))).to_float();
  263. }
  264. return color_space;
  265. }
  266. PDFErrorOr<Color> CalGrayColorSpace::color(ReadonlySpan<Value> arguments) const
  267. {
  268. VERIFY(arguments.size() == 1);
  269. auto a = clamp(arguments[0].to_float(), 0.0f, 1.0f);
  270. auto ag = powf(a, m_gamma);
  271. auto x = m_whitepoint[0] * ag;
  272. auto y = m_whitepoint[1] * ag;
  273. auto z = m_whitepoint[2] * ag;
  274. auto flattened_xyz = flatten_and_normalize_whitepoint(m_whitepoint, { x, y, z });
  275. auto scaled_black_point_xyz = scale_black_point(m_blackpoint, flattened_xyz);
  276. auto d65_normalized = convert_to_d65(scaled_black_point_xyz);
  277. auto srgb = convert_to_srgb(d65_normalized);
  278. auto red = static_cast<u8>(clamp(srgb[0], 0.0f, 1.0f) * 255.0f);
  279. auto green = static_cast<u8>(clamp(srgb[1], 0.0f, 1.0f) * 255.0f);
  280. auto blue = static_cast<u8>(clamp(srgb[2], 0.0f, 1.0f) * 255.0f);
  281. return Color(red, green, blue);
  282. }
  283. Vector<float> CalGrayColorSpace::default_decode() const
  284. {
  285. return { 0.0f, 1.0f };
  286. }
  287. PDFErrorOr<NonnullRefPtr<CalRGBColorSpace>> CalRGBColorSpace::create(Document* document, Vector<Value>&& parameters)
  288. {
  289. if (parameters.size() != 1)
  290. return Error { Error::Type::MalformedPDF, "RGB color space expects one parameter" };
  291. auto param = parameters[0];
  292. if (!param.has<NonnullRefPtr<Object>>() || !param.get<NonnullRefPtr<Object>>()->is<DictObject>())
  293. return Error { Error::Type::MalformedPDF, "RGB color space expects a dict parameter" };
  294. auto dict = param.get<NonnullRefPtr<Object>>()->cast<DictObject>();
  295. if (!dict->contains(CommonNames::WhitePoint))
  296. return Error { Error::Type::MalformedPDF, "RGB color space expects a Whitepoint key" };
  297. auto white_point_array = TRY(dict->get_array(document, CommonNames::WhitePoint));
  298. if (white_point_array->size() != 3)
  299. return Error { Error::Type::MalformedPDF, "RGB color space expects 3 Whitepoint parameters" };
  300. auto color_space = adopt_ref(*new CalRGBColorSpace());
  301. color_space->m_whitepoint[0] = white_point_array->at(0).to_float();
  302. color_space->m_whitepoint[1] = white_point_array->at(1).to_float();
  303. color_space->m_whitepoint[2] = white_point_array->at(2).to_float();
  304. if (color_space->m_whitepoint[1] != 1.0f)
  305. return Error { Error::Type::MalformedPDF, "RGB color space expects 2nd Whitepoint to be 1.0" };
  306. if (dict->contains(CommonNames::BlackPoint)) {
  307. auto black_point_array = TRY(dict->get_array(document, CommonNames::BlackPoint));
  308. if (black_point_array->size() == 3) {
  309. color_space->m_blackpoint[0] = black_point_array->at(0).to_float();
  310. color_space->m_blackpoint[1] = black_point_array->at(1).to_float();
  311. color_space->m_blackpoint[2] = black_point_array->at(2).to_float();
  312. }
  313. }
  314. if (dict->contains(CommonNames::Gamma)) {
  315. auto gamma_array = TRY(dict->get_array(document, CommonNames::Gamma));
  316. if (gamma_array->size() == 3) {
  317. color_space->m_gamma[0] = gamma_array->at(0).to_float();
  318. color_space->m_gamma[1] = gamma_array->at(1).to_float();
  319. color_space->m_gamma[2] = gamma_array->at(2).to_float();
  320. }
  321. }
  322. if (dict->contains(CommonNames::Matrix)) {
  323. auto matrix_array = TRY(dict->get_array(document, CommonNames::Matrix));
  324. if (matrix_array->size() == 9) {
  325. color_space->m_matrix[0] = matrix_array->at(0).to_float();
  326. color_space->m_matrix[1] = matrix_array->at(1).to_float();
  327. color_space->m_matrix[2] = matrix_array->at(2).to_float();
  328. color_space->m_matrix[3] = matrix_array->at(3).to_float();
  329. color_space->m_matrix[4] = matrix_array->at(4).to_float();
  330. color_space->m_matrix[5] = matrix_array->at(5).to_float();
  331. color_space->m_matrix[6] = matrix_array->at(6).to_float();
  332. color_space->m_matrix[7] = matrix_array->at(7).to_float();
  333. color_space->m_matrix[8] = matrix_array->at(8).to_float();
  334. }
  335. }
  336. return color_space;
  337. }
  338. PDFErrorOr<Color> CalRGBColorSpace::color(ReadonlySpan<Value> arguments) const
  339. {
  340. VERIFY(arguments.size() == 3);
  341. auto a = clamp(arguments[0].to_float(), 0.0f, 1.0f);
  342. auto b = clamp(arguments[1].to_float(), 0.0f, 1.0f);
  343. auto c = clamp(arguments[2].to_float(), 0.0f, 1.0f);
  344. auto agr = powf(a, m_gamma[0]);
  345. auto bgg = powf(b, m_gamma[1]);
  346. auto cgb = powf(c, m_gamma[2]);
  347. auto x = m_matrix[0] * agr + m_matrix[3] * bgg + m_matrix[6] * cgb;
  348. auto y = m_matrix[1] * agr + m_matrix[4] * bgg + m_matrix[7] * cgb;
  349. auto z = m_matrix[2] * agr + m_matrix[5] * bgg + m_matrix[8] * cgb;
  350. auto flattened_xyz = flatten_and_normalize_whitepoint(m_whitepoint, { x, y, z });
  351. auto scaled_black_point_xyz = scale_black_point(m_blackpoint, flattened_xyz);
  352. auto d65_normalized = convert_to_d65(scaled_black_point_xyz);
  353. auto srgb = convert_to_srgb(d65_normalized);
  354. auto red = static_cast<u8>(clamp(srgb[0], 0.0f, 1.0f) * 255.0f);
  355. auto green = static_cast<u8>(clamp(srgb[1], 0.0f, 1.0f) * 255.0f);
  356. auto blue = static_cast<u8>(clamp(srgb[2], 0.0f, 1.0f) * 255.0f);
  357. return Color(red, green, blue);
  358. }
  359. Vector<float> CalRGBColorSpace::default_decode() const
  360. {
  361. return { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f };
  362. }
  363. PDFErrorOr<NonnullRefPtr<ColorSpace>> ICCBasedColorSpace::create(Document* document, Vector<Value>&& parameters)
  364. {
  365. if (parameters.is_empty())
  366. return Error { Error::Type::MalformedPDF, "ICCBased color space expected one parameter" };
  367. auto param = TRY(document->resolve(parameters[0]));
  368. if (!param.has<NonnullRefPtr<Object>>() || !param.get<NonnullRefPtr<Object>>()->is<StreamObject>())
  369. return Error { Error::Type::MalformedPDF, "ICCBased color space expects a stream parameter" };
  370. auto stream = param.get<NonnullRefPtr<Object>>()->cast<StreamObject>();
  371. auto dict = stream->dict();
  372. auto maybe_profile = Gfx::ICC::Profile::try_load_from_externally_owned_memory(stream->bytes());
  373. if (!maybe_profile.is_error())
  374. return adopt_ref(*new ICCBasedColorSpace(maybe_profile.release_value()));
  375. if (dict->contains(CommonNames::Alternate)) {
  376. auto alternate_color_space_object = MUST(dict->get_object(document, CommonNames::Alternate));
  377. if (alternate_color_space_object->is<NameObject>())
  378. return ColorSpace::create(alternate_color_space_object->cast<NameObject>()->name());
  379. return Error { Error::Type::Internal, "Alternate color spaces in array format are not supported" };
  380. }
  381. return Error { Error::Type::MalformedPDF, "Failed to load ICC color space with malformed profile and no alternate" };
  382. }
  383. ICCBasedColorSpace::ICCBasedColorSpace(NonnullRefPtr<Gfx::ICC::Profile> profile)
  384. : m_profile(profile)
  385. {
  386. }
  387. PDFErrorOr<Color> ICCBasedColorSpace::color(ReadonlySpan<Value> arguments) const
  388. {
  389. if (!s_srgb_profile)
  390. s_srgb_profile = TRY(Gfx::ICC::sRGB());
  391. Vector<u8> bytes;
  392. for (auto const& arg : arguments) {
  393. VERIFY(arg.has_number());
  394. bytes.append(static_cast<u8>(arg.to_float() * 255.0f));
  395. }
  396. auto pcs = TRY(m_profile->to_pcs(bytes));
  397. Array<u8, 3> output;
  398. TRY(s_srgb_profile->from_pcs(pcs, output.span()));
  399. return Color(output[0], output[1], output[2]);
  400. }
  401. int ICCBasedColorSpace::number_of_components() const
  402. {
  403. return Gfx::ICC::number_of_components_in_color_space(m_profile->data_color_space());
  404. }
  405. Vector<float> ICCBasedColorSpace::default_decode() const
  406. {
  407. auto color_space = m_profile->data_color_space();
  408. switch (color_space) {
  409. case Gfx::ICC::ColorSpace::Gray:
  410. return { 0.0, 1.0 };
  411. case Gfx::ICC::ColorSpace::RGB:
  412. return { 0.0, 1.0, 0.0, 1.0, 0.0, 1.0 };
  413. case Gfx::ICC::ColorSpace::CMYK:
  414. return { 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0 };
  415. default:
  416. warnln("PDF: Unknown default_decode params for color space {}", Gfx::ICC::data_color_space_name(color_space));
  417. Vector<float> decoding_ranges;
  418. for (u8 i = 0; i < Gfx::ICC::number_of_components_in_color_space(color_space); i++) {
  419. decoding_ranges.append(0.0);
  420. decoding_ranges.append(1.0);
  421. }
  422. return decoding_ranges;
  423. }
  424. }
  425. PDFErrorOr<NonnullRefPtr<LabColorSpace>> LabColorSpace::create(Document* document, Vector<Value>&& parameters)
  426. {
  427. if (parameters.size() != 1)
  428. return Error { Error::Type::MalformedPDF, "Lab color space expects one parameter" };
  429. auto param = parameters[0];
  430. if (!param.has<NonnullRefPtr<Object>>() || !param.get<NonnullRefPtr<Object>>()->is<DictObject>())
  431. return Error { Error::Type::MalformedPDF, "Lab color space expects a dict parameter" };
  432. auto dict = param.get<NonnullRefPtr<Object>>()->cast<DictObject>();
  433. if (!dict->contains(CommonNames::WhitePoint))
  434. return Error { Error::Type::MalformedPDF, "Lab color space expects a Whitepoint key" };
  435. auto white_point_array = TRY(dict->get_array(document, CommonNames::WhitePoint));
  436. if (white_point_array->size() != 3)
  437. return Error { Error::Type::MalformedPDF, "Lab color space expects 3 Whitepoint parameters" };
  438. auto color_space = adopt_ref(*new LabColorSpace());
  439. color_space->m_whitepoint[0] = white_point_array->at(0).to_float();
  440. color_space->m_whitepoint[1] = white_point_array->at(1).to_float();
  441. color_space->m_whitepoint[2] = white_point_array->at(2).to_float();
  442. if (color_space->m_whitepoint[1] != 1.0f)
  443. return Error { Error::Type::MalformedPDF, "Lab color space expects 2nd Whitepoint to be 1.0" };
  444. if (dict->contains(CommonNames::BlackPoint)) {
  445. auto black_point_array = TRY(dict->get_array(document, CommonNames::BlackPoint));
  446. if (black_point_array->size() == 3) {
  447. color_space->m_blackpoint[0] = black_point_array->at(0).to_float();
  448. color_space->m_blackpoint[1] = black_point_array->at(1).to_float();
  449. color_space->m_blackpoint[2] = black_point_array->at(2).to_float();
  450. }
  451. }
  452. if (dict->contains(CommonNames::Range)) {
  453. auto range_array = TRY(dict->get_array(document, CommonNames::Range));
  454. if (range_array->size() == 4) {
  455. color_space->m_range[0] = range_array->at(0).to_float();
  456. color_space->m_range[1] = range_array->at(1).to_float();
  457. color_space->m_range[2] = range_array->at(2).to_float();
  458. color_space->m_range[3] = range_array->at(3).to_float();
  459. }
  460. }
  461. return color_space;
  462. }
  463. PDFErrorOr<Color> LabColorSpace::color(ReadonlySpan<Value> arguments) const
  464. {
  465. VERIFY(arguments.size() == 3);
  466. auto L_star = clamp(arguments[0].to_float(), 0.0f, 100.0f);
  467. auto a_star = clamp(arguments[1].to_float(), m_range[0], m_range[1]);
  468. auto b_star = clamp(arguments[2].to_float(), m_range[2], m_range[3]);
  469. auto L = (L_star + 16) / 116 + a_star / 500;
  470. auto M = (L_star + 16) / 116;
  471. auto N = (L_star + 16) / 116 - b_star / 200;
  472. auto g = [](float x) {
  473. if (x >= 6.0f / 29.0f)
  474. return powf(x, 3);
  475. return 108.0f / 841.0f * (x - 4.0f / 29.0f);
  476. };
  477. auto x = m_whitepoint[0] * g(L);
  478. auto y = m_whitepoint[1] * g(M);
  479. auto z = m_whitepoint[2] * g(N);
  480. auto flattened_xyz = flatten_and_normalize_whitepoint(m_whitepoint, { x, y, z });
  481. auto scaled_black_point_xyz = scale_black_point(m_blackpoint, flattened_xyz);
  482. auto d65_normalized = convert_to_d65(scaled_black_point_xyz);
  483. auto srgb = convert_to_srgb(d65_normalized);
  484. auto red = static_cast<u8>(clamp(srgb[0], 0.0f, 1.0f) * 255.0f);
  485. auto green = static_cast<u8>(clamp(srgb[1], 0.0f, 1.0f) * 255.0f);
  486. auto blue = static_cast<u8>(clamp(srgb[2], 0.0f, 1.0f) * 255.0f);
  487. return Color(red, green, blue);
  488. }
  489. Vector<float> LabColorSpace::default_decode() const
  490. {
  491. return { 0.0f, 100.0f, m_range[0], m_range[1], m_range[2], m_range[3] };
  492. }
  493. PDFErrorOr<NonnullRefPtr<ColorSpace>> IndexedColorSpace::create(Document* document, Vector<Value>&& parameters)
  494. {
  495. if (parameters.size() != 3)
  496. return Error { Error::Type::MalformedPDF, "Indexed color space expected three parameters" };
  497. // "The base parameter is an array or name that identifies the base color space in which the values
  498. // in the color table are to be interpreted. It can be any device or CIE-based color space or (in PDF 1.3)
  499. // a Separation or DeviceN space, but not a Pattern space or another Indexed space."
  500. auto param0 = TRY(document->resolve(parameters[0]));
  501. if (!param0.has<NonnullRefPtr<Object>>())
  502. return Error { Error::Type::MalformedPDF, "Indexed color space expects object for first arg" };
  503. auto base_object = param0.get<NonnullRefPtr<Object>>();
  504. auto base = TRY(ColorSpace::create(document, base_object));
  505. if (base->family() == ColorSpaceFamily::Pattern || base->family() == ColorSpaceFamily::Indexed)
  506. return Error { Error::Type::MalformedPDF, "Indexed color space has invalid base color space" };
  507. // "The hival parameter is an integer that specifies the maximum valid index value. In other words,
  508. // the color table is to be indexed by integers in the range 0 to hival. hival can be no greater than 255"
  509. auto param1 = TRY(document->resolve(parameters[1]));
  510. if (!param1.has<int>())
  511. return Error { Error::Type::MalformedPDF, "Indexed color space expects int for second arg" };
  512. auto hival = param1.get<int>();
  513. // "The color table is defined by the lookup parameter, which can be either a stream or (in PDF 1.2) a byte string.
  514. // It provides the mapping between index values and the corresponding colors in the base color space.
  515. // The color table data must be m × (hival + 1) bytes long, where m is the number of color components in the
  516. // base color space. Each byte is an unsigned integer in the range 0 to 255 that is scaled to the range of
  517. // the corresponding color component in the base color space; that is, 0 corresponds to the minimum value
  518. // in the range for that component, and 255 corresponds to the maximum."
  519. auto param2 = TRY(document->resolve(parameters[2]));
  520. if (!param2.has<NonnullRefPtr<Object>>())
  521. return Error { Error::Type::MalformedPDF, "Indexed color space expects object for third arg" };
  522. auto lookup_object = param2.get<NonnullRefPtr<Object>>();
  523. Vector<u8> lookup;
  524. if (lookup_object->is<StreamObject>()) {
  525. lookup = Vector<u8> { lookup_object->cast<StreamObject>()->bytes() };
  526. } else if (lookup_object->is<StringObject>()) {
  527. // FIXME: Check if it's a hex string.
  528. auto const& string = lookup_object->cast<StringObject>()->string();
  529. lookup = Vector<u8> { ReadonlyBytes { string.characters(), string.length() } };
  530. } else {
  531. return Error { Error::Type::MalformedPDF, "Indexed color space expects stream or string for third arg" };
  532. }
  533. if (static_cast<int>(lookup.size()) != (hival + 1) * base->number_of_components())
  534. return Error { Error::Type::MalformedPDF, "Indexed color space lookup table doesn't match size" };
  535. auto color_space = adopt_ref(*new IndexedColorSpace(move(base)));
  536. color_space->m_hival = hival;
  537. color_space->m_lookup = move(lookup);
  538. return color_space;
  539. }
  540. IndexedColorSpace::IndexedColorSpace(NonnullRefPtr<ColorSpace> base)
  541. : m_base(move(base))
  542. {
  543. }
  544. PDFErrorOr<Color> IndexedColorSpace::color(ReadonlySpan<Value> arguments) const
  545. {
  546. VERIFY(arguments.size() == 1);
  547. auto index = arguments[0].to_int();
  548. if (index < 0 || index > m_hival)
  549. return Error { Error::Type::MalformedPDF, "Indexed color space index out of range" };
  550. Vector<Value, 4> components;
  551. size_t const n = m_base->number_of_components();
  552. for (size_t i = 0; i < n; ++i)
  553. TRY(components.try_append(Value(m_lookup[index * n + i] / 255.0f)));
  554. return m_base->color(components);
  555. }
  556. Vector<float> IndexedColorSpace::default_decode() const
  557. {
  558. return { 0.0, static_cast<float>(m_hival) };
  559. }
  560. PDFErrorOr<NonnullRefPtr<SeparationColorSpace>> SeparationColorSpace::create(Document* document, Vector<Value>&& parameters)
  561. {
  562. if (parameters.size() != 3)
  563. return Error { Error::Type::MalformedPDF, "Separation color space expected three parameters" };
  564. // "The name parameter is a name object specifying the name of the colorant that this Separation color space
  565. // is intended to represent (or one of the special names All or None; see below)"
  566. auto param0 = TRY(document->resolve(parameters[0]));
  567. if (!param0.has<NonnullRefPtr<Object>>())
  568. return Error { Error::Type::MalformedPDF, "Separation color space expects object for first arg" };
  569. auto name_object = param0.get<NonnullRefPtr<Object>>();
  570. if (!name_object->is<NameObject>())
  571. return Error { Error::Type::MalformedPDF, "Separation color space expects name object for first arg" };
  572. auto name = name_object->cast<NameObject>()->name();
  573. // "The alternateSpace parameter must be an array or name object that identifies the alternate color space,
  574. // which can be any device or CIE-based color space but not another special color space
  575. // (Pattern, Indexed, Separation, or DeviceN)."
  576. auto param1 = TRY(document->resolve(parameters[1]));
  577. if (!param1.has<NonnullRefPtr<Object>>())
  578. return Error { Error::Type::MalformedPDF, "Separation color space expects object for second arg" };
  579. auto alternate_space_object = param1.get<NonnullRefPtr<Object>>();
  580. auto alternate_space = TRY(ColorSpace::create(document, alternate_space_object));
  581. auto family = alternate_space->family();
  582. if (family == ColorSpaceFamily::Pattern || family == ColorSpaceFamily::Indexed || family == ColorSpaceFamily::Separation || family == ColorSpaceFamily::DeviceN)
  583. return Error { Error::Type::MalformedPDF, "Separation color space has invalid alternate color space" };
  584. // "The tintTransform parameter must be a function"
  585. auto param2 = TRY(document->resolve(parameters[2]));
  586. if (!param2.has<NonnullRefPtr<Object>>())
  587. return Error { Error::Type::MalformedPDF, "Separation color space expects object for third arg" };
  588. auto tint_transform_object = param2.get<NonnullRefPtr<Object>>();
  589. auto tint_transform = TRY(Function::create(document, tint_transform_object));
  590. auto color_space = adopt_ref(*new SeparationColorSpace(move(alternate_space), move(tint_transform)));
  591. color_space->m_name = move(name);
  592. return color_space;
  593. }
  594. SeparationColorSpace::SeparationColorSpace(NonnullRefPtr<ColorSpace> alternate_space, NonnullRefPtr<Function> tint_transform)
  595. : m_alternate_space(move(alternate_space))
  596. , m_tint_transform(move(tint_transform))
  597. {
  598. }
  599. PDFErrorOr<Color> SeparationColorSpace::color(ReadonlySpan<Value> arguments) const
  600. {
  601. // "For an additive device such as a computer display, a Separation color space never applies a process colorant directly;
  602. // it always reverts to the alternate color space as described below."
  603. // "During subsequent painting operations, an application calls [the tint] function to transform a tint value into
  604. // color component values in the alternate color space."
  605. // FIXME: Does this need handling for the special colorant names "All" and "None"?
  606. // FIXME: When drawing to a printer, do something else.
  607. VERIFY(arguments.size() == 1);
  608. auto a = arguments[0].to_float();
  609. auto tint_output = TRY(m_tint_transform->evaluate(ReadonlySpan<float> { &a, 1 }));
  610. m_tint_output_values.resize(tint_output.size());
  611. for (size_t i = 0; i < tint_output.size(); ++i)
  612. m_tint_output_values[i] = tint_output[i];
  613. return m_alternate_space->color(m_tint_output_values);
  614. }
  615. Vector<float> SeparationColorSpace::default_decode() const
  616. {
  617. return { 0.0f, 1.0f };
  618. }
  619. }