1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
use std::sync::Arc;
use crate::core::api::BsdfState;
use crate::core::interaction::SurfaceInteraction;
use crate::core::material::{Material, TransportMode};
use crate::core::paramset::TextureParams;
use crate::core::pbrt::{Float, Spectrum};
use crate::core::reflection::{Bsdf, Bxdf, FourierBSDF, FourierBSDFTable};
use crate::core::texture::Texture;
pub struct FourierMaterial {
pub bsdf_table: Arc<FourierBSDFTable>,
pub bump_map: Option<Arc<dyn Texture<Float> + Sync + Send>>,
}
impl FourierMaterial {
pub fn new(
bsdf_table: Arc<FourierBSDFTable>,
bump_map: Option<Arc<dyn Texture<Float> + Sync + Send>>,
) -> Self {
FourierMaterial {
bump_map,
bsdf_table,
}
}
pub fn create(mp: &mut TextureParams, bsdf_state: &mut BsdfState) -> Arc<Material> {
let bump_map: Option<Arc<dyn Texture<Float> + Send + Sync>> =
mp.get_float_texture_or_null("bumpmap");
let bsdffile: String = mp.find_filename("bsdffile", String::new());
if let Some(bsdf_table) = bsdf_state.loaded_bsdfs.get(&bsdffile) {
Arc::new(Material::Fourier(Box::new(FourierMaterial::new(
bsdf_table.clone(),
bump_map,
))))
} else {
let mut bsdf_table: FourierBSDFTable = FourierBSDFTable::default();
println!(
"reading {:?} returns {}",
bsdffile,
bsdf_table.read(&bsdffile)
);
let bsdf_table_arc: Arc<FourierBSDFTable> = Arc::new(bsdf_table);
Arc::new(Material::Fourier(Box::new(FourierMaterial::new(
bsdf_table_arc,
bump_map,
))))
}
}
pub fn compute_scattering_functions(
&self,
si: &mut SurfaceInteraction,
mode: TransportMode,
_allow_multiple_lobes: bool,
_material: Option<Arc<Material>>,
scale_opt: Option<Spectrum>,
) {
let mut use_scale: bool = false;
let mut sc: Spectrum = Spectrum::default();
if let Some(scale) = scale_opt {
use_scale = true;
sc = scale;
}
if let Some(ref bump) = self.bump_map {
Material::bump(bump, si);
}
si.bsdf = Some(Bsdf::new(si, 1.0));
if let Some(bsdf) = &mut si.bsdf {
if use_scale {
bsdf.add(Bxdf::Fourier(FourierBSDF::new(
self.bsdf_table.clone(),
mode,
Some(sc),
)));
} else {
bsdf.add(Bxdf::Fourier(FourierBSDF::new(
self.bsdf_table.clone(),
mode,
None,
)));
}
}
}
}