#![doc(html_logo_url = "https://gif.ski/icon.png")]
#[macro_use] extern crate quick_error;
use imagequant::*;
use imgref::*;
use rgb::*;
mod error;
pub use crate::error::*;
mod ordqueue;
use crate::ordqueue::*;
pub mod progress;
use crate::progress::*;
pub mod c_api;
mod encoderust;
#[cfg(feature = "gifsicle")]
mod encodegifsicle;
use std::io::prelude::*;
use std::path::PathBuf;
use crossbeam_channel::{Sender, Receiver};
use std::thread;
type DecodedImage = CatResult<(ImgVec<RGBA8>, f64)>;
#[derive(Debug, Copy, Clone)]
pub enum Repeat {
Finite(u16),
Infinite,
}
#[derive(Copy, Clone)]
pub struct Settings {
pub width: Option<u32>,
pub height: Option<u32>,
pub quality: u8,
pub fast: bool,
pub repeat: Repeat,
}
impl Settings {
#[cfg(not(feature = "gifsicle"))]
pub(crate) fn color_quality(&self) -> u8 {
self.quality
}
#[cfg(feature = "gifsicle")]
pub(crate) fn color_quality(&self) -> u8 {
(self.quality * 2).min(100)
}
pub fn dimensions_for_image(&self, width: usize, height: usize) -> (usize, usize) {
dimensions_for_image((width, height), (self.width, self.height))
}
}
pub struct Collector {
width: Option<u32>,
height: Option<u32>,
queue: OrdQueue<DecodedImage>,
}
pub struct Writer {
queue_iter: Option<OrdQueueIter<DecodedImage>>,
settings: Settings,
}
struct GIFFrame {
left: u16,
top: u16,
screen_width: u16,
screen_height: u16,
image: ImgVec<u8>,
pal: Vec<RGBA8>,
dispose: gif::DisposalMethod,
transparent_index: Option<u8>,
}
trait Encoder {
fn write_frame(&mut self, frame: GIFFrame, delay: u16, settings: &Settings) -> CatResult<()>;
fn finish(&mut self) -> CatResult<()> {
Ok(())
}
}
struct DiffMessage {
ordinal_frame_number: usize,
end_pts: f64,
dispose: gif::DisposalMethod,
image: ImgVec<RGBA8>,
importance_map: Vec<u8>,
}
struct RemapMessage {
ordinal_frame_number: usize,
end_pts: f64,
dispose: gif::DisposalMethod,
liq: Attributes,
remap: QuantizationResult,
liq_image: Image<'static>,
}
struct FrameMessage {
ordinal_frame_number: usize,
end_pts: f64,
frame: GIFFrame,
}
pub fn new(settings: Settings) -> CatResult<(Collector, Writer)> {
let (queue, queue_iter) = ordqueue::new(4);
Ok((
Collector {
queue,
width: settings.width,
height: settings.height,
},
Writer {
queue_iter: Some(queue_iter),
settings,
},
))
}
impl Collector {
pub fn add_frame_rgba(&mut self, frame_index: usize, image: ImgVec<RGBA8>, presentation_timestamp: f64) -> CatResult<()> {
self.queue.push(frame_index, Ok((Self::resized_binary_alpha(image, self.width, self.height), presentation_timestamp)))
}
pub fn add_frame_png_file(&mut self, frame_index: usize, path: PathBuf, presentation_timestamp: f64) -> CatResult<()> {
let width = self.width;
let height = self.height;
let image = lodepng::decode32_file(&path)
.map_err(|err| Error::PNG(format!("Can't load {}: {}", path.display(), err)))?;
self.queue.push(frame_index, Ok((Self::resized_binary_alpha(ImgVec::new(image.buffer, image.width, image.height), width, height), presentation_timestamp)))
}
fn resized_binary_alpha(mut image: ImgVec<RGBA8>, width: Option<u32>, height: Option<u32>) -> ImgVec<RGBA8> {
let (width, height) = dimensions_for_image((image.width(), image.height()), (width, height));
if width != image.width() || height != image.height() {
let (buf, img_width, img_height) = image.into_contiguous_buf();
assert_eq!(buf.len(), img_width * img_height);
let mut r = resize::new(img_width, img_height, width, height, resize::Pixel::RGBA, resize::Type::Lanczos3);
let mut dst = vec![RGBA8::new(0, 0, 0, 0); width * height];
r.resize(buf.as_bytes(), dst.as_bytes_mut());
image = ImgVec::new(dst, width, height)
}
const DITHER: [u8; 64] = [
0*2+8,48*2+8,12*2+8,60*2+8, 3*2+8,51*2+8,15*2+8,63*2+8,
32*2+8,16*2+8,44*2+8,28*2+8,35*2+8,19*2+8,47*2+8,31*2+8,
8*2+8,56*2+8, 4*2+8,52*2+8,11*2+8,59*2+8, 7*2+8,55*2+8,
40*2+8,24*2+8,36*2+8,20*2+8,43*2+8,27*2+8,39*2+8,23*2+8,
2*2+8,50*2+8,14*2+8,62*2+8, 1*2+8,49*2+8,13*2+8,61*2+8,
34*2+8,18*2+8,46*2+8,30*2+8,33*2+8,17*2+8,45*2+8,29*2+8,
10*2+8,58*2+8, 6*2+8,54*2+8, 9*2+8,57*2+8, 5*2+8,53*2+8,
42*2+8,26*2+8,38*2+8,22*2+8,41*2+8,25*2+8,37*2+8,21*2+8];
for (y, row) in image.rows_mut().enumerate() {
for (x, px) in row.iter_mut().enumerate() {
if px.a < 255 {
px.a = if px.a < DITHER[(y & 7) * 8 + (x & 7)] { 0 } else { 255 };
}
}
}
image
}
}
fn dimensions_for_image((img_w, img_h): (usize, usize), resize_to: (Option<u32>, Option<u32>)) -> (usize, usize) {
match resize_to {
(None, None) => {
let factor = (img_w * img_h + 800*600) / (800*600);
if factor > 1 {
(img_w / factor, img_h / factor)
} else {
(img_w, img_h)
}
},
(Some(w), Some(h)) => {
((w as usize).min(img_w), (h as usize).min(img_h))
},
(Some(w), None) => {
let w = (w as usize).min(img_w);
(w, img_h * w / img_w)
}
(None, Some(h)) => {
let h = (h as usize).min(img_h);
(img_w * h / img_h, h)
},
}
}
impl Writer {
fn quantize(image: ImgRef<'_, RGBA8>, importance_map: &[u8], has_prev_frame: bool, settings: &Settings) -> CatResult<(Attributes, QuantizationResult, Image<'static>)> {
let mut liq = Attributes::new();
if settings.fast {
liq.set_speed(10);
}
let quality = if has_prev_frame {
settings.color_quality().into()
} else {
100
};
liq.set_quality(0, quality);
let mut img = liq.new_image_stride_copy(image.buf(), image.width(), image.height(), image.stride(), 0.).expect("stridecopy");
img.set_importance_map(importance_map).expect("immap");
if has_prev_frame {
img.add_fixed_color(RGBA8::new(0, 0, 0, 0));
}
let res = liq.quantize(&img).expect("quantize");
Ok((liq, res, img))
}
fn remap(liq: Attributes, mut res: QuantizationResult, mut img: Image<'static>, background: Option<ImgRef<'_, RGBA8>>, settings: &Settings) -> CatResult<(ImgVec<u8>, Vec<RGBA8>)> {
if let Some(bg) = background {
img.set_background(liq.new_image_stride(bg.buf(), bg.width(), bg.height(), bg.stride(), 0.)?)?;
}
res.set_dithering_level(settings.quality as f32 / 150.0);
let (pal, pal_img) = res.remapped(&mut img)?;
debug_assert_eq!(img.width() * img.height(), pal_img.len());
Ok((Img::new(pal_img, img.width(), img.height()), pal))
}
fn write_frames(write_queue: Receiver<FrameMessage>, enc: &mut dyn Encoder, settings: &Settings, reporter: &mut dyn ProgressReporter) -> CatResult<()> {
let mut pts_in_delay_units = 0_u64;
let mut n_done = 0;
for FrameMessage {frame, ordinal_frame_number, end_pts, ..} in write_queue {
let delay = ((end_pts * 100.0).round() as u64)
.saturating_sub(pts_in_delay_units)
.min(30000) as u16;
pts_in_delay_units += u64::from(delay);
if delay != 0 {
enc.write_frame(frame, delay, settings)?;
}
while n_done < ordinal_frame_number {
n_done += 1;
if !reporter.increase() {
return Err(Error::Aborted.into());
}
}
}
enc.finish()?;
Ok(())
}
#[allow(unused_mut)]
pub fn write<W: Write>(self, mut writer: W, reporter: &mut dyn ProgressReporter) -> CatResult<()> {
#[cfg(feature = "gifsicle")]
{
if self.settings.quality < 100 {
let loss = (100 - self.settings.quality as u32) * 6;
let mut gifsicle = encodegifsicle::Gifsicle::new(loss, &mut writer);
return self.write_with_encoder(&mut gifsicle, reporter);
}
}
self.write_with_encoder(&mut encoderust::RustEncoder::new(writer), reporter)
}
fn write_with_encoder(mut self, encoder: &mut dyn Encoder, reporter: &mut dyn ProgressReporter) -> CatResult<()> {
let decode_queue_recv = self.queue_iter.take().ok_or(Error::Aborted)?;
let settings = self.settings;
let (quant_queue, quant_queue_recv) = crossbeam_channel::bounded(4);
let diff_thread = thread::Builder::new().name("diff".into()).spawn(move || {
Self::make_diffs(decode_queue_recv, quant_queue, &settings)
})?;
let (remap_queue, remap_queue_recv) = crossbeam_channel::bounded(8);
let quant_thread = thread::Builder::new().name("quant".into()).spawn(move || {
Self::quantize_frames(quant_queue_recv, remap_queue, &settings)
})?;
let (write_queue, write_queue_recv) = crossbeam_channel::bounded(6);
let remap_thread = thread::Builder::new().name("remap".into()).spawn(move || {
Self::remap_frames(remap_queue_recv, write_queue, &settings)
})?;
Self::write_frames(write_queue_recv, encoder, &self.settings, reporter)?;
diff_thread.join().map_err(|_| Error::ThreadSend)??;
quant_thread.join().map_err(|_| Error::ThreadSend)??;
remap_thread.join().map_err(|_| Error::ThreadSend)??;
Ok(())
}
fn make_diffs(mut inputs: OrdQueueIter<DecodedImage>, quant_queue: Sender<DiffMessage>, _settings: &Settings) -> CatResult<()> {
let mut next_frame = inputs.next().transpose()?;
let first_frame_pts = next_frame.as_ref().map(|&(_, pts)| pts).unwrap_or_default();
let mut prev_frame_pts = 0.0;
let mut ordinal_frame_number = 0;
while let Some((image, mut pts)) = {
let curr_frame = next_frame.take();
next_frame = inputs.next().transpose()?;
curr_frame
} {
pts -= first_frame_pts;
ordinal_frame_number += 1;
let mut dispose = gif::DisposalMethod::Keep;
let importance_map = if let Some((next, _)) = &next_frame {
if next.width() != image.width() || next.height() != image.height() {
return Err(Error::WrongSize(format!("Frame {} has wrong size ({}×{}, expected {}×{})", ordinal_frame_number,
next.width(), next.height(), image.width(), image.height())));
}
if next.as_ref() == image.as_ref() {
prev_frame_pts = pts;
continue;
}
let mut importance_map = Vec::with_capacity(image.width() * image.height());
importance_map.extend(next.rows().zip(image.rows()).flat_map(|(n, curr)| n.iter().copied().zip(curr.iter().copied())).map(|(n, curr)| {
if n.a < curr.a {
dispose = gif::DisposalMethod::Background;
}
255 - (colordiff(n, curr) / (255 * 255 * 6 / 170)) as u8
}));
importance_map
} else {
dispose = gif::DisposalMethod::Background;
vec![255; image.width() * image.height()]
};
let end_pts = if let Some((_, next_pts)) = next_frame {
next_pts - first_frame_pts
} else if first_frame_pts > 1./100. {
pts + first_frame_pts
} else {
pts + (pts - prev_frame_pts)
};
prev_frame_pts = pts;
quant_queue.send(DiffMessage {
dispose,
importance_map,
ordinal_frame_number,
image,
end_pts,
})?;
}
Ok(())
}
fn quantize_frames(inputs: Receiver<DiffMessage>, remap_queue: Sender<RemapMessage>, settings: &Settings) -> CatResult<()> {
let next_frame = inputs.recv().map_err(|_| Error::NoFrames)?;
let mut next_frame = Some(next_frame);
let mut prev_frame: Option<ImgVec<_>> = None;
while let Some(DiffMessage {image, end_pts, dispose, ordinal_frame_number, mut importance_map}) = {
let curr_frame = next_frame.take();
next_frame = inputs.recv().ok();
curr_frame
} {
if let Some(prev_frame) = &prev_frame {
let q = 100 - u32::from(settings.color_quality());
let min_diff = 80 + q * q;
importance_map
.chunks_exact_mut(image.width())
.zip(prev_frame.rows().zip(image.rows()))
.flat_map(|(imp, (bg, px))| {
imp.iter_mut().zip(bg.iter().copied().zip(px.iter().copied()))
})
.for_each(|(imp, (bg, px))| {
let diff = colordiff(bg, px);
*imp = if diff < min_diff {
0
} else {
let t = diff / 32;
((t * t).min(256) as u16 * u16::from(*imp) / 256) as u8
}
});
}
let (liq, remap, liq_image) = Self::quantize(image.as_ref(), &importance_map, ordinal_frame_number > 1, settings)?;
remap_queue.send(RemapMessage {
ordinal_frame_number,
end_pts,
dispose,
liq, remap,
liq_image,
})?;
prev_frame = if dispose == gif::DisposalMethod::Keep { Some(image) } else { None };
}
Ok(())
}
fn remap_frames(inputs: Receiver<RemapMessage>, write_queue: Sender<FrameMessage>, settings: &Settings) -> CatResult<()> {
let next_frame = inputs.recv().map_err(|_| Error::NoFrames)?;
let mut screen = gif_dispose::Screen::new(next_frame.liq_image.width(), next_frame.liq_image.height(), RGBA8::new(0, 0, 0, 0), None);
let mut next_frame = Some(next_frame);
let mut first_frame = true;
while let Some(RemapMessage {ordinal_frame_number, end_pts, dispose, liq, remap, liq_image}) = {
let curr_frame = next_frame.take();
next_frame = inputs.recv().ok();
curr_frame
} {
let screen_width = screen.pixels.width() as u16;
let screen_height = screen.pixels.height() as u16;
let mut screen_after_dispose = screen.dispose();
let (mut image8, mut image8_pal) = {
let bg = if !first_frame { Some(screen_after_dispose.pixels()) } else { None };
Self::remap(liq, remap, liq_image, bg, settings)?
};
let mut transparent_index = None;
for (i, p) in image8_pal.iter_mut().enumerate() {
if p.a <= 128 {
p.a = 0;
let new_index = i as u8;
if let Some(old_index) = transparent_index {
image8.pixels_mut().filter(|px| **px == new_index).for_each(|px| *px = old_index);
} else {
transparent_index = Some(new_index);
}
}
}
let (left, top, image8) = if !first_frame && next_frame.is_some() {
match trim_image(image8, &image8_pal, transparent_index, screen_after_dispose.pixels()) {
Some(trimmed) => trimmed,
None => continue,
}
} else {
(0, 0, image8)
};
let frame = GIFFrame {
left,
top,
screen_width,
screen_height,
image: image8,
pal: image8_pal,
transparent_index,
dispose,
};
screen_after_dispose.then_blit(Some(&frame.pal), dispose, left, top as _, frame.image.as_ref(), transparent_index)?;
write_queue.send(FrameMessage {
ordinal_frame_number,
end_pts,
frame,
})?;
first_frame = false;
}
Ok(())
}
}
fn trim_image(mut image8: ImgVec<u8>, image8_pal: &[RGBA8], transparent_index: Option<u8>, screen: ImgRef<RGBA8>) -> Option<(u16, u16, ImgVec<u8>)> {
let mut image_trimmed = image8.as_ref();
let bottom = image_trimmed.rows().zip(screen.rows()).rev()
.take_while(|(img_row, screen_row)| {
img_row.iter().copied().zip(screen_row.iter().copied())
.all(|(px, bg)| {
Some(px) == transparent_index || image8_pal.get(px as usize) == Some(&bg)
})
})
.count();
if bottom > 0 {
if bottom == image_trimmed.height() {
return None;
}
image_trimmed = image_trimmed.sub_image(0, 0, image_trimmed.width(), image_trimmed.height() - bottom);
}
let top = image_trimmed.rows().zip(screen.rows())
.take_while(|(img_row, screen_row)| {
img_row.iter().copied().zip(screen_row.iter().copied())
.all(|(px, bg)| {
Some(px) == transparent_index || image8_pal.get(px as usize) == Some(&bg)
})
})
.count();
if top > 0 {
image_trimmed = image_trimmed.sub_image(0, top, image_trimmed.width(), image_trimmed.height() - top);
}
if image_trimmed.height() != image8.height() {
let (buf, width, height) = image_trimmed.to_contiguous_buf();
image8 = Img::new(buf.into_owned(), width, height);
}
Some((0, top as _, image8))
}
#[inline]
fn colordiff(a: RGBA8, b: RGBA8) -> u32 {
if a.a == 0 || b.a == 0 {
return 255 * 255 * 6;
}
(i32::from(i16::from(a.r) - i16::from(b.r)) * i32::from(i16::from(a.r) - i16::from(b.r))) as u32 * 2 +
(i32::from(i16::from(a.g) - i16::from(b.g)) * i32::from(i16::from(a.g) - i16::from(b.g))) as u32 * 3 +
(i32::from(i16::from(a.b) - i16::from(b.b)) * i32::from(i16::from(a.b) - i16::from(b.b))) as u32
}