mirror of
https://github.com/zaphar/kitchen.git
synced 2025-07-22 19:40:14 -04:00
Sort the meals in the meal plan by category
This commit is contained in:
parent
ab0937c3b2
commit
b95b297deb
@ -30,6 +30,7 @@ use crate::api::{HttpStore, LocalStore};
|
|||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub struct AppState {
|
pub struct AppState {
|
||||||
pub recipe_counts: BTreeMap<String, usize>,
|
pub recipe_counts: BTreeMap<String, usize>,
|
||||||
|
pub recipe_categories: BTreeMap<String, String>,
|
||||||
pub extras: Vec<(String, String)>,
|
pub extras: Vec<(String, String)>,
|
||||||
pub staples: Option<BTreeSet<Ingredient>>,
|
pub staples: Option<BTreeSet<Ingredient>>,
|
||||||
pub recipes: BTreeMap<String, Recipe>,
|
pub recipes: BTreeMap<String, Recipe>,
|
||||||
@ -45,6 +46,7 @@ impl AppState {
|
|||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
Self {
|
Self {
|
||||||
recipe_counts: BTreeMap::new(),
|
recipe_counts: BTreeMap::new(),
|
||||||
|
recipe_categories: BTreeMap::new(),
|
||||||
extras: Vec::new(),
|
extras: Vec::new(),
|
||||||
staples: None,
|
staples: None,
|
||||||
recipes: BTreeMap::new(),
|
recipes: BTreeMap::new(),
|
||||||
@ -187,6 +189,18 @@ impl StateMachine {
|
|||||||
|
|
||||||
if let Some(recipe_entries) = recipe_entries {
|
if let Some(recipe_entries) = recipe_entries {
|
||||||
local_store.set_all_recipes(recipe_entries);
|
local_store.set_all_recipes(recipe_entries);
|
||||||
|
state.recipe_categories = recipe_entries
|
||||||
|
.iter()
|
||||||
|
.map(|entry| {
|
||||||
|
(
|
||||||
|
entry.recipe_id().to_owned(),
|
||||||
|
entry
|
||||||
|
.category()
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_else(|| "Unknown".to_owned()),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.collect::<BTreeMap<String, String>>();
|
||||||
}
|
}
|
||||||
|
|
||||||
info!("Fetching meal plan list");
|
info!("Fetching meal plan list");
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
// Copyright 2022 Jeremy Wall
|
// Copyright 2022 Jeremy Wall
|
||||||
//
|
//
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
@ -12,30 +14,45 @@
|
|||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
use recipes::Recipe;
|
use recipes::Recipe;
|
||||||
|
use reqwasm::websocket::State;
|
||||||
use sycamore::prelude::*;
|
use sycamore::prelude::*;
|
||||||
use tracing::instrument;
|
use tracing::{debug, instrument};
|
||||||
|
|
||||||
use crate::app_state::{Message, StateHandler};
|
use crate::app_state::{Message, StateHandler};
|
||||||
use crate::components::recipe_selection::*;
|
use crate::components::recipe_selection::*;
|
||||||
|
|
||||||
|
#[derive(Props)]
|
||||||
|
pub struct CategoryGroupProps<'ctx> {
|
||||||
|
sh: StateHandler<'ctx>,
|
||||||
|
category: String,
|
||||||
|
recipes: Vec<(String, Recipe)>,
|
||||||
|
row_size: usize,
|
||||||
|
}
|
||||||
|
|
||||||
#[allow(non_snake_case)]
|
#[allow(non_snake_case)]
|
||||||
#[instrument(skip_all)]
|
pub fn CategoryGroup<'ctx, G: Html>(
|
||||||
pub fn RecipePlan<'ctx, G: Html>(cx: Scope<'ctx>, sh: StateHandler<'ctx>) -> View<G> {
|
cx: Scope<'ctx>,
|
||||||
|
CategoryGroupProps {
|
||||||
|
sh,
|
||||||
|
category,
|
||||||
|
recipes,
|
||||||
|
row_size,
|
||||||
|
}: CategoryGroupProps<'ctx>,
|
||||||
|
) -> View<G> {
|
||||||
let rows = sh.get_selector(cx, move |state| {
|
let rows = sh.get_selector(cx, move |state| {
|
||||||
let mut rows = Vec::new();
|
let mut rows = Vec::new();
|
||||||
for row in state
|
for row in recipes
|
||||||
.get()
|
|
||||||
.recipes
|
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(k, v)| create_signal(cx, (k.clone(), v.clone())))
|
.map(|(id, r)| create_signal(cx, (id.clone(), r.clone())))
|
||||||
.collect::<Vec<&Signal<(String, Recipe)>>>()
|
.collect::<Vec<&Signal<(String, Recipe)>>>()
|
||||||
.chunks(4)
|
.chunks(row_size)
|
||||||
{
|
{
|
||||||
rows.push(create_signal(cx, Vec::from(row)));
|
rows.push(create_signal(cx, Vec::from(row)));
|
||||||
}
|
}
|
||||||
rows
|
rows
|
||||||
});
|
});
|
||||||
view! {cx,
|
view! {cx,
|
||||||
|
h2 { (category) }
|
||||||
table(class="recipe_selector no-print") {
|
table(class="recipe_selector no-print") {
|
||||||
(View::new_fragment(
|
(View::new_fragment(
|
||||||
rows.get().iter().cloned().map(|r| {
|
rows.get().iter().cloned().map(|r| {
|
||||||
@ -54,6 +71,44 @@ pub fn RecipePlan<'ctx, G: Html>(cx: Scope<'ctx>, sh: StateHandler<'ctx>) -> Vie
|
|||||||
}).collect()
|
}).collect()
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(non_snake_case)]
|
||||||
|
#[instrument(skip_all)]
|
||||||
|
pub fn RecipePlan<'ctx, G: Html>(cx: Scope<'ctx>, sh: StateHandler<'ctx>) -> View<G> {
|
||||||
|
let recipe_category_groups = sh.get_selector(cx, |state| {
|
||||||
|
state
|
||||||
|
.get()
|
||||||
|
.recipe_categories
|
||||||
|
.iter()
|
||||||
|
.fold(BTreeMap::new(), |mut map, (r, cat)| {
|
||||||
|
debug!(?cat, recipe_id=?r, "Accumulating recipe into category");
|
||||||
|
map.entry(cat.clone()).or_insert(Vec::new()).push((
|
||||||
|
r.clone(),
|
||||||
|
state
|
||||||
|
.get()
|
||||||
|
.recipes
|
||||||
|
.get(r)
|
||||||
|
.expect("Failed to find recipe")
|
||||||
|
.clone(),
|
||||||
|
));
|
||||||
|
map
|
||||||
|
})
|
||||||
|
.iter()
|
||||||
|
.map(|(cat, rs)| (cat.clone(), rs.clone()))
|
||||||
|
.collect::<Vec<(String, Vec<(String, Recipe)>)>>()
|
||||||
|
});
|
||||||
|
view! {cx,
|
||||||
|
Keyed(
|
||||||
|
iterable=recipe_category_groups,
|
||||||
|
view=move |cx, (cat, recipes)| {
|
||||||
|
view! {cx,
|
||||||
|
CategoryGroup(sh=sh, category=cat, recipes=recipes, row_size=4)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
key=|(ref cat, _)| cat.clone(),
|
||||||
|
)
|
||||||
span(role="button", on:click=move |_| {
|
span(role="button", on:click=move |_| {
|
||||||
sh.dispatch(cx, Message::LoadState(None));
|
sh.dispatch(cx, Message::LoadState(None));
|
||||||
}) { "Reset" } " "
|
}) { "Reset" } " "
|
||||||
|
Loading…
x
Reference in New Issue
Block a user