use egui_extras::Column;
use re_renderer::{
external::re_video::VideoLoadError, resource_managers::SourceImageDataFormat,
video::VideoFrameTexture,
};
use re_types::components::VideoTimestamp;
use re_ui::{
list_item::{self, PropertyContent},
DesignTokens, UiExt,
};
use re_video::{decode::FrameInfo, VideoData};
use re_viewer_context::UiLayout;
pub fn video_result_ui(
ui: &mut egui::Ui,
ui_layout: UiLayout,
video_result: &Result<re_renderer::video::Video, VideoLoadError>,
) {
re_tracing::profile_function!();
#[allow(clippy::match_same_arms)]
match video_result {
Ok(video) => {
if !ui_layout.is_single_line() {
re_ui::list_item::list_item_scope(ui, "video_blob_info", |ui| {
video_data_ui(ui, ui_layout, video.data());
});
}
}
Err(VideoLoadError::MimeTypeIsNotAVideo { .. }) => {
}
Err(VideoLoadError::UnrecognizedMimeType) => {
}
Err(err) => {
let error_message = format!("Failed to load video: {err}");
if ui_layout.is_single_line() {
ui.error_with_details_on_hover(error_message);
} else {
ui.error_label(error_message);
}
}
}
}
fn video_data_ui(ui: &mut egui::Ui, ui_layout: UiLayout, video_data: &VideoData) {
re_tracing::profile_function!();
ui.list_item_flat_noninteractive(PropertyContent::new("Dimensions").value_text(format!(
"{}x{}",
video_data.width(),
video_data.height()
)));
if let Some(bit_depth) = video_data.config.stsd.contents.bit_depth() {
ui.list_item_flat_noninteractive(PropertyContent::new("Bit depth").value_fn(|ui, _| {
ui.label(bit_depth.to_string());
if 8 < bit_depth {
ui.warning_label("HDR").on_hover_ui(|ui| {
ui.label("High-dynamic-range videos not yet supported by Rerun");
ui.hyperlink("https://github.com/rerun-io/rerun/issues/7594");
});
}
if video_data.is_monochrome() == Some(true) {
ui.label("(monochrome)");
}
}));
}
if let Some(subsampling_mode) = video_data.subsampling_mode() {
if video_data.is_monochrome() != Some(true) {
ui.list_item_flat_noninteractive(
PropertyContent::new("Subsampling").value_text(subsampling_mode.to_string()),
);
}
}
ui.list_item_flat_noninteractive(PropertyContent::new("Duration").value_text(format!(
"{}",
re_log_types::Duration::from(video_data.duration())
)));
ui.list_item_flat_noninteractive(
PropertyContent::new("Frame count")
.value_text(re_format::format_uint(video_data.num_samples())),
);
ui.list_item_flat_noninteractive(
PropertyContent::new("Codec").value_text(video_data.human_readable_codec_string()),
);
if ui_layout != UiLayout::Tooltip {
ui.list_item_collapsible_noninteractive_label("MP4 tracks", false, |ui| {
for (track_id, track_kind) in &video_data.mp4_tracks {
let track_kind_string = match track_kind {
Some(re_video::TrackKind::Audio) => "audio",
Some(re_video::TrackKind::Subtitle) => "subtitle",
Some(re_video::TrackKind::Video) => "video",
None => "unknown",
};
ui.list_item_flat_noninteractive(
PropertyContent::new(format!("Track {track_id}")).value_text(track_kind_string),
);
}
});
ui.list_item_collapsible_noninteractive_label("More video statistics", false, |ui| {
ui.list_item_flat_noninteractive(
PropertyContent::new("Number of GOPs")
.value_text(video_data.gops.len().to_string()),
)
.on_hover_text("The total number of Group of Pictures (GOPs) in the video.");
let re_video::SamplesStatistics {dts_always_equal_pts, has_sample_highest_pts_so_far: _} = &video_data.samples_statistics;
ui.list_item_flat_noninteractive(
PropertyContent::new("All PTS equal DTS").value_bool(*dts_always_equal_pts)
).on_hover_text("Whether all decode timestamps are equal to presentation timestamps. If true, the video typically has no B-frames.");
});
ui.list_item_collapsible_noninteractive_label("Video samples", false, |ui| {
egui::Resize::default()
.with_stroke(true)
.resizable([false, true])
.max_height(611.0) .show(ui, |ui| {
samples_table_ui(ui, video_data);
});
});
}
}
fn samples_table_ui(ui: &mut egui::Ui, video_data: &VideoData) {
re_tracing::profile_function!();
egui_extras::TableBuilder::new(ui)
.auto_shrink([false, true])
.vscroll(true)
.max_scroll_height(611.0) .columns(Column::auto(), 8)
.cell_layout(egui::Layout::left_to_right(egui::Align::Center))
.header(DesignTokens::table_header_height(), |mut header| {
DesignTokens::setup_table_header(&mut header);
header.col(|ui| {
ui.strong("Sample");
});
header.col(|ui| {
ui.strong("Frame");
});
header.col(|ui| {
ui.strong("GOP");
});
header.col(|ui| {
ui.strong("Sync");
});
header.col(|ui| {
ui.strong("DTS").on_hover_text("Decode timestamp");
});
header.col(|ui| {
ui.strong("PTS").on_hover_text("Presentation timestamp");
});
header.col(|ui| {
ui.strong("Duration");
});
header.col(|ui| {
ui.strong("Size");
});
})
.body(|mut body| {
DesignTokens::setup_table_body(&mut body);
body.rows(
DesignTokens::table_line_height(),
video_data.samples.len(),
|mut row| {
let sample_idx = row.index();
let sample = &video_data.samples[sample_idx];
let re_video::Sample {
is_sync,
sample_idx: sample_idx_in_sample,
frame_nr,
decode_timestamp,
presentation_timestamp,
duration,
byte_offset: _,
byte_length,
} = *sample;
debug_assert_eq!(sample_idx, sample_idx_in_sample);
row.col(|ui| {
ui.monospace(re_format::format_uint(sample_idx));
});
row.col(|ui| {
ui.monospace(re_format::format_uint(frame_nr));
});
row.col(|ui| {
if let Some(gop_index) = video_data
.gop_index_containing_presentation_timestamp(presentation_timestamp)
{
ui.monospace(re_format::format_uint(gop_index));
}
});
row.col(|ui| {
if is_sync {
ui.label("sync");
}
});
row.col(|ui| {
timestamp_ui(ui, video_data, decode_timestamp);
});
row.col(|ui| {
timestamp_ui(ui, video_data, presentation_timestamp);
});
row.col(|ui| {
ui.monospace(
re_log_types::Duration::from(duration.duration(video_data.timescale))
.to_string(),
);
});
row.col(|ui| {
ui.monospace(re_format::format_bytes(byte_length as _));
});
},
);
});
}
fn timestamp_ui(ui: &mut egui::Ui, video_data: &VideoData, timestamp: re_video::Time) {
ui.monospace(re_format::format_int(timestamp.0))
.on_hover_ui(|ui| {
ui.monospace(re_format::format_timestamp_seconds(
timestamp.into_secs(video_data.timescale),
));
});
}
pub fn show_decoded_frame_info(
render_ctx: Option<&re_renderer::RenderContext>,
ui: &mut egui::Ui,
ui_layout: UiLayout,
video: &re_renderer::video::Video,
video_timestamp: Option<VideoTimestamp>,
blob: &re_types::datatypes::Blob,
) {
let Some(render_ctx) = render_ctx else {
return;
};
let timestamp_in_seconds = if let Some(video_timestamp) = video_timestamp {
video_timestamp.as_seconds()
} else {
ui.ctx().request_repaint(); ui.input(|i| i.time) % video.data().duration().as_secs_f64()
};
let player_stream_id =
re_renderer::video::VideoPlayerStreamId(ui.id().with("video_player").value());
match video.frame_at(
render_ctx,
player_stream_id,
timestamp_in_seconds,
blob.as_slice(),
) {
Ok(VideoFrameTexture {
texture,
is_pending,
show_spinner,
frame_info,
source_pixel_format,
}) => {
if let Some(frame_info) = frame_info {
re_ui::list_item::list_item_scope(ui, "decoded_frame_ui", |ui| {
let id = ui.id().with("decoded_frame_collapsible");
let default_open = false;
let label = if let Some(frame_nr) = frame_info.frame_nr {
format!("Decoded frame #{}", re_format::format_uint(frame_nr))
} else {
"Current decoded frame".to_owned()
};
ui.list_item()
.interactive(false)
.show_hierarchical_with_children(
ui,
id,
default_open,
list_item::LabelContent::new(label),
|ui| {
list_item::list_item_scope(ui, id, |ui| {
frame_info_ui(ui, &frame_info, video.data());
source_image_data_format_ui(ui, &source_pixel_format);
});
},
)
});
}
let response = crate::image::texture_preview_ui(
render_ctx,
ui,
ui_layout,
"video_preview",
re_renderer::renderer::ColormappedTexture::from_unorm_rgba(texture),
);
if is_pending {
ui.ctx().request_repaint(); }
if show_spinner {
let smaller_rect = egui::Rect::from_center_size(
response.rect.center(),
0.75 * response.rect.size(),
);
egui::Spinner::new().paint_at(ui, smaller_rect);
}
}
Err(err) => {
ui.error_label(err.to_string());
#[cfg(not(target_arch = "wasm32"))]
if let re_renderer::video::VideoPlayerError::Decoding(
re_video::decode::Error::Ffmpeg(err),
) = &err
{
match err.as_ref() {
re_video::decode::FFmpegError::UnsupportedFFmpegVersion { .. }
| re_video::decode::FFmpegError::FailedToDetermineFFmpegVersion(_)
| re_video::decode::FFmpegError::FFmpegNotInstalled => {
if let Some(download_url) = re_video::decode::ffmpeg_download_url() {
ui.markdown_ui(&format!("You can download a build of `FFmpeg` [here]({download_url}). For Rerun to be able to use it, its binaries need to be reachable from `PATH`."));
}
}
_ => {}
}
}
}
}
}
fn frame_info_ui(ui: &mut egui::Ui, frame_info: &FrameInfo, video_data: &re_video::VideoData) {
let FrameInfo {
is_sync,
sample_idx,
frame_nr,
presentation_timestamp,
duration,
latest_decode_timestamp,
} = *frame_info;
if let Some(is_sync) = is_sync {
ui.list_item_flat_noninteractive(PropertyContent::new("Sync").value_bool(is_sync))
.on_hover_text(
"The start of a new GOP (Group of Frames)?\n\
If true, it likely means the frame is a keyframe.",
);
}
let presentation_time_range = presentation_timestamp..presentation_timestamp + duration;
ui.list_item_flat_noninteractive(PropertyContent::new("Time range").value_text(format!(
"{} - {}",
re_format::format_timestamp_seconds(presentation_time_range.start.into_secs(
video_data.timescale
)),
re_format::format_timestamp_seconds(presentation_time_range.end.into_secs(
video_data.timescale
)),
)))
.on_hover_text("Time range in which this frame is valid.");
fn value_fn_for_time(
time: re_video::Time,
video_data: &re_video::VideoData,
) -> impl FnOnce(&mut egui::Ui, egui::style::WidgetVisuals) + '_ {
move |ui, _| {
timestamp_ui(ui, video_data, time);
}
}
if let Some(sample_idx) = sample_idx {
ui.list_item_flat_noninteractive(PropertyContent::new("Sample").value_fn(move |ui, _| {
ui.monospace(re_format::format_uint(sample_idx));
}))
.on_hover_text(
"The sample number of this frame in the video. In MP4, one sample is one frame, but not necessareily in the same order!",
);
}
if let Some(frame_nr) = frame_nr {
ui.list_item_flat_noninteractive(PropertyContent::new("Frame").value_fn(move |ui, _| {
ui.monospace(re_format::format_uint(frame_nr));
}))
.on_hover_text("The frame number, as ordered by presentation time");
}
if let Some(dts) = latest_decode_timestamp {
ui.list_item_flat_noninteractive(
PropertyContent::new("DTS").value_fn(value_fn_for_time(dts, video_data)),
)
.on_hover_text("Raw decode timestamp prior to applying the timescale.\n\
If a frame is made up of multiple chunks, this is the last decode timestamp that was needed to decode the frame.");
}
ui.list_item_flat_noninteractive(
PropertyContent::new("PTS").value_fn(value_fn_for_time(presentation_timestamp, video_data)),
)
.on_hover_text("Raw presentation timestamp prior to applying the timescale.\n\
This specifies the time at which the frame should be shown relative to the start of a video stream.");
#[cfg(debug_assertions)]
{
if let Some(has_sample_highest_pts_so_far) = video_data
.samples_statistics
.has_sample_highest_pts_so_far
.as_ref()
{
if let Some(sample_idx) =
video_data.latest_sample_index_at_presentation_timestamp(presentation_timestamp)
{
ui.list_item_flat_noninteractive(
PropertyContent::new("Highest PTS so far").value_bool(has_sample_highest_pts_so_far[sample_idx])
).on_hover_text("Whether the presentation timestamp (PTS) at the this frame is the highest encountered so far. If false there are lower PTS values prior in the list.");
}
}
}
if let Some(gop_index) =
video_data.gop_index_containing_presentation_timestamp(presentation_timestamp)
{
ui.list_item_flat_noninteractive(
PropertyContent::new("GOP index").value_text(gop_index.to_string()),
)
.on_hover_text("The index of the group of picture (GOP) that this sample belongs to.");
if let Some(gop) = video_data.gops.get(gop_index) {
let first_sample = video_data.samples.get(gop.sample_range.start as usize);
let last_sample = video_data
.samples
.get((gop.sample_range.end as usize).saturating_sub(1));
if let (Some(first_sample), Some(last_sample)) = (first_sample, last_sample) {
ui.list_item_flat_noninteractive(PropertyContent::new("GOP DTS range").value_text(
format!("{} - {}", re_format::format_int(first_sample.decode_timestamp.0), re_format::format_int(last_sample.decode_timestamp.0))
))
.on_hover_text(
"The range of decode timestamps in the currently active group of picture (GOP).",
);
} else {
ui.error_label("GOP has invalid sample range"); }
} else {
ui.error_label("Invalid GOP index"); }
}
}
fn source_image_data_format_ui(ui: &mut egui::Ui, format: &SourceImageDataFormat) {
let label = "Decoder output format";
match format {
SourceImageDataFormat::WgpuCompatible(format) => {
ui.list_item_flat_noninteractive(PropertyContent::new(label).value_text(format!("{format:?}")))
.on_hover_text("Pixel format as returned from the decoder.\n\
Decoders may do arbitrary post processing, so this is not necessarily the format that is actually encoded in the video data!"
);
}
SourceImageDataFormat::Yuv {
layout,
range,
coefficients,
} => {
let default_open = true;
ui.list_item_collapsible_noninteractive_label(label, default_open, |ui| {
ui.list_item_flat_noninteractive(
PropertyContent::new("Data layout").value_text(layout.to_string()),
)
.on_hover_text("Subsampling ratio & layout of the pixel data.");
ui.list_item_flat_noninteractive(
PropertyContent::new("Color range").value_text(range.to_string()),
)
.on_hover_text("Valid range of the pixel data values.");
ui.list_item_flat_noninteractive(
PropertyContent::new("Yuv Coefficients").value_text(coefficients.to_string()),
)
.on_hover_text("Matrix coefficients used to convert the pixel data to RGB.");
});
}
};
}