use anyhow::{Context as _, Result};
use color_quant::NeuQuant;
use image::{
DynamicImage, GenericImageView, ImageBuffer, Pixel, Rgb,
imageops::{FilterType, colorops},
};
use rustix::event::{PollFd, PollFlags, Timespec, poll};
use rustix::io::read;
use rustix::termios::{LocalModes, OptionalActions, tcgetattr, tcgetwinsize, tcsetattr};
use std::io::{Write, stdout};
use std::os::fd::AsFd;
use std::time::Instant;
pub struct SixelBackend;
impl SixelBackend {
pub fn supported() -> Result<bool> {
let stdin = std::io::stdin();
let old_attributes = {
let old = tcgetattr(&stdin).context("Failed to recieve terminal attibutes")?;
let mut new = old.clone();
new.local_modes &= !LocalModes::ICANON;
new.local_modes &= !LocalModes::ECHO;
tcsetattr(&stdin, OptionalActions::Now, &new)
.context("Failed to update terminal attributes")?;
old
};
print!("\x1B[c");
stdout().flush()?;
let start_time = Instant::now();
let stdin_fd = stdin.as_fd();
let mut stdin_pollfd = [PollFd::new(&stdin_fd, PollFlags::IN)];
let mut buf = Vec::<u8>::new();
loop {
while poll(&mut stdin_pollfd, Some(&Timespec::default()))? < 1 {
if start_time.elapsed().as_millis() > 50 {
tcsetattr(stdin, OptionalActions::Now, &old_attributes)
.context("Failed to update terminal attributes")?;
return Ok(false);
}
}
let mut byte = [0];
read(&stdin, &mut byte)?;
buf.push(byte[0]);
if buf.starts_with(&[0x1B, b'[', b'?']) && buf.ends_with(b"c") {
for attribute in buf[3..(buf.len() - 1)].split(|x| *x == b';') {
if attribute == [b'4'] {
tcsetattr(stdin, OptionalActions::Now, &old_attributes)
.context("Failed to update terminal attributes")?;
return Ok(true);
}
}
}
}
}
}
impl super::ImageBackend for SixelBackend {
#[allow(clippy::map_entry)]
fn add_image(&self, lines: Vec<String>, image: &DynamicImage, colors: usize) -> Result<String> {
let tty_size = tcgetwinsize(std::io::stdin())?;
let cw = tty_size.ws_xpixel / tty_size.ws_col;
let lh = tty_size.ws_ypixel / tty_size.ws_row;
let width_ratio = 1.0 / cw as f64;
let height_ratio = 1.0 / lh as f64;
let image = image.resize(
u32::MAX,
(lines.len() as f64 / height_ratio) as u32,
FilterType::Lanczos3,
);
let image_columns = width_ratio * image.width() as f64;
let image_rows = height_ratio * image.height() as f64;
let rgba_image = image.to_rgba8(); let flat_samples = rgba_image.as_flat_samples();
let mut rgba_image = rgba_image.clone();
let pixels = flat_samples
.image_slice()
.context("Error while slicing the image")?;
colorops::dither(&mut rgba_image, &NeuQuant::new(10, colors, pixels));
let rgb_image = ImageBuffer::from_fn(rgba_image.width(), rgba_image.height(), |x, y| {
let rgba_pixel = rgba_image.get_pixel(x, y);
let mut rgb_pixel = rgba_pixel.to_rgb();
for subpixel in &mut rgb_pixel.0 {
*subpixel = (*subpixel as f32 / 255.0 * rgba_pixel[3] as f32) as u8;
}
rgb_pixel
});
let mut image_data = Vec::<u8>::new();
image_data.extend(b"\x1BPq"); image_data.extend(format!("\"1;1;{};{}", image.width(), image.height()).as_bytes());
let mut colors = std::collections::HashMap::<Rgb<u8>, u8>::new();
for i in 0..((rgb_image.height() - 1) / 6 + 1) {
let sixel_row = rgb_image.view(
0,
i * 6,
rgb_image.width(),
std::cmp::min(6, rgb_image.height() - i * 6),
);
for (_, _, pixel) in sixel_row.pixels() {
if !colors.contains_key(&pixel) {
let color_multiplier = 100.0 / 255.0;
image_data.extend(
format!(
"#{};2;{};{};{}",
colors.len(),
(pixel[0] as f32 * color_multiplier) as u32,
(pixel[1] as f32 * color_multiplier) as u32,
(pixel[2] as f32 * color_multiplier) as u32
)
.as_bytes(),
);
colors.insert(pixel, colors.len() as u8);
}
}
for (color, color_index) in &colors {
let mut sixel_samples = vec![0; sixel_row.width() as usize];
sixel_samples.resize(sixel_row.width() as usize, 0);
for (x, y, pixel) in sixel_row.pixels() {
if color == &pixel {
sixel_samples[x as usize] |= 1 << y;
}
}
image_data.extend(format!("#{color_index}").bytes());
image_data.extend(sixel_samples.iter().map(|x| x + 0x3F));
image_data.push(b'$');
}
image_data.push(b'-');
}
image_data.extend(b"\x1B\\");
image_data.extend(format!("\x1B[{}A", image_rows as u32 - 1).as_bytes()); image_data.extend(format!("\x1B[{}C", image_columns as u32 + 1).as_bytes()); let mut i = 0;
for line in &lines {
image_data.extend(format!("\x1B[s{line}\x1B[u\x1B[1B").as_bytes());
i += 1;
}
image_data
.extend(format!("\n\x1B[{}B", lines.len().max(image_rows as usize) - i).as_bytes());
Ok(String::from_utf8(image_data)?)
}
}