kitchen/web/src/web.rs

117 lines
3.5 KiB
Rust
Raw Normal View History

2022-01-13 18:07:01 -05:00
// 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.
2022-01-23 14:28:01 -05:00
#![allow(non_snake_case)]
use std::iter::Iterator;
use crate::console_log;
2022-01-13 18:07:01 -05:00
use dioxus::prelude::*;
2022-01-23 14:28:01 -05:00
use reqwasm::http;
use recipes::{parse, Recipe};
2022-01-24 19:59:16 -05:00
#[derive(Props, PartialEq, Clone)]
struct AppService {
recipes: Vec<Recipe>,
2022-01-23 14:28:01 -05:00
}
2022-01-24 19:59:16 -05:00
impl AppService {
fn new() -> Self {
Self {
recipes: Vec::new(),
}
}
2022-01-23 14:28:01 -05:00
2022-01-24 19:59:16 -05:00
async fn fetch_recipes() -> Result<Vec<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_log!("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_log!("Error parsing recipe {}", e);
break;
2022-01-23 14:28:01 -05:00
}
2022-01-24 19:59:16 -05:00
};
console_log!("We parsed a recipe {}", recipe.title);
parsed_list.push(recipe);
2022-01-23 14:28:01 -05:00
}
2022-01-24 19:59:16 -05:00
// TODO(jwall): It would appear that their API doesn't support this
// model for async operations.
//self.recipes = parsed_list;
return Ok(parsed_list);
2022-01-23 14:28:01 -05:00
}
2022-01-24 19:59:16 -05:00
}
fn get_recipes(&self) -> &Vec<Recipe> {
&self.recipes
}
fn set_recipes(&mut self, recipes: Vec<Recipe>) {
self.recipes = recipes;
}
}
#[derive(Props)]
struct RecipeListProps<'a> {
app_service: UseState<'a, AppService>,
}
/// Component to list available recipes.
fn recipe_list<'a>(cx: Scope<'a, RecipeListProps<'a>>) -> Element {
let props = cx.props.app_service;
2022-01-23 14:28:01 -05:00
cx.render(rsx! {
ul {
2022-01-24 19:59:16 -05:00
props.get_recipes().into_iter().map(|i| {
2022-01-23 14:28:01 -05:00
let title = &i.title;
rsx!(li { "{title}" })
})
}
})
}
2022-01-13 18:07:01 -05:00
pub fn ui(cx: Scope) -> Element {
2022-01-24 19:59:16 -05:00
let app_state = use_state(&cx, AppService::new);
let fut = use_future(&cx, || async move { AppService::fetch_recipes().await });
2022-01-13 18:07:01 -05:00
cx.render(rsx! {
div { "hello chefs!" }
2022-01-24 19:59:16 -05:00
{match fut.value() {
Some(Ok(recipes)) => {
app_state.modify().set_recipes(recipes.clone());
rsx!{ recipe_list(app_service: app_state) }
}
Some(Err(e)) => {
console_log!("{}", e);
rsx!{ div { class: "error", "{e}" } }
}
None => {
//panic!("We seem to have failed to execute our future.")
rsx!{ div { "Loading recipe list..." }}
}
}}
2022-01-13 18:07:01 -05:00
})
2022-01-23 14:28:01 -05:00
}