use std::{
ffi::OsStr,
io::{BufRead, Seek},
};
use crate::{ColorSpace, DecodeSettings, Image};
use ::image::error::{DecodingError, ImageFormatHint};
use ::image::{ColorType, ExtendedColorType, ImageDecoder, ImageError, ImageResult};
use image::hooks::{decoding_hook_registered, register_format_detection_hook};
use moxcms::{CmsError, ColorProfile, Layout, TransformOptions};
const CMYK_PROFILE: &[u8] = include_bytes!("../assets/CGATS001Compat-v2-micro.icc");
impl ImageDecoder for Image<'_> {
fn dimensions(&self) -> (u32, u32) {
(self.width(), self.height())
}
fn color_type(&self) -> ColorType {
let channel_count = self.color_space.num_channels();
let has_alpha = self.has_alpha;
match (channel_count, has_alpha) {
(1, false) => ColorType::L8,
(1, true) => ColorType::La8,
(3, false) => ColorType::Rgb8,
(3, true) => ColorType::Rgba8,
(4, false) => ColorType::Rgb8,
(4, true) => ColorType::Rgba8,
_ => ColorType::Rgb8,
}
}
fn original_color_type(&self) -> ExtendedColorType {
let channel_count = self.color_space.num_channels();
let has_alpha = self.has_alpha;
let depth = self.original_bit_depth();
match (channel_count, depth, has_alpha) {
(1, 1, false) => ExtendedColorType::L1,
(1, 1, true) => ExtendedColorType::La1,
(1, 2, false) => ExtendedColorType::L2,
(1, 2, true) => ExtendedColorType::La2,
(1, 4, false) => ExtendedColorType::L4,
(1, 4, true) => ExtendedColorType::La4,
(1, 8, false) => ExtendedColorType::L8,
(1, 8, true) => ExtendedColorType::La8,
(1, 16, false) => ExtendedColorType::L8,
(1, 16, true) => ExtendedColorType::La8,
(3, 1, false) => ExtendedColorType::Rgb1,
(3, 1, true) => ExtendedColorType::Rgba1,
(3, 2, false) => ExtendedColorType::Rgb2,
(3, 2, true) => ExtendedColorType::Rgba2,
(3, 4, false) => ExtendedColorType::Rgb4,
(3, 4, true) => ExtendedColorType::Rgba4,
(3, 8, false) => ExtendedColorType::Rgb8,
(3, 8, true) => ExtendedColorType::Rgba8,
(3, 16, false) => ExtendedColorType::Rgb8,
(3, 16, true) => ExtendedColorType::Rgba8,
(4, 8, false) => ExtendedColorType::Cmyk8,
(4, 16, false) => ExtendedColorType::Cmyk16,
_ => ExtendedColorType::Unknown(orig_bits_per_pixel(self)),
}
}
fn read_image(self, buf: &mut [u8]) -> ImageResult<()>
where
Self: Sized,
{
convert_inner(&self, buf)
}
fn read_image_boxed(self: Box<Self>, buf: &mut [u8]) -> ImageResult<()> {
convert_inner(&self, buf)
}
}
#[doc(hidden)]
pub struct Jp2Decoder {
input: Vec<u8>,
width: u32,
height: u32,
color_type: ColorType,
orig_color_type: ExtendedColorType,
}
impl Jp2Decoder {
pub fn new<R: BufRead + Seek>(r: R) -> ImageResult<Self> {
let mut input = Vec::new();
let mut r = r;
r.read_to_end(&mut input)?;
let headers = Image::new(&input, &DecodeSettings::default())?;
Ok(Self {
width: headers.width(),
height: headers.height(),
color_type: headers.color_type(),
orig_color_type: headers.original_color_type(),
input,
})
}
}
impl ImageDecoder for Jp2Decoder {
fn dimensions(&self) -> (u32, u32) {
(self.width, self.height)
}
fn color_type(&self) -> ColorType {
self.color_type
}
fn original_color_type(&self) -> ExtendedColorType {
self.orig_color_type
}
fn read_image(self, buf: &mut [u8]) -> ImageResult<()>
where
Self: Sized,
{
let decoder = Image::new(&self.input, &DecodeSettings::default()).unwrap();
decoder.read_image(buf)
}
fn read_image_boxed(self: Box<Self>, buf: &mut [u8]) -> ImageResult<()> {
let decoder = Image::new(&self.input, &DecodeSettings::default()).unwrap();
decoder.read_image(buf)
}
}
fn orig_bits_per_pixel(img: &Image<'_>) -> u8 {
let mut channel_count = img.color_space().num_channels();
if img.has_alpha {
channel_count += 1;
}
channel_count * img.original_bit_depth()
}
fn convert_inner(image: &Image<'_>, buf: &mut [u8]) -> ImageResult<()> {
let width = image.width();
let height = image.height();
let color_space = image.color_space().clone();
let has_alpha = image.has_alpha();
fn from_icc(
icc: &[u8],
num_channels: u8,
has_alpha: bool,
width: u32,
height: u32,
input_data: &[u8],
) -> Result<Vec<u8>, CmsError> {
let src_profile = ColorProfile::new_from_slice(icc)?;
let dest_profile = ColorProfile::new_srgb();
let (src_layout, dest_layout, out_channels) = match (num_channels, has_alpha) {
(1, false) => (Layout::Gray, Layout::Gray, 1),
(1, true) => (Layout::GrayAlpha, Layout::GrayAlpha, 2),
(3, false) => (Layout::Rgb, Layout::Rgb, 3),
(3, true) => (Layout::Rgba, Layout::Rgba, 4),
(4, false) => (Layout::Rgba, Layout::Rgb, 3),
_ => {
return Err(CmsError::UnsupportedChannelConfiguration);
}
};
let transform = src_profile.create_transform_8bit(
src_layout,
&dest_profile,
dest_layout,
TransformOptions::default(),
)?;
let mut transformed = vec![0; (width * height * out_channels) as usize];
transform.transform(input_data, &mut transformed)?;
Ok(transformed)
}
fn process(
image: &Image<'_>,
buf: &mut [u8],
width: u32,
height: u32,
has_alpha: bool,
cs: ColorSpace,
) -> Result<(), ImageError> {
match (cs, has_alpha) {
(ColorSpace::Gray, false) => {
image.decode_into(buf)?;
}
(ColorSpace::Gray, true) => {
image.decode_into(buf)?;
}
(ColorSpace::RGB, false) => {
image.decode_into(buf)?;
}
(ColorSpace::RGB, true) => {
image.decode_into(buf)?;
}
(ColorSpace::CMYK, false) => {
let decoded = image.decode()?;
let transformed = from_icc(CMYK_PROFILE, 4, has_alpha, width, height, &decoded)
.map_err(icc_err_to_image)?;
buf.copy_from_slice(&transformed);
}
(ColorSpace::CMYK, true) => {
let decoded = image.decode()?;
let mut cmyk = vec![];
let mut alpha = vec![];
for sample in decoded.chunks_exact(5) {
cmyk.extend_from_slice(&sample[..4]);
alpha.push(sample[4]);
}
let rgb = from_icc(CMYK_PROFILE, 4, false, width, height, &cmyk)
.map_err(icc_err_to_image)?;
for (out, pixel) in buf.chunks_exact_mut(4).zip(
rgb.chunks_exact(3)
.zip(alpha)
.map(|(rgb, alpha)| [rgb[0], rgb[1], rgb[2], alpha]),
) {
out.copy_from_slice(&pixel);
}
}
(
ColorSpace::Icc {
profile,
num_channels: num_components,
},
has_alpha,
) => {
let decoded = image.decode()?;
let transformed =
from_icc(&profile, num_components, has_alpha, width, height, &decoded);
if let Ok(transformed) = transformed {
buf.copy_from_slice(&transformed);
} else {
match num_components {
1 => process(image, buf, width, height, has_alpha, ColorSpace::Gray)?,
3 => process(image, buf, width, height, has_alpha, ColorSpace::RGB)?,
4 => process(image, buf, width, height, has_alpha, ColorSpace::CMYK)?,
_ => {
return Err(unsupported_color_error(image.original_color_type()));
}
}
};
}
(ColorSpace::Unknown { .. }, _) => {
return Err(unsupported_color_error(image.original_color_type()));
}
};
Ok(())
}
process(image, buf, width, height, has_alpha, color_space)
}
impl From<crate::DecodeError> for DecodingError {
fn from(value: crate::DecodeError) -> Self {
let format = ImageFormatHint::Name("JPEG2000".to_owned());
Self::new(format, value)
}
}
impl From<crate::DecodeError> for ImageError {
fn from(value: crate::DecodeError) -> Self {
Self::Decoding(value.into())
}
}
fn icc_err_to_image(err: CmsError) -> ImageError {
let format = ImageFormatHint::Name("JPEG2000".to_owned());
ImageError::Decoding(DecodingError::new(format, err))
}
fn unsupported_color_error(color: ExtendedColorType) -> ImageError {
ImageError::Unsupported(image::error::UnsupportedError::from_format_and_kind(
ImageFormatHint::Name("JPEG2000".to_owned()),
image::error::UnsupportedErrorKind::Color(color),
))
}
pub fn register_decoding_hook() -> bool {
if decoding_hook_registered(OsStr::new("jp2")) {
return false;
}
for extension in ["jp2", "jpg2", "j2k", "jpf"] {
image::hooks::register_decoding_hook(
extension.into(),
Box::new(|r| Ok(Box::new(Jp2Decoder::new(r)?))),
);
register_format_detection_hook(extension.into(), crate::JP2_MAGIC, None);
}
for extension in ["j2c", "jpc"] {
image::hooks::register_decoding_hook(
extension.into(),
Box::new(|r| Ok(Box::new(Jp2Decoder::new(r)?))),
);
register_format_detection_hook(extension.into(), crate::CODESTREAM_MAGIC, None);
}
true
}