diff --git a/.accepted_words.txt b/.accepted_words.txt index d34ad96a..2e4aab3d 100644 --- a/.accepted_words.txt +++ b/.accepted_words.txt @@ -36,6 +36,7 @@ JSON ld LD libfontconfig +libsdl microsoft minimalistic mosquitto @@ -50,6 +51,8 @@ repo Repo rustup sdk +sdl +SDL snapd sudo timothee diff --git a/.github/workflows/rust-ci.yml b/.github/workflows/rust-ci.yml index 6e7ea40e..08e6c9b6 100644 --- a/.github/workflows/rust-ci.yml +++ b/.github/workflows/rust-ci.yml @@ -19,8 +19,8 @@ jobs: uses: actions/checkout@v3 with: submodules: recursive - - name: Install protobuf-compiler - run: sudo apt-get install -y protobuf-compiler + - name: Install packages + run: sudo apt-get install -y protobuf-compiler libsdl2-dev - name: Install .NET 7.0 uses: actions/setup-dotnet@v3 with: @@ -53,8 +53,8 @@ jobs: uses: actions/checkout@v3 with: submodules: recursive - - name: Install protobuf-compiler - run: sudo apt-get install -y protobuf-compiler + - name: Install packages + run: sudo apt-get install -y protobuf-compiler libsdl2-dev - name: Install .NET 7.0 uses: actions/setup-dotnet@v3 with: diff --git a/Cargo.toml b/Cargo.toml index 43136353..0b1755ec 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -31,7 +31,7 @@ members = [ "samples/mixed", "samples/property", "samples/seat_massager", - # "samples/streaming", + "samples/streaming", ] [workspace.dependencies] @@ -56,10 +56,10 @@ parking_lot = "0.12.1" prost = "0.12" prost-types = "0.12" regex = " 1.9.3" +sdl2 = "0.35.2" serde = "1.0.160" serde_derive = "1.0.163" serde_json = "^1.0" -show-image = "0.13.1" strum = "0.25" strum_macros = "0.25.1" tokio = "1.29.1" diff --git a/README.md b/README.md index 81656c9f..c25016ca 100644 --- a/README.md +++ b/README.md @@ -6,7 +6,7 @@ - [Install gcc](#install-gcc) - [Install Rust](#install-rust) - [Install Protobuf Compiler](#install-protobuf-compiler) - - [Install fontconfig-dev library](#install-fontconfig-dev-library) + - [Install SDL2 library](#install-sdl2-library) - [Install MQTT Broker](#install-mqtt-broker) - [Cloning the Repo](#cloning-the-repo) - [Developer Notes](#developer-notes) @@ -66,12 +66,12 @@ You will need to install the Protobuf Compiler. This can be done by executing: sudo apt install -y protobuf-compiler ``` -### Install fontconfig-dev library +### Install SDL2 library -You will need to install the fontconfig-dev library. This can be done by executing: +You will need to install the libsdl2-dev library. This can be done by executing: ```shell -sudo apt install -y libfontconfig-dev +sudo apt install -y libsdl2-dev ``` ### Install MQTT Broker diff --git a/samples/common/Cargo.toml b/samples/common/Cargo.toml index 32bcda56..ac2b9ac2 100644 --- a/samples/common/Cargo.toml +++ b/samples/common/Cargo.toml @@ -10,8 +10,10 @@ license = "MIT" [dependencies] config = { workspace = true } +image = { workspace = true } log = { workspace = true } samples-protobuf-data-access = { path = "../protobuf_data_access" } +sdl2 = { workspace = true } serde = { workspace = true, features = ["derive"] } serde_derive = { workspace = true } tokio = { workspace = true } diff --git a/samples/common/src/image_rendering.rs b/samples/common/src/image_rendering.rs new file mode 100644 index 00000000..4ad32d59 --- /dev/null +++ b/samples/common/src/image_rendering.rs @@ -0,0 +1,105 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT license. +// SPDX-License-Identifier: MIT + +use image::{imageops::FilterType, DynamicImage}; +use sdl2::pixels::{Color, PixelFormatEnum}; +use sdl2::rect::Rect; +use sdl2::render::WindowCanvas; +use sdl2::surface::Surface; +use sdl2::Sdl; + +// This module is based on SDL2 (Simple DirectMedia Layer). It complements the sdl2 crate by providing +// methods that make it easier to render images. + +/// Create a canvas with an enclosing window. +/// +/// # Arguments +/// * `sdl_context` - The SDL context. +/// * `window_title` - The window's title. +/// * `window_width` - The window's width. +/// * `window_height` - The window's height. +pub fn create_canvas( + sdl_context: &mut Sdl, + window_title: &str, + window_width: u32, + window_height: u32, +) -> Result { + let video_subsystem = sdl_context.video()?; + + let window = video_subsystem + .window(window_title, window_width, window_height) + .position_centered() + .allow_highdpi() + .build() + .map_err(|err| format!("{}", err))?; + + let mut canvas = window.into_canvas().build().map_err(|err| format!("{}", err))?; + + // Set the background color to black. + canvas.set_draw_color(Color::RGB(0, 0, 0)); + + Ok(canvas) +} + +/// Resize an image to fit inside a canvas. +/// +/// # Arguments +/// * `image` - The image that needs to be resized. +/// * `canvas` - The canvas that it needs to fit in. +pub fn resize_image_to_fit_in_canvas( + image: DynamicImage, + canvas: &WindowCanvas, +) -> Result { + let (window_width, window_height): (u32, u32) = canvas.output_size()?; + + let width_scale = window_width as f32 / image.width() as f32; + let height_scale = window_height as f32 / image.height() as f32; + let scale: f32 = height_scale.min(width_scale); + + let resized_image_width = (scale * image.width() as f32) as u32; + let resized_image_height = (scale * image.height() as f32) as u32; + + Ok(image.resize(resized_image_width, resized_image_height, FilterType::Triangle)) +} + +/// Render an image to a canvas. +/// +/// # Arguments +/// * `image` - The image that we want to render. +/// * `canvas` - The canvas that will render the image. +pub fn render_image_to_canvas( + image: &DynamicImage, + canvas: &mut WindowCanvas, +) -> Result<(), String> { + // Prepare the image for copying it to a surface. + let rgb_image = image.to_rgb8(); + let mut image_buffer = rgb_image.into_raw(); + + let image_width = image.width(); + let image_height = image.height(); + // The pitch is the width of the texture times the size of a single pixel in bytes. + // Since we are using 24 bit pixels (RGB24), we need to mutiple the width by 3. + let image_pitch: u32 = image_width * 3; + + let surface = Surface::from_data( + &mut image_buffer, + image_width, + image_height, + image_pitch, + PixelFormatEnum::RGB24, + ) + .map_err(|err| err.to_string())?; + + let texture_creator = canvas.texture_creator(); + let texture = texture_creator + .create_texture_from_surface(surface) + .map_err(|err| format!("Failed to create texture from surface due to: {err}"))?; + + // Render the image. + canvas.clear(); + canvas.copy(&texture, None, Rect::new(0, 0, image_width, image_height))?; + canvas.present(); + + Ok(()) +} diff --git a/samples/common/src/lib.rs b/samples/common/src/lib.rs index 27524aae..43ca95c0 100644 --- a/samples/common/src/lib.rs +++ b/samples/common/src/lib.rs @@ -4,5 +4,6 @@ pub mod constants; pub mod consumer_config; +pub mod image_rendering; pub mod provider_config; pub mod utils; diff --git a/samples/streaming/Cargo.toml b/samples/streaming/Cargo.toml index 70a201a5..1a1b8b41 100644 --- a/samples/streaming/Cargo.toml +++ b/samples/streaming/Cargo.toml @@ -20,10 +20,10 @@ parking_lot = { workspace = true } prost = { workspace = true } samples-common = { path = "../common" } samples-protobuf-data-access = { path = "../protobuf_data_access" } +sdl2 = { workspace = true } serde = { workspace = true, features = ["derive"] } serde_derive = { workspace = true } serde_json = { workspace = true } -show-image = { workspace = true } tokio = { workspace = true, features = ["macros", "rt-multi-thread", "signal", "sync"] } tokio-stream = { workspace = true } tonic = { workspace = true } diff --git a/samples/streaming/consumer/src/main.rs b/samples/streaming/consumer/src/main.rs index af1b12e5..9eb1bfbf 100644 --- a/samples/streaming/consumer/src/main.rs +++ b/samples/streaming/consumer/src/main.rs @@ -6,16 +6,15 @@ mod streaming_consumer_config; use digital_twin_model::sdv_v1 as sdv; use env_logger::{Builder, Target}; - -use image::io::Reader as ImageReader; +use image::{DynamicImage, io::Reader as ImageReader}; use log::{info, LevelFilter, warn}; use samples_common::constants::{digital_twin_operation, digital_twin_protocol}; +use samples_common::image_rendering::{create_canvas, render_image_to_canvas, resize_image_to_fit_in_canvas}; use samples_common::utils::{ discover_digital_twin_provider_using_ibeji, retrieve_invehicle_digital_twin_uri, }; use samples_protobuf_data_access::sample_grpc::v1::digital_twin_provider::StreamRequest; use samples_protobuf_data_access::sample_grpc::v1::digital_twin_provider::digital_twin_provider_client::DigitalTwinProviderClient; -use show_image::{ImageView, ImageInfo, create_window, WindowProxy}; use std::error::Error; use std::io::Cursor; use tokio_stream::StreamExt; @@ -25,14 +24,17 @@ use tonic::transport::Channel; /// /// # Arguments /// * `client` - The client connection to the service that will transfer the stream. +/// * `entity_id` - The entity id that is to be streamed. /// * `number_of_images` - The number of images that we will stream. -/// * `window` - The window where the streamed images will be shown. async fn stream_images( client: &mut DigitalTwinProviderClient, entity_id: &str, number_of_images: usize, - window: &mut WindowProxy, ) -> Result<(), Box> { + let mut sdl_context = sdl2::init()?; + + let mut canvas = create_canvas(&mut sdl_context, "Streamed Image", 800, 500)?; + let stream = client.stream(StreamRequest { entity_id: entity_id.to_string() }).await?.into_inner(); @@ -46,11 +48,20 @@ async fn stream_images( } let media_content = opt_media.unwrap().media_content; let image_reader = ImageReader::new(Cursor::new(media_content)).with_guessed_format()?; - let image = image_reader.decode()?; - let image_data = image.as_bytes().to_vec(); - let image_view = - ImageView::new(ImageInfo::rgb8(image.width(), image.height()), &image_data); - window.set_image("some file", image_view)?; + let image: DynamicImage = image_reader.decode()?; + + let resized_image = match resize_image_to_fit_in_canvas(image, &canvas) { + Ok(value) => value, + Err(err) => { + warn!("Failed to resize the image due to: {err}"); + // Skip this image. + continue; + } + }; + + if let Err(err) = render_image_to_canvas(&resized_image, &mut canvas) { + warn!("Failed to render the image due to: {err}"); + } } // The stream is dropped when we exit the function and the disconnect info is sent to the server. @@ -59,7 +70,6 @@ async fn stream_images( } #[tokio::main] -#[show_image::main] async fn main() -> Result<(), Box> { // Setup logging. Builder::new().filter(None, LevelFilter::Info).target(Target::Stdout).init(); @@ -87,17 +97,8 @@ async fn main() -> Result<(), Box> { let provider_uri = provider_endpoint_info.uri; info!("The provider URI for the Cabin Camera Feed property's provider is {provider_uri}"); - // Create a window with default options and display the image. - let mut window = create_window("image", Default::default())?; - let mut client = DigitalTwinProviderClient::connect(provider_uri.clone()).await.unwrap(); - stream_images( - &mut client, - sdv::camera::feed::ID, - settings.number_of_images.into(), - &mut window, - ) - .await?; + stream_images(&mut client, sdv::camera::feed::ID, settings.number_of_images.into()).await?; info!("The Consumer has completed.");