use crate::image::Image;
use lcms2::{LCMSResult, Profile, InfoType, Locale};
use std::fs;
use std::io;
use std::io::Read;
use std::path::Path;
#[derive(Copy, Clone, Eq, PartialEq, Default)]
pub enum Profiles {
#[default]
All,
NonsRGB,
None,
}
#[derive(Clone, Default)]
pub struct Loader {
pub max_image_area: usize,
pub(crate) discard_alpha: bool,
pub(crate) metadata: bool,
pub(crate) profiles: Profiles,
}
impl Loader {
#[inline(always)]
#[must_use]
pub fn new() -> Self {
Loader {
discard_alpha: false,
metadata: false,
profiles: Profiles::NonsRGB,
max_image_area: 16000 * 16000,
}
}
#[inline(always)]
pub fn opaque(&mut self, discard_alpha: bool) -> &mut Self {
self.discard_alpha = discard_alpha;
self
}
#[inline(always)]
pub fn metadata(&mut self, keep_metadata: bool) -> &mut Self {
self.metadata = keep_metadata;
self
}
#[inline(always)]
pub fn profiles(&mut self, convert_profiles: Profiles) -> &mut Self {
self.profiles = convert_profiles;
self
}
#[inline(always)]
pub fn max_image_area(&mut self, max_image_area: usize) -> &mut Self {
self.max_image_area = max_image_area;
self
}
pub fn load_path<P: AsRef<Path>>(&self, path: P) -> Result<Image, crate::Error> {
let path = path.as_ref();
let mut data = Vec::new();
let (data, stat) = if path.as_os_str() == "-" {
data.try_reserve(1 << 16)?; io::stdin().lock().read_to_end(&mut data)?;
(data, None)
} else {
let mut file = fs::File::open(path)?;
let stat = file.metadata()?;
#[cfg(unix)] {
use std::os::unix::prelude::MetadataExt; data.try_reserve(stat.size() as usize + 1)?;
}
file.read_to_end(&mut data)?;
(data, Some(stat))
};
self.load_data_with_stat(&data, stat)
}
#[inline(always)]
pub fn load_data(&self, data: &[u8]) -> Result<Image, crate::Error> {
self.load_data_with_stat(data, None)
}
fn load_data_with_stat(&self, data: &[u8], meta: Option<fs::Metadata>) -> Result<Image, crate::Error> {
if data.starts_with(b"\x89PNG") {
return self.load_png(data, meta);
}
#[cfg(feature = "avif")]
if data.get(4..4+8) == Some(b"ftypavif") {
return self.load_avif(data, meta).map_err(|_| lodepng::Error::new(28).into());
}
#[cfg(feature = "webp")]
if data.get(0..4) == Some(b"RIFF") {
return self.load_webp(data, meta);
}
#[cfg(feature = "mozjpeg")]
if data.first() == Some(&0xFF) {
return self.load_mozjpeg(data, meta);
}
#[cfg(all(not(feature = "mozjpeg"), feature = "jpeg"))]
if data.first() == Some(&0xFF) {
return self.load_jpeg(data, meta);
}
Err(crate::Error::UnsupportedFileFormat)
}
pub(crate) fn process_profile(&self, profile: LCMSResult<Profile>) -> Option<Profile> {
match profile {
Err(_) => None,
Ok(profile) => {
if self.profiles == Profiles::NonsRGB {
if let Some(desc) = profile.info(InfoType::Description, Locale::new("en_US")) {
if desc.starts_with("sRGB ") {
return None;
}
}
}
if self.profiles == Profiles::None {
return None;
}
Some(profile)
},
}
}
pub(crate) fn check_dimensions(&self, width: usize, height: usize) -> Result<(), crate::Error> {
if width.checked_mul(height).is_none_or(|area| area > self.max_image_area) {
Err(crate::Error::ImageTooLarge)
} else {
Ok(())
}
}
}