I'm currently building an image carousel. I want to count the contents of a directory that is only accessible at the OS level and pass the result to a static hashmap inside a web assembly module in Yew.
Here's a use_reducer
paired with a Context
in which I'd like the default state to hold the count of files.
use std::{cmp::max, collections::HashMap, fs, rc::Rc};
use yew::prelude::*;
extern crate web_sys;
// A macro to provide `println!(..)`-style syntax for `console.log` logging.
macro_rules! log {
( $( $t:tt )* ) => {
web_sys::console::log_1(&format!( $( $t )* ).into());
}
}
pub enum CarouselAction {
Prev,
Next,
}
#[derive(Debug, PartialEq, Eq, Clone)]
pub struct CarouselState {
chapter_state: HashMap<i32, i32>,
pub page: i32,
pub chapter: i32,
}
pub type CarouselContext = UseReducerHandle<CarouselState>;
impl Default for CarouselState {
fn default() -> Self {
let dir = "./assets/carousel/op";
let total_chapters = fs::read_dir(dir)
.unwrap()
.filter(|entry| entry.as_ref().unwrap().metadata().unwrap().is_file())
.count();
log!("TOTAL CHAPTERS {}", total_chapters);
Self {
chapter_state: HashMap::new(),
page: 1,
chapter: 1043,
}
}
}
impl Reducible for CarouselState {
// Reducer Action Type
type Action = CarouselAction;
fn reduce(self: Rc<Self>, action: Self::Action) -> Rc<Self> {
match action {
CarouselAction::Prev => Self {
page: max(self.page - 1, 1),
chapter: self.chapter,
chapter_state: self.chapter_state.to_owned(),
}
.into(),
CarouselAction::Next => {
log!("self.page {}", self.page);
Self {
page: self.page + 1,
chapter: self.chapter,
chapter_state: self.chapter_state.to_owned(),
}
.into()
}
}
}
}
#[derive(PartialEq, Debug, Properties)]
pub struct CarouselContextProps {
#[prop_or_default]
pub children: Children,
}
#[function_component(CarouselContextProvider)]
pub fn carousel_context_provider(props: &CarouselContextProps) -> Html {
let state = use_reducer(CarouselState::default);
html! {
<ContextProvider<CarouselContext> context={state}>
{props.children.clone()}
</ContextProvider<CarouselContext>>
}
}
pub fn use_carousel_context() -> impl Hook<Output = Option<UseReducerHandle<CarouselState>>> {
use_context::<CarouselContext>()
}
The code in question is:
impl Default for CarouselState {
fn default() -> Self {
let dir = "./assets/carousel/op";
let total_chapters = fs::read_dir(dir)
.unwrap()
.filter(|entry| entry.as_ref().unwrap().metadata().unwrap().is_file())
.count();
log!("TOTAL CHAPTERS {}", total_chapters);
Self {
chapter_state: HashMap::new(),
page: 1,
chapter: 1043,
}
}
}
I'm running the filesystem in the browser, which is not accessible, and it's returning an error (obviously). I want to run this block of code before it gets compiled to wasm and holds its result in the chapter_state
hash map.
Is this possible?
I'm currently thinking of using a separate script that generates a file with the hashmap result and then including that file in yew.
As @cafce25 pointed out, the trick is to use a build.rs file.
Yew has no way of "injecting" a JSON file. So we programmatically created a
info.rs
file that contained a HashMap from our output.Here's the content of
build.rs
(which lives in the root, outsidesrc
)Here's our code:
What we do here (feel free to suggest a better way) is: