2020-01-03 03:51:07 +00:00
|
|
|
use crate::ui::{Graphic, SampleStrat, Transform, Ui};
|
2019-10-09 19:28:05 +00:00
|
|
|
use common::{
|
|
|
|
assets::{self, watch::ReloadIndicator, Asset},
|
2020-03-28 05:51:52 +00:00
|
|
|
comp::item::{
|
2020-07-18 00:05:28 +00:00
|
|
|
armor::{Armor, ArmorKind},
|
2020-03-28 05:51:52 +00:00
|
|
|
tool::{Tool, ToolKind},
|
2020-08-04 23:21:42 +00:00
|
|
|
Item, ItemKind, Lantern, Throwable, Utility,
|
2020-03-28 05:51:52 +00:00
|
|
|
},
|
2020-04-06 02:50:27 +00:00
|
|
|
figure::Segment,
|
2019-10-09 19:28:05 +00:00
|
|
|
};
|
|
|
|
use conrod_core::image::Id;
|
|
|
|
use dot_vox::DotVoxData;
|
|
|
|
use hashbrown::HashMap;
|
|
|
|
use image::DynamicImage;
|
|
|
|
use serde_derive::{Deserialize, Serialize};
|
|
|
|
use std::{fs::File, io::BufReader, sync::Arc};
|
2020-06-21 10:22:26 +00:00
|
|
|
use tracing::{error, warn};
|
2019-10-09 19:28:05 +00:00
|
|
|
use vek::*;
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)]
|
2019-10-22 18:18:40 +00:00
|
|
|
pub enum ItemKey {
|
2019-12-30 13:56:42 +00:00
|
|
|
Tool(ToolKind),
|
2020-08-04 23:21:42 +00:00
|
|
|
Lantern(String),
|
2020-07-18 00:05:28 +00:00
|
|
|
Armor(ArmorKind),
|
2020-01-29 12:01:28 +00:00
|
|
|
Utility(Utility),
|
2020-08-04 23:21:42 +00:00
|
|
|
Consumable(String),
|
2020-07-04 23:55:13 +00:00
|
|
|
Throwable(Throwable),
|
2020-08-04 23:21:42 +00:00
|
|
|
Ingredient(String),
|
2020-03-16 13:27:52 +00:00
|
|
|
Empty,
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
2019-10-22 18:18:40 +00:00
|
|
|
impl From<&Item> for ItemKey {
|
2019-10-09 19:28:05 +00:00
|
|
|
fn from(item: &Item) -> Self {
|
2019-10-22 18:18:40 +00:00
|
|
|
match &item.kind {
|
2020-08-02 01:21:32 +00:00
|
|
|
ItemKind::Tool(Tool { kind, .. }) => ItemKey::Tool(kind.clone()),
|
2020-08-04 23:21:42 +00:00
|
|
|
ItemKind::Lantern(Lantern { kind, .. }) => ItemKey::Lantern(kind.clone()),
|
2020-08-03 03:41:32 +00:00
|
|
|
ItemKind::Armor(Armor { kind, .. }) => ItemKey::Armor(kind.clone()),
|
2020-06-18 21:25:48 +00:00
|
|
|
ItemKind::Utility { kind, .. } => ItemKey::Utility(*kind),
|
2020-08-04 23:21:42 +00:00
|
|
|
ItemKind::Consumable { kind, .. } => ItemKey::Consumable(kind.clone()),
|
2020-07-04 23:55:13 +00:00
|
|
|
ItemKind::Throwable { kind, .. } => ItemKey::Throwable(*kind),
|
2020-08-04 23:21:42 +00:00
|
|
|
ItemKind::Ingredient { kind, .. } => ItemKey::Ingredient(kind.clone()),
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
enum ImageSpec {
|
|
|
|
Png(String),
|
|
|
|
Vox(String),
|
|
|
|
// (specifier, offset, (axis, 2 * angle / pi), zoom)
|
|
|
|
VoxTrans(String, [f32; 3], [f32; 3], f32),
|
|
|
|
}
|
|
|
|
impl ImageSpec {
|
|
|
|
fn create_graphic(&self) -> Graphic {
|
|
|
|
match self {
|
|
|
|
ImageSpec::Png(specifier) => Graphic::Image(graceful_load_img(&specifier)),
|
|
|
|
ImageSpec::Vox(specifier) => Graphic::Voxel(
|
2020-04-06 02:50:27 +00:00
|
|
|
graceful_load_segment_no_skin(&specifier),
|
2019-10-09 19:28:05 +00:00
|
|
|
Transform {
|
|
|
|
stretch: false,
|
|
|
|
..Default::default()
|
|
|
|
},
|
2020-01-03 03:51:07 +00:00
|
|
|
SampleStrat::None,
|
2019-10-09 19:28:05 +00:00
|
|
|
),
|
|
|
|
ImageSpec::VoxTrans(specifier, offset, [rot_x, rot_y, rot_z], zoom) => Graphic::Voxel(
|
2020-04-06 02:50:27 +00:00
|
|
|
graceful_load_segment_no_skin(&specifier),
|
2019-10-09 19:28:05 +00:00
|
|
|
Transform {
|
|
|
|
ori: Quaternion::rotation_x(rot_x * std::f32::consts::PI / 180.0)
|
|
|
|
.rotated_y(rot_y * std::f32::consts::PI / 180.0)
|
|
|
|
.rotated_z(rot_z * std::f32::consts::PI / 180.0),
|
|
|
|
offset: Vec3::from(*offset),
|
|
|
|
zoom: *zoom,
|
|
|
|
orth: true, // TODO: Is this what we want here? @Pfau
|
|
|
|
stretch: false,
|
|
|
|
},
|
2020-01-03 03:51:07 +00:00
|
|
|
SampleStrat::None,
|
2019-10-09 19:28:05 +00:00
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize)]
|
2019-10-22 18:18:40 +00:00
|
|
|
struct ItemImagesSpec(HashMap<ItemKey, ImageSpec>);
|
2019-10-09 19:28:05 +00:00
|
|
|
impl Asset for ItemImagesSpec {
|
|
|
|
const ENDINGS: &'static [&'static str] = &["ron"];
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2019-10-09 19:28:05 +00:00
|
|
|
fn parse(buf_reader: BufReader<File>) -> Result<Self, assets::Error> {
|
2020-03-29 01:09:19 +00:00
|
|
|
ron::de::from_reader(buf_reader).map_err(assets::Error::parse_error)
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-04 05:40:00 +00:00
|
|
|
// TODO: when there are more images don't load them all into memory
|
2019-10-09 19:28:05 +00:00
|
|
|
pub struct ItemImgs {
|
2019-10-22 18:18:40 +00:00
|
|
|
map: HashMap<ItemKey, Id>,
|
2019-10-09 19:28:05 +00:00
|
|
|
indicator: ReloadIndicator,
|
2020-04-04 05:40:00 +00:00
|
|
|
not_found: Id,
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
impl ItemImgs {
|
2020-04-04 05:40:00 +00:00
|
|
|
pub fn new(ui: &mut Ui, not_found: Id) -> Self {
|
2019-10-09 19:28:05 +00:00
|
|
|
let mut indicator = ReloadIndicator::new();
|
|
|
|
Self {
|
|
|
|
map: assets::load_watched::<ItemImagesSpec>(
|
|
|
|
"voxygen.item_image_manifest",
|
|
|
|
&mut indicator,
|
|
|
|
)
|
|
|
|
.expect("Unable to load item image manifest")
|
|
|
|
.0
|
|
|
|
.iter()
|
2020-04-04 05:40:00 +00:00
|
|
|
// TODO: what if multiple kinds map to the same image, it would be nice to use the same
|
|
|
|
// image id for both, although this does interfere with the current hot-reloading
|
|
|
|
// strategy
|
2019-10-09 19:28:05 +00:00
|
|
|
.map(|(kind, spec)| (kind.clone(), ui.add_graphic(spec.create_graphic())))
|
|
|
|
.collect(),
|
|
|
|
indicator,
|
2020-04-04 05:40:00 +00:00
|
|
|
not_found,
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2019-10-09 19:28:05 +00:00
|
|
|
/// Checks if the manifest has been changed and reloads the images if so
|
|
|
|
/// Reuses img ids
|
|
|
|
pub fn reload_if_changed(&mut self, ui: &mut Ui) {
|
|
|
|
if self.indicator.reloaded() {
|
|
|
|
for (kind, spec) in assets::load::<ItemImagesSpec>("voxygen.item_image_manifest")
|
|
|
|
.expect("Unable to load item image manifest")
|
|
|
|
.0
|
|
|
|
.iter()
|
|
|
|
{
|
|
|
|
// Load new graphic
|
|
|
|
let graphic = spec.create_graphic();
|
|
|
|
// See if we already have an id we can use
|
|
|
|
match self.map.get(&kind) {
|
|
|
|
Some(id) => ui.replace_graphic(*id, graphic),
|
2019-10-22 18:18:40 +00:00
|
|
|
// Otherwise, generate new id and insert it into our Id -> ItemKey map
|
2019-10-09 19:28:05 +00:00
|
|
|
None => {
|
|
|
|
self.map.insert(kind.clone(), ui.add_graphic(graphic));
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2019-10-22 18:18:40 +00:00
|
|
|
pub fn img_id(&self, item_kind: ItemKey) -> Option<Id> {
|
2019-10-09 19:28:05 +00:00
|
|
|
match self.map.get(&item_kind) {
|
|
|
|
Some(id) => Some(*id),
|
|
|
|
// There was no specification in the ron
|
|
|
|
None => {
|
|
|
|
warn!(
|
2020-06-21 21:47:49 +00:00
|
|
|
?item_kind,
|
|
|
|
"missing specified image file (note: hot-reloading won't work here)",
|
2019-10-09 19:28:05 +00:00
|
|
|
);
|
|
|
|
None
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
}
|
2020-04-04 05:40:00 +00:00
|
|
|
|
|
|
|
pub fn img_id_or_not_found_img(&self, item_kind: ItemKey) -> Id {
|
|
|
|
self.img_id(item_kind).unwrap_or(self.not_found)
|
|
|
|
}
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Copied from figure/load.rs
|
|
|
|
// TODO: remove code dup?
|
|
|
|
fn graceful_load_vox(specifier: &str) -> Arc<DotVoxData> {
|
|
|
|
let full_specifier: String = ["voxygen.", specifier].concat();
|
|
|
|
match assets::load::<DotVoxData>(full_specifier.as_str()) {
|
|
|
|
Ok(dot_vox) => dot_vox,
|
|
|
|
Err(_) => {
|
2020-06-21 21:47:49 +00:00
|
|
|
error!(?full_specifier, "Could not load vox file for item images",);
|
2019-10-09 19:28:05 +00:00
|
|
|
assets::load_expect::<DotVoxData>("voxygen.voxel.not_found")
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fn graceful_load_img(specifier: &str) -> Arc<DynamicImage> {
|
|
|
|
let full_specifier: String = ["voxygen.", specifier].concat();
|
|
|
|
match assets::load::<DynamicImage>(full_specifier.as_str()) {
|
|
|
|
Ok(img) => img,
|
|
|
|
Err(_) => {
|
2020-06-21 21:47:49 +00:00
|
|
|
error!(?full_specifier, "Could not load image file for item images");
|
2019-10-09 19:28:05 +00:00
|
|
|
assets::load_expect::<DynamicImage>("voxygen.element.not_found")
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-10-09 19:28:05 +00:00
|
|
|
}
|
|
|
|
}
|
2020-04-06 02:50:27 +00:00
|
|
|
|
|
|
|
fn graceful_load_segment_no_skin(specifier: &str) -> Arc<Segment> {
|
|
|
|
use common::figure::{mat_cell::MatCell, MatSegment};
|
|
|
|
let mat_seg = MatSegment::from(&*graceful_load_vox(specifier));
|
|
|
|
let seg = mat_seg
|
|
|
|
.map(|mat_cell| match mat_cell {
|
|
|
|
MatCell::None => None,
|
|
|
|
MatCell::Mat(_) => Some(MatCell::None),
|
|
|
|
MatCell::Normal(_) => None,
|
|
|
|
})
|
|
|
|
.to_segment(|_| Rgb::broadcast(255));
|
|
|
|
Arc::new(seg)
|
|
|
|
}
|