summaryrefslogtreecommitdiff
path: root/src/gfx/media.rs
blob: 36554bf3b304b4a8dc0d8fa6714dac138eefeccc (plain)
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
use rand::Rng;
use std::collections::HashMap;

#[derive(rust_embed::RustEmbed)]
#[folder = "assets/textures"]
pub struct BaseFolder; // copied from github.com/minetest/minetest

pub struct MediaMgr {
    packs: Vec<HashMap<String, Vec<u8>>>,
    srv_idx: usize,
}

impl MediaMgr {
    pub fn new() -> Self {
        Self {
            packs: [
                BaseFolder::iter()
                    .map(|file| {
                        (
                            file.to_string(),
                            BaseFolder::get(&file).unwrap().data.into_owned(),
                        )
                    })
                    .collect(),
                HashMap::new(),
            ]
            .into(),
            srv_idx: 1,
        }
    }

    pub fn add_server_media(&mut self, files: HashMap<String, Vec<u8>>) {
        self.packs[self.srv_idx].extend(files.into_iter());
    }

    pub fn get(&self, file: &str) -> Option<&[u8]> {
        self.packs
            .iter()
            .rev()
            .find_map(|pack| pack.get(file))
            .map(Vec::as_slice)
    }

    pub fn rand_img() -> image::RgbaImage {
        let mut img = image::RgbImage::new(1, 1);
        rand::thread_rng().fill(&mut img.get_pixel_mut(0, 0).0);

        image::DynamicImage::from(img).to_rgba8()
    }

    pub fn texture(&self, texture: &str) -> image::RgbaImage {
        match match self.get(texture) {
            Some(payload) => image::load_from_memory(payload)
                .or_else(|_| image::load_from_memory_with_format(payload, image::ImageFormat::Tga))
                .map_err(|e| eprintln!("while loading {texture}: {e}"))
                .ok(),
            None => {
                eprintln!("unknown texture: {texture}");
                None
            }
        } {
            Some(v) => image::imageops::flip_vertical(&v),
            None => Self::rand_img(),
        }
    }

    pub fn texture_string(&self, texture: &str) -> image::RgbaImage {
        texture
            .split('^')
            .fold(None, |mut base, next| {
                if let Some(overlay) = match next {
                    "" => Some(self.texture("no_texture.png")),
                    texmod if matches!(texmod.chars().next(), Some('[')) => {
                        eprintln!("unknown texture modifier: {texmod}");
                        None
                    }
                    texture => Some(self.texture(texture)),
                } {
                    if let Some(base) = &mut base {
                        image::imageops::overlay(base, &overlay, 0, 0);
                    } else {
                        base = Some(overlay);
                    }
                }

                base
            })
            .unwrap_or_else(Self::rand_img)
    }
}