@@ -2,7 +2,9 @@ use std::thread;
2
2
use std:: time:: Duration ;
3
3
4
4
use chrono:: Local ;
5
+ use image:: codecs:: gif:: GifDecoder ;
5
6
use image:: { io:: Reader as ImageReader , Luma } ;
7
+ use image:: { AnimationDecoder , DynamicImage , ImageBuffer } ;
6
8
use rand:: prelude:: * ;
7
9
use serialport:: { SerialPort , SerialPortInfo , SerialPortType } ;
8
10
@@ -280,9 +282,12 @@ pub fn serial_commands(args: &crate::ClapCli) {
280
282
if let Some ( fps) = b1display_args. animation_fps {
281
283
animation_fps_cmd ( serialdev, fps) ;
282
284
}
283
- if let Some ( image_path) = & b1display_args. image_bw {
285
+ if let Some ( image_path) = & b1display_args. image {
284
286
b1display_bw_image_cmd ( serialdev, image_path) ;
285
287
}
288
+ if let Some ( image_path) = & b1display_args. animated_gif {
289
+ gif_cmd ( serialdev, image_path) ;
290
+ }
286
291
if b1display_args. clear_ram {
287
292
simple_cmd ( serialdev, Command :: ClearRam , & [ 0x00 ] ) ;
288
293
}
@@ -904,17 +909,48 @@ fn set_color_cmd(serialdev: &str, color: Color) {
904
909
simple_cmd ( serialdev, Command :: SetColor , args) ;
905
910
}
906
911
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
+
907
935
/// Display an image in black and white
908
936
/// Confirmed working with PNG and GIF.
909
937
/// Must be 300x400 in size.
910
938
/// 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 ) {
912
940
let mut serialport = open_serialport ( serialdev) ;
913
941
let img = ImageReader :: open ( image_path)
914
942
. unwrap ( )
915
943
. decode ( )
916
944
. unwrap ( )
917
945
. 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 > > ) {
918
954
let width = img. width ( ) ;
919
955
let height = img. height ( ) ;
920
956
assert ! ( width == 300 ) ;
@@ -957,10 +993,10 @@ fn b1display_bw_image_cmd(serialdev: &str, image_path: &str) {
957
993
}
958
994
}
959
995
960
- simple_open_cmd ( & mut serialport, Command :: SetPixelColumn , & vals) ;
996
+ simple_open_cmd ( serialport, Command :: SetPixelColumn , & vals) ;
961
997
}
962
998
963
- simple_open_cmd ( & mut serialport, Command :: FlushFramebuffer , & [ ] ) ;
999
+ simple_open_cmd ( serialport, Command :: FlushFramebuffer , & [ ] ) ;
964
1000
}
965
1001
966
1002
fn b1_display_color ( serialdev : & str , black : bool ) {
0 commit comments