use std::ops::{Bound, Deref, DerefMut, Range, RangeBounds};
use std::sync::Arc;
use euclid::default::{Rect, Size2D};
use image::codecs::jpeg::JpegEncoder;
use image::codecs::png::PngEncoder;
use image::codecs::webp::WebPEncoder;
use image::{ExtendedColorType, GenericImageView, ImageEncoder, ImageError, Rgb};
use malloc_size_of_derive::MallocSizeOf;
use serde::{Deserialize, Serialize};
use servo_base::generic_channel::GenericSharedMemory;
use crate::{EncodedImageType, Multiply, rgba8_get_rect, transform_inplace};
#[derive(Clone, Copy, Debug, Default, Deserialize, Eq, MallocSizeOf, PartialEq, Serialize)]
pub enum SnapshotPixelFormat {
#[default]
RGBA,
BGRA,
}
#[derive(Clone, Copy, Debug, Deserialize, Eq, MallocSizeOf, PartialEq, Serialize)]
pub enum Alpha {
Premultiplied,
NotPremultiplied,
DontCare,
}
impl Alpha {
pub const fn from_premultiplied(is_premultiplied: bool) -> Self {
if is_premultiplied {
Self::Premultiplied
} else {
Self::NotPremultiplied
}
}
pub const fn needs_alpha_multiplication(&self) -> bool {
match self {
Alpha::Premultiplied => false,
Alpha::NotPremultiplied => true,
Alpha::DontCare => false,
}
}
}
#[derive(Clone, Copy, Debug, Deserialize, Eq, MallocSizeOf, PartialEq, Serialize)]
pub enum SnapshotAlphaMode {
Opaque,
AsOpaque { premultiplied: bool },
Transparent { premultiplied: bool },
}
impl Default for SnapshotAlphaMode {
fn default() -> Self {
Self::Transparent {
premultiplied: true,
}
}
}
impl SnapshotAlphaMode {
pub const fn alpha(&self) -> Alpha {
match self {
SnapshotAlphaMode::Opaque => Alpha::DontCare,
SnapshotAlphaMode::AsOpaque { premultiplied } => {
Alpha::from_premultiplied(*premultiplied)
},
SnapshotAlphaMode::Transparent { premultiplied } => {
Alpha::from_premultiplied(*premultiplied)
},
}
}
}
#[derive(Clone, Debug, MallocSizeOf)]
pub enum SnapshotData {
SharedMemory(
#[conditional_malloc_size_of] Arc<GenericSharedMemory>,
Range<usize>,
),
SharedVec(#[conditional_malloc_size_of] Arc<Vec<u8>>, Range<usize>),
Owned(Vec<u8>),
}
impl SnapshotData {
fn to_vec(&self) -> Vec<u8> {
match &self {
SnapshotData::SharedMemory(data, byte_range) => Vec::from(&data[byte_range.clone()]),
SnapshotData::SharedVec(data, byte_range) => Vec::from(&data[byte_range.clone()]),
SnapshotData::Owned(data) => data.clone(),
}
}
}
impl DerefMut for SnapshotData {
fn deref_mut(&mut self) -> &mut Self::Target {
match self {
SnapshotData::SharedMemory(..) | SnapshotData::SharedVec(..) => {
*self = SnapshotData::Owned(self.to_vec());
&mut *self
},
SnapshotData::Owned(items) => items,
}
}
}
impl Deref for SnapshotData {
type Target = [u8];
fn deref(&self) -> &Self::Target {
match &self {
SnapshotData::SharedMemory(data, byte_range) => &data[byte_range.clone()],
SnapshotData::SharedVec(data, byte_range) => &data[byte_range.clone()],
SnapshotData::Owned(items) => items,
}
}
}
#[derive(Clone, Debug, MallocSizeOf)]
pub struct Snapshot {
size: Size2D<u32>,
data: SnapshotData,
format: SnapshotPixelFormat,
alpha_mode: SnapshotAlphaMode,
}
impl Snapshot {
pub const fn size(&self) -> Size2D<u32> {
self.size
}
pub const fn format(&self) -> SnapshotPixelFormat {
self.format
}
pub const fn alpha_mode(&self) -> SnapshotAlphaMode {
self.alpha_mode
}
pub fn empty() -> Self {
Self {
size: Size2D::zero(),
data: SnapshotData::Owned(vec![]),
format: SnapshotPixelFormat::RGBA,
alpha_mode: SnapshotAlphaMode::Transparent {
premultiplied: true,
},
}
}
pub fn cleared(size: Size2D<u32>) -> Self {
Self {
size,
data: SnapshotData::Owned(vec![0; size.area() as usize * 4]),
format: SnapshotPixelFormat::RGBA,
alpha_mode: SnapshotAlphaMode::Transparent {
premultiplied: true,
},
}
}
pub fn from_vec(
size: Size2D<u32>,
format: SnapshotPixelFormat,
alpha_mode: SnapshotAlphaMode,
data: Vec<u8>,
) -> Self {
Self {
size,
data: SnapshotData::Owned(data),
format,
alpha_mode,
}
}
pub fn from_arc_vec(
size: Size2D<u32>,
format: SnapshotPixelFormat,
alpha_mode: SnapshotAlphaMode,
data: Arc<Vec<u8>>,
byte_range_bounds: impl RangeBounds<usize>,
) -> Self {
let range_start = match byte_range_bounds.start_bound() {
Bound::Included(bound) => *bound,
Bound::Excluded(bound) => *bound + 1,
Bound::Unbounded => 0,
};
let range_end = match byte_range_bounds.end_bound() {
Bound::Included(bound) => *bound + 1,
Bound::Excluded(bound) => *bound,
Bound::Unbounded => data.len(),
};
Self {
size,
data: SnapshotData::SharedVec(data, range_start..range_end),
format,
alpha_mode,
}
}
pub fn get_rect(&self, rect: Rect<u32>) -> Self {
let data = rgba8_get_rect(self.as_raw_bytes(), self.size(), rect).to_vec();
Self::from_vec(rect.size, self.format, self.alpha_mode, data)
}
pub fn transform(
&mut self,
target_alpha_mode: SnapshotAlphaMode,
target_format: SnapshotPixelFormat,
) {
if self.alpha_mode == target_alpha_mode && target_format == self.format {
return;
}
let swap_rb = target_format != self.format;
let multiply = match (self.alpha_mode, target_alpha_mode) {
(SnapshotAlphaMode::Opaque, _) => Multiply::None,
(alpha_mode, SnapshotAlphaMode::Opaque)
if alpha_mode.alpha() == Alpha::Premultiplied =>
{
Multiply::UnMultiply
},
(_, SnapshotAlphaMode::Opaque) => Multiply::None,
(
SnapshotAlphaMode::Transparent { premultiplied } |
SnapshotAlphaMode::AsOpaque { premultiplied },
SnapshotAlphaMode::Transparent {
premultiplied: target_premultiplied,
} |
SnapshotAlphaMode::AsOpaque {
premultiplied: target_premultiplied,
},
) => {
if premultiplied == target_premultiplied {
Multiply::None
} else if target_premultiplied {
Multiply::PreMultiply
} else {
Multiply::UnMultiply
}
},
};
let clear_alpha = !matches!(self.alpha_mode, SnapshotAlphaMode::Opaque) &&
matches!(target_alpha_mode, SnapshotAlphaMode::Opaque);
if matches!(multiply, Multiply::None) && !swap_rb && !clear_alpha {
return;
}
transform_inplace(self.data.deref_mut(), multiply, swap_rb, clear_alpha);
self.alpha_mode = target_alpha_mode;
self.format = target_format;
}
pub fn as_raw_bytes(&self) -> &[u8] {
&self.data
}
pub fn as_raw_bytes_mut(&mut self) -> &mut [u8] {
&mut self.data
}
pub fn to_shared(&self) -> SharedSnapshot {
let (data, byte_range) = match &self.data {
SnapshotData::SharedMemory(data, byte_range) => (data.clone(), byte_range.clone()),
SnapshotData::SharedVec(data, byte_range) => (
Arc::new(GenericSharedMemory::from_bytes(data)),
byte_range.clone(),
),
SnapshotData::Owned(data) => (
Arc::new(GenericSharedMemory::from_bytes(data)),
0..data.len(),
),
};
SharedSnapshot {
size: self.size,
data,
byte_range,
format: self.format,
alpha_mode: self.alpha_mode,
}
}
pub fn encode_for_mime_type<W: std::io::Write>(
&mut self,
image_type: &EncodedImageType,
quality: Option<f64>,
encoder: &mut W,
) -> Result<(), ImageError> {
let width = self.size.width;
let height = self.size.height;
let alpha_mode = match image_type {
EncodedImageType::Jpeg => SnapshotAlphaMode::AsOpaque {
premultiplied: true,
},
_ => SnapshotAlphaMode::Transparent {
premultiplied: false,
},
};
self.transform(alpha_mode, SnapshotPixelFormat::RGBA);
let data = &self.data;
match image_type {
EncodedImageType::Png => {
PngEncoder::new(encoder).write_image(data, width, height, ExtendedColorType::Rgba8)
},
EncodedImageType::Jpeg => {
let mut jpeg_encoder = if let Some(quality) = quality {
if (0.0..=1.0).contains(&quality) {
JpegEncoder::new_with_quality(
encoder,
(quality * 100.0).round().clamp(1.0, 100.0) as u8,
)
} else {
JpegEncoder::new(encoder)
}
} else {
JpegEncoder::new(encoder)
};
struct RgbaDataForJpegEncoder<'a> {
width: u32,
height: u32,
data: &'a [u8],
}
impl<'a> GenericImageView for RgbaDataForJpegEncoder<'a> {
type Pixel = Rgb<u8>;
fn dimensions(&self) -> (u32, u32) {
(self.width, self.height)
}
fn get_pixel(&self, x: u32, y: u32) -> Self::Pixel {
let offset = (self.width * y + x) as usize * 4;
Rgb([
self.data[offset],
self.data[offset + 1],
self.data[offset + 2],
])
}
}
let image = RgbaDataForJpegEncoder {
width,
height,
data,
};
jpeg_encoder.encode_image(&image)
},
EncodedImageType::Webp => {
WebPEncoder::new_lossless(encoder).write_image(
data,
width,
height,
ExtendedColorType::Rgba8,
)
},
}
}
}
impl From<Snapshot> for Vec<u8> {
fn from(value: Snapshot) -> Self {
match value.data {
SnapshotData::SharedMemory(..) => Vec::from(value.as_raw_bytes()),
SnapshotData::SharedVec(..) => Vec::from(value.as_raw_bytes()),
SnapshotData::Owned(data) => data,
}
}
}
#[derive(Clone, Debug, Deserialize, MallocSizeOf, Serialize)]
pub struct SharedSnapshot {
size: Size2D<u32>,
#[conditional_malloc_size_of]
data: Arc<GenericSharedMemory>,
byte_range: Range<usize>,
format: SnapshotPixelFormat,
alpha_mode: SnapshotAlphaMode,
}
impl SharedSnapshot {
pub fn to_owned(&self) -> Snapshot {
Snapshot {
size: self.size,
data: SnapshotData::SharedMemory(self.data.clone(), self.byte_range.clone()),
format: self.format,
alpha_mode: self.alpha_mode,
}
}
pub fn cleared(size: Size2D<u32>) -> Self {
let length_in_bytes = size.area() as usize * 4;
Self {
size,
data: Arc::new(GenericSharedMemory::from_byte(0, length_in_bytes)),
byte_range: 0..length_in_bytes,
format: SnapshotPixelFormat::RGBA,
alpha_mode: SnapshotAlphaMode::Transparent {
premultiplied: true,
},
}
}
pub const fn size(&self) -> Size2D<u32> {
self.size
}
pub const fn format(&self) -> SnapshotPixelFormat {
self.format
}
pub const fn alpha_mode(&self) -> SnapshotAlphaMode {
self.alpha_mode
}
pub fn data(&self) -> &[u8] {
&(&self.data)[self.byte_range.clone()]
}
pub fn shared_memory(&self) -> GenericSharedMemory {
(*self.data).clone()
}
}