use re_types::{
archetypes::EncodedImage,
components::{Blob, DrawOrder, MediaType, Opacity},
Component as _,
};
use re_view::{diff_component_filter, HybridResults};
use re_viewer_context::{
ApplicableEntities, IdentifiedViewSystem, ImageDecodeCache, QueryContext,
TypedComponentFallbackProvider, ViewContext, ViewContextCollection, ViewQuery,
ViewSystemExecutionError, VisualizableEntities, VisualizableFilterContext,
VisualizerAdditionalApplicabilityFilter, VisualizerQueryInfo, VisualizerSystem,
};
use crate::{
contexts::SpatialSceneEntityContext,
view_kind::SpatialViewKind,
visualizers::{filter_visualizable_2d_entities, textured_rect_from_image},
PickableRectSourceData, PickableTexturedRect,
};
use super::{entity_iterator::process_archetype, SpatialViewVisualizerData};
pub struct EncodedImageVisualizer {
pub data: SpatialViewVisualizerData,
}
impl Default for EncodedImageVisualizer {
fn default() -> Self {
Self {
data: SpatialViewVisualizerData::new(Some(SpatialViewKind::TwoD)),
}
}
}
impl IdentifiedViewSystem for EncodedImageVisualizer {
fn identifier() -> re_viewer_context::ViewSystemIdentifier {
"EncodedImage".into()
}
}
struct ImageMediaTypeFilter;
impl VisualizerAdditionalApplicabilityFilter for ImageMediaTypeFilter {
fn update_applicability(&mut self, event: &re_chunk_store::ChunkStoreEvent) -> bool {
diff_component_filter(event, |media_type: &re_types::components::MediaType| {
media_type.is_image()
}) || diff_component_filter(event, |image: &re_types::components::Blob| {
MediaType::guess_from_data(&image.0).map_or(false, |media| media.is_image())
})
}
}
impl VisualizerSystem for EncodedImageVisualizer {
fn visualizer_query_info(&self) -> VisualizerQueryInfo {
VisualizerQueryInfo::from_archetype::<EncodedImage>()
}
fn applicability_filter(
&self,
) -> Option<Box<dyn re_viewer_context::VisualizerAdditionalApplicabilityFilter>> {
Some(Box::new(ImageMediaTypeFilter))
}
fn filter_visualizable_entities(
&self,
entities: ApplicableEntities,
context: &dyn VisualizableFilterContext,
) -> VisualizableEntities {
re_tracing::profile_function!();
filter_visualizable_2d_entities(entities, context)
}
fn execute(
&mut self,
ctx: &ViewContext<'_>,
view_query: &ViewQuery<'_>,
context_systems: &ViewContextCollection,
) -> Result<Vec<re_renderer::QueueableDrawData>, ViewSystemExecutionError> {
let Some(render_ctx) = ctx.viewer_ctx.render_ctx else {
return Err(ViewSystemExecutionError::NoRenderContextError);
};
process_archetype::<Self, EncodedImage, _>(
ctx,
view_query,
context_systems,
|ctx, spatial_ctx, results| {
self.process_encoded_image(ctx, results, spatial_ctx);
Ok(())
},
)?;
self.data.pickable_rects.sort_by_key(|image| {
(
image.textured_rect.options.depth_offset,
egui::emath::OrderedFloat(image.textured_rect.options.multiplicative_tint.a()),
)
});
Ok(vec![PickableTexturedRect::to_draw_data(
render_ctx,
&self.data.pickable_rects,
)?])
}
fn data(&self) -> Option<&dyn std::any::Any> {
Some(self.data.as_any())
}
fn as_any(&self) -> &dyn std::any::Any {
self
}
fn fallback_provider(&self) -> &dyn re_viewer_context::ComponentFallbackProvider {
self
}
}
impl EncodedImageVisualizer {
fn process_encoded_image(
&mut self,
ctx: &QueryContext<'_>,
results: &HybridResults<'_>,
spatial_ctx: &SpatialSceneEntityContext<'_>,
) {
use super::entity_iterator::iter_buffer;
use re_view::RangeResultsExt as _;
let entity_path = ctx.target_entity_path;
let Some(all_blob_chunks) = results.get_required_chunks(&Blob::name()) else {
return;
};
let timeline = ctx.query.timeline();
let all_blobs_indexed = iter_buffer::<u8>(&all_blob_chunks, timeline, Blob::name());
let all_media_types = results.iter_as(timeline, MediaType::name());
let all_opacities = results.iter_as(timeline, Opacity::name());
for ((_time, tensor_data_row_id), blobs, media_types, opacities) in re_query::range_zip_1x2(
all_blobs_indexed,
all_media_types.string(),
all_opacities.primitive::<f32>(),
) {
let Some(blob) = blobs.first() else {
continue;
};
let media_type = media_types
.and_then(|media_types| media_types.first().cloned())
.map(|media_type| MediaType(media_type.into()));
let image = ctx.viewer_ctx.cache.entry(|c: &mut ImageDecodeCache| {
c.entry(tensor_data_row_id, blob, media_type.as_ref())
});
let image = match image {
Ok(image) => image,
Err(err) => {
re_log::warn_once!(
"Failed to decode EncodedImage at path {entity_path}: {err}"
);
continue;
}
};
let opacity: Option<&Opacity> =
opacities.and_then(|opacity| opacity.first().map(bytemuck::cast_ref));
let opacity = opacity.copied().unwrap_or_else(|| self.fallback_for(ctx));
let multiplicative_tint =
re_renderer::Rgba::from_white_alpha(opacity.0.clamp(0.0, 1.0));
let colormap = None;
if let Some(textured_rect) = textured_rect_from_image(
ctx.viewer_ctx,
entity_path,
spatial_ctx,
&image,
colormap,
multiplicative_tint,
"EncodedImage",
&mut self.data,
) {
self.data.pickable_rects.push(PickableTexturedRect {
ent_path: entity_path.clone(),
textured_rect,
source_data: PickableRectSourceData::Image {
image,
depth_meter: None,
},
});
}
}
}
}
impl TypedComponentFallbackProvider<Opacity> for EncodedImageVisualizer {
fn fallback_for(&self, _ctx: &re_viewer_context::QueryContext<'_>) -> Opacity {
1.0.into()
}
}
impl TypedComponentFallbackProvider<DrawOrder> for EncodedImageVisualizer {
fn fallback_for(&self, _ctx: &QueryContext<'_>) -> DrawOrder {
DrawOrder::DEFAULT_IMAGE
}
}
re_viewer_context::impl_component_fallback_provider!(EncodedImageVisualizer => [DrawOrder, Opacity]);