Skip to content

Commit 4b64a5a

Browse files
committed
b1display: enable displaying animated GIFs
Signed-off-by: Daniel Schaefer <[email protected]>
1 parent 4c2603d commit 4b64a5a

File tree

3 files changed

+50
-8
lines changed

3 files changed

+50
-8
lines changed

b1display/README.md

+4-2
Original file line numberDiff line numberDiff line change
@@ -70,8 +70,10 @@ Options:
7070
Set/get power mode [possible values: low, high]
7171
--animation-fps [<ANIMATION_FPS>]
7272
Set/get animation FPS
73-
--image-bw <IMAGE_BW>
74-
Display black&white image (300x400px)
73+
--image <IMAGE>
74+
Display a black&white image (300x400px)
75+
--animated-gif <ANIMATED_GIF>
76+
Display an animated black&white GIF (300x400px)
7577
--clear-ram
7678
Clear display RAM
7779
-h, --help

inputmodule-control/src/b1display.rs

+6-2
Original file line numberDiff line numberDiff line change
@@ -76,9 +76,13 @@ pub struct B1DisplaySubcommand {
7676
#[arg(long)]
7777
pub animation_fps: Option<Option<u16>>,
7878

79-
/// Display black&white image (300x400px)
79+
/// Display a black&white image (300x400px)
8080
#[arg(long)]
81-
pub image_bw: Option<String>,
81+
pub image: Option<String>,
82+
83+
/// Display an animated black&white GIF (300x400px)
84+
#[arg(long)]
85+
pub animated_gif: Option<String>,
8286

8387
/// Clear display RAM
8488
#[arg(long)]

inputmodule-control/src/inputmodule.rs

+40-4
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,9 @@ use std::thread;
22
use std::time::Duration;
33

44
use chrono::Local;
5+
use image::codecs::gif::GifDecoder;
56
use image::{io::Reader as ImageReader, Luma};
7+
use image::{AnimationDecoder, DynamicImage, ImageBuffer};
68
use rand::prelude::*;
79
use serialport::{SerialPort, SerialPortInfo, SerialPortType};
810

@@ -280,9 +282,12 @@ pub fn serial_commands(args: &crate::ClapCli) {
280282
if let Some(fps) = b1display_args.animation_fps {
281283
animation_fps_cmd(serialdev, fps);
282284
}
283-
if let Some(image_path) = &b1display_args.image_bw {
285+
if let Some(image_path) = &b1display_args.image {
284286
b1display_bw_image_cmd(serialdev, image_path);
285287
}
288+
if let Some(image_path) = &b1display_args.animated_gif {
289+
gif_cmd(serialdev, image_path);
290+
}
286291
if b1display_args.clear_ram {
287292
simple_cmd(serialdev, Command::ClearRam, &[0x00]);
288293
}
@@ -904,17 +909,48 @@ fn set_color_cmd(serialdev: &str, color: Color) {
904909
simple_cmd(serialdev, Command::SetColor, args);
905910
}
906911

912+
fn gif_cmd(serialdev: &str, image_path: &str) {
913+
let mut serialport = open_serialport(serialdev);
914+
915+
loop {
916+
let img = std::fs::File::open(image_path).unwrap();
917+
let gif = GifDecoder::new(img).unwrap();
918+
let frames = gif.into_frames();
919+
for (_i, frame) in frames.enumerate() {
920+
//println!("Frame {i}");
921+
let frame = frame.unwrap();
922+
//let delay = frame.delay();
923+
//println!(" Delay: {:?}", Duration::from(delay));
924+
let frame_img = frame.into_buffer();
925+
let frame_img = DynamicImage::from(frame_img);
926+
let frame_img = frame_img.resize(300, 400, image::imageops::FilterType::Gaussian);
927+
let frame_img = frame_img.into_luma8();
928+
display_img(&mut serialport, &frame_img);
929+
// Not delaying any further. Current transmission delay is big enough
930+
//thread::sleep(delay.into());
931+
}
932+
}
933+
}
934+
907935
/// Display an image in black and white
908936
/// Confirmed working with PNG and GIF.
909937
/// Must be 300x400 in size.
910938
/// Sends one 400px column in a single commands and a flush at the end
911-
fn b1display_bw_image_cmd(serialdev: &str, image_path: &str) {
939+
fn generic_img_cmd(serialdev: &str, image_path: &str) {
912940
let mut serialport = open_serialport(serialdev);
913941
let img = ImageReader::open(image_path)
914942
.unwrap()
915943
.decode()
916944
.unwrap()
917945
.to_luma8();
946+
display_img(&mut serialport, &img);
947+
}
948+
949+
fn b1display_bw_image_cmd(serialdev: &str, image_path: &str) {
950+
generic_img_cmd(serialdev, image_path);
951+
}
952+
953+
fn display_img(serialport: &mut Box<dyn SerialPort>, img: &ImageBuffer<Luma<u8>, Vec<u8>>) {
918954
let width = img.width();
919955
let height = img.height();
920956
assert!(width == 300);
@@ -957,10 +993,10 @@ fn b1display_bw_image_cmd(serialdev: &str, image_path: &str) {
957993
}
958994
}
959995

960-
simple_open_cmd(&mut serialport, Command::SetPixelColumn, &vals);
996+
simple_open_cmd(serialport, Command::SetPixelColumn, &vals);
961997
}
962998

963-
simple_open_cmd(&mut serialport, Command::FlushFramebuffer, &[]);
999+
simple_open_cmd(serialport, Command::FlushFramebuffer, &[]);
9641000
}
9651001

9661002
fn b1_display_color(serialdev: &str, black: bool) {

0 commit comments

Comments
 (0)