2020-07-26 22:06:01 +00:00
|
|
|
use super::{Texture, TextureFormat};
|
|
|
|
use anyhow::Result;
|
2020-10-18 20:48:15 +00:00
|
|
|
use bevy_asset::{AssetLoader, LoadContext, LoadedAsset};
|
2020-07-26 22:06:01 +00:00
|
|
|
use bevy_math::Vec2;
|
|
|
|
|
2020-08-16 03:27:41 +00:00
|
|
|
/// Loads HDR textures as Texture assets
|
2020-07-26 22:06:01 +00:00
|
|
|
#[derive(Clone, Default)]
|
|
|
|
pub struct HdrTextureLoader;
|
|
|
|
|
2020-10-18 20:48:15 +00:00
|
|
|
impl AssetLoader for HdrTextureLoader {
|
|
|
|
fn load(&self, bytes: &[u8], load_context: &mut LoadContext) -> Result<()> {
|
2020-07-26 22:06:01 +00:00
|
|
|
let format = TextureFormat::Rgba32Float;
|
|
|
|
debug_assert_eq!(
|
|
|
|
format.pixel_size(),
|
2020-08-16 07:30:04 +00:00
|
|
|
4 * 4,
|
2020-07-26 22:06:01 +00:00
|
|
|
"Format should have 32bit x 4 size"
|
|
|
|
);
|
|
|
|
|
2020-10-18 20:48:15 +00:00
|
|
|
let decoder = image::hdr::HdrDecoder::new(bytes)?;
|
2020-07-26 22:06:01 +00:00
|
|
|
let info = decoder.metadata();
|
|
|
|
let rgb_data = decoder.read_image_hdr()?;
|
|
|
|
let mut rgba_data = Vec::with_capacity(rgb_data.len() * format.pixel_size());
|
|
|
|
|
|
|
|
for rgb in rgb_data {
|
|
|
|
let alpha = 1.0f32;
|
|
|
|
|
|
|
|
rgba_data.extend_from_slice(&rgb.0[0].to_ne_bytes());
|
|
|
|
rgba_data.extend_from_slice(&rgb.0[1].to_ne_bytes());
|
|
|
|
rgba_data.extend_from_slice(&rgb.0[2].to_ne_bytes());
|
|
|
|
rgba_data.extend_from_slice(&alpha.to_ne_bytes());
|
|
|
|
}
|
|
|
|
|
2020-10-18 20:48:15 +00:00
|
|
|
let texture = Texture::new(
|
2020-07-26 22:06:01 +00:00
|
|
|
Vec2::new(info.width as f32, info.height as f32),
|
|
|
|
rgba_data,
|
|
|
|
format,
|
2020-10-18 20:48:15 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
load_context.set_default_asset(LoadedAsset::new(texture));
|
|
|
|
Ok(())
|
2020-07-26 22:06:01 +00:00
|
|
|
}
|
2020-07-28 21:24:03 +00:00
|
|
|
|
2020-07-26 22:06:01 +00:00
|
|
|
fn extensions(&self) -> &[&str] {
|
|
|
|
static EXTENSIONS: &[&str] = &["hdr"];
|
|
|
|
EXTENSIONS
|
|
|
|
}
|
|
|
|
}
|