mirror of
https://github.com/zaphar/kitchen.git
synced 2025-07-22 19:40:14 -04:00
115 lines
3.8 KiB
Rust
115 lines
3.8 KiB
Rust
// Copyright 2022 Jeremy Wall
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
use std::collections::BTreeMap;
|
|
|
|
use crate::{console_debug, console_error};
|
|
|
|
use reqwasm::http;
|
|
use sycamore::prelude::*;
|
|
|
|
use recipes::{parse, Ingredient, IngredientAccumulator, IngredientKey, Recipe};
|
|
|
|
#[derive(Clone)]
|
|
pub struct AppService {
|
|
// TODO(jwall): Should each Recipe also be a Signal?
|
|
recipes: Signal<Vec<(usize, Signal<Recipe>)>>,
|
|
menu_list: Signal<BTreeMap<usize, usize>>,
|
|
}
|
|
|
|
impl AppService {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
recipes: Signal::new(Vec::new()),
|
|
menu_list: Signal::new(BTreeMap::new()),
|
|
}
|
|
}
|
|
|
|
pub async fn fetch_recipes() -> Result<Vec<(usize, Recipe)>, String> {
|
|
let resp = match http::Request::get("/api/v1/recipes").send().await {
|
|
Ok(resp) => resp,
|
|
Err(e) => return Err(format!("Error: {}", e)),
|
|
};
|
|
if resp.status() != 200 {
|
|
return Err(format!("Status: {}", resp.status()));
|
|
} else {
|
|
console_debug!("We got a valid response back!");
|
|
let recipe_list = match resp.json::<Vec<String>>().await {
|
|
Ok(recipes) => recipes,
|
|
Err(e) => return Err(format!("Eror getting recipe list as json {}", e)),
|
|
};
|
|
let mut parsed_list = Vec::new();
|
|
for r in recipe_list {
|
|
let recipe = match parse::as_recipe(&r) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
console_error!("Error parsing recipe {}", e);
|
|
continue;
|
|
}
|
|
};
|
|
console_debug!("We parsed a recipe {}", recipe.title);
|
|
parsed_list.push(recipe);
|
|
}
|
|
return Ok(parsed_list.drain(0..).enumerate().collect());
|
|
}
|
|
}
|
|
|
|
pub fn get_recipe_by_index(&self, idx: usize) -> Option<Signal<Recipe>> {
|
|
self.recipes.get().get(idx).map(|(_, r)| r.clone())
|
|
}
|
|
|
|
pub fn get_shopping_list(&self) -> BTreeMap<IngredientKey, Ingredient> {
|
|
let mut acc = IngredientAccumulator::new();
|
|
let recipe_counts = self.menu_list.get();
|
|
for (idx, count) in recipe_counts.iter() {
|
|
for _ in 0..*count {
|
|
acc.accumulate_from(self.get_recipe_by_index(*idx).unwrap().get().as_ref());
|
|
}
|
|
}
|
|
acc.ingredients()
|
|
}
|
|
|
|
pub fn set_recipe_count_by_index(&mut self, i: usize, count: usize) {
|
|
let mut v = (*self.menu_list.get()).clone();
|
|
v.insert(i, count);
|
|
self.menu_list.set(v);
|
|
}
|
|
|
|
pub fn get_recipe_count_by_index(&self, i: usize) -> usize {
|
|
self.menu_list.get().get(&i).map(|i| *i).unwrap_or_default()
|
|
}
|
|
|
|
pub fn get_recipes(&self) -> Signal<Vec<(usize, Signal<Recipe>)>> {
|
|
self.recipes.clone()
|
|
}
|
|
|
|
pub fn get_menu_list(&self) -> Vec<(usize, usize)> {
|
|
self.menu_list
|
|
.get()
|
|
.iter()
|
|
// We exclude recipes in the menu_list with count 0
|
|
.filter(|&(_, count)| *count != 0)
|
|
.map(|(idx, count)| (*idx, *count))
|
|
.collect()
|
|
}
|
|
|
|
pub fn set_recipes(&mut self, mut recipes: Vec<(usize, Recipe)>) {
|
|
self.recipes.set(
|
|
recipes
|
|
.drain(0..)
|
|
.map(|(i, r)| (i, Signal::new(r)))
|
|
.collect(),
|
|
);
|
|
}
|
|
}
|