update demo-recipe-app,
This commit is contained in:
@@ -26,7 +26,7 @@ import { BookmarkStore } from '../store';
|
||||
|
||||
const Categories = () => {
|
||||
const router = useIonRouter();
|
||||
const pageRef = useRef();
|
||||
const pageRef = useRef(null);
|
||||
const [recipeCategories, setRecipeCategories] = useState([]);
|
||||
const bookmarks = useStoreState(BookmarkStore, getBookmarks);
|
||||
|
||||
|
@@ -39,7 +39,7 @@ import { useStoreState } from 'pullstate';
|
||||
import { getBookmarks } from '../store/Selectors';
|
||||
|
||||
const Recipe = () => {
|
||||
const pageRef = useRef();
|
||||
const pageRef = useRef(null);
|
||||
const { state } = useLocation();
|
||||
const [recipe, setRecipe] = useState([]);
|
||||
const [fromSearch, setFromSearch] = useState(false);
|
||||
|
@@ -25,7 +25,7 @@ import { performSearch } from '../utils';
|
||||
import { RecipeListItem } from '../components/RecipeListItem';
|
||||
|
||||
const Search = () => {
|
||||
const searchRef = useRef();
|
||||
const searchRef = useRef(null);
|
||||
const [searchResults, setSearchResults] = useState([]);
|
||||
const [showLoader, hideLoader] = useIonLoading();
|
||||
|
||||
|
@@ -0,0 +1,16 @@
|
||||
import { IonItem, IonLabel } from "@ionic/react";
|
||||
import styles from "./Ingredient.module.scss";
|
||||
|
||||
export const Ingredient = ({ ingredient }) => {
|
||||
|
||||
return (
|
||||
|
||||
<IonItem lines="full" className={ styles.ingredientItem }>
|
||||
<img alt="ingredient" src={ ingredient.image } className={ styles.ingredientImage } />
|
||||
<IonLabel className="ion-text-wrap ion-margin-start">
|
||||
<h3>{ ingredient.text }</h3>
|
||||
<p>{ ingredient.weight.toFixed(2) }g</p>
|
||||
</IonLabel>
|
||||
</IonItem>
|
||||
);
|
||||
}
|
@@ -0,0 +1,13 @@
|
||||
.ingredientImage {
|
||||
|
||||
height: 3rem;
|
||||
width: 3rem;
|
||||
border-radius: 10px;
|
||||
border: 1px solid rgb(172, 172, 172);
|
||||
}
|
||||
|
||||
.ingredientItem {
|
||||
|
||||
--padding-top: 1rem;
|
||||
--padding-bottom: 1rem;
|
||||
}
|
@@ -0,0 +1,30 @@
|
||||
import { IonButton, IonButtons, IonContent, IonGrid, IonHeader, IonList, IonPage, IonRow, IonTitle, IonToolbar } from "@ionic/react"
|
||||
import { Ingredient } from "./Ingredient";
|
||||
|
||||
const IngredientsModal = ({ dismiss, ingredients}) => {
|
||||
|
||||
return (
|
||||
|
||||
<IonPage>
|
||||
<IonHeader>
|
||||
<IonToolbar>
|
||||
<IonTitle>View Ingredients</IonTitle>
|
||||
<IonButtons slot="start">
|
||||
<IonButton color="main" onClick={ dismiss }>Close</IonButton>
|
||||
</IonButtons>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
|
||||
<IonContent>
|
||||
<IonList>
|
||||
{ ingredients.map((ingredient, index) => {
|
||||
|
||||
return <Ingredient key={ `ingredient_${ index }` } ingredient={ ingredient } />;
|
||||
})}
|
||||
</IonList>
|
||||
</IonContent>
|
||||
</IonPage>
|
||||
);
|
||||
}
|
||||
|
||||
export default IngredientsModal;
|
@@ -0,0 +1,48 @@
|
||||
import { IonButton, IonButtons, IonCardSubtitle, IonCol, IonContent, IonGrid, IonHeader, IonPage, IonRow, IonTitle, IonToolbar } from "@ionic/react"
|
||||
import NutritionalFact from "./NutritionalFact";
|
||||
|
||||
const NutritionModal = ({ dismiss, recipe }) => {
|
||||
|
||||
return (
|
||||
|
||||
<IonPage>
|
||||
<IonHeader>
|
||||
<IonToolbar>
|
||||
<IonTitle>View Nutrition</IonTitle>
|
||||
<IonButtons slot="start">
|
||||
<IonButton color="main" onClick={ dismiss }>Close</IonButton>
|
||||
</IonButtons>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
|
||||
<IonContent>
|
||||
|
||||
{ (recipe && recipe.digest) &&
|
||||
|
||||
<IonGrid>
|
||||
<IonRow>
|
||||
<IonCol size="12">
|
||||
<IonCardSubtitle className="ion-text-center" color="main">
|
||||
Based on a serving size of { recipe.totalWeight.toFixed(0) }g
|
||||
</IonCardSubtitle>
|
||||
<NutritionalFact type="calories" amount={ recipe.calories.toFixed(0) } />
|
||||
<NutritionalFact type="fat" amount={ recipe.digest[0].total.toFixed(0) } />
|
||||
<NutritionalFact type="trans_fat" amount={ recipe.digest[0].sub[1].total.toFixed(0) } inset={ true } />
|
||||
<NutritionalFact type="saturated_fat" amount={ recipe.digest[0].sub[0].total.toFixed(0) } inset={ true } />
|
||||
<NutritionalFact type="polyunsaturated_fat" amount={ recipe.digest[0].sub[3].total.toFixed(0) } inset={ true } />
|
||||
<NutritionalFact type="monounsaturated_fat" amount={ recipe.digest[0].sub[2].total.toFixed(0) } inset={ true } />
|
||||
<NutritionalFact type="carbs" amount={ recipe.digest[1].total.toFixed(0) } />
|
||||
<NutritionalFact type="sugar" amount={ recipe.digest[1].sub[2].total.toFixed(0) } inset={ true } />
|
||||
<NutritionalFact type="fibre" amount={ recipe.digest[1].sub[1].total.toFixed(0) } inset={ true } />
|
||||
<NutritionalFact type="sugars_added" amount={ recipe.digest[1].sub[3].total.toFixed(0) } inset={ true } />
|
||||
<NutritionalFact type="protein" amount={ recipe.digest[2].total.toFixed(0) } />
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
</IonGrid>
|
||||
}
|
||||
</IonContent>
|
||||
</IonPage>
|
||||
);
|
||||
}
|
||||
|
||||
export default NutritionModal;
|
@@ -0,0 +1,25 @@
|
||||
import { IonCardTitle, IonCol, IonRow } from "@ionic/react";
|
||||
|
||||
const NutritionalFact = ({ type, amount, inset }) => {
|
||||
|
||||
const label = type.replace("_", " ").replace(/(^\w{1})|(\s+\w{1})/g, letter => letter.toUpperCase());
|
||||
|
||||
return (
|
||||
|
||||
<IonRow style={{ borderBottom: "1px solid #242424", padding: "0.5rem" }}>
|
||||
<IonCol size="9">
|
||||
<IonCardTitle style={{ fontSize: "0.9rem", marginLeft: inset ? "1.5rem" : "" }}>
|
||||
{ label }
|
||||
</IonCardTitle>
|
||||
</IonCol>
|
||||
|
||||
<IonCol size="3">
|
||||
<IonCardTitle style={{ fontSize: "0.9rem" }}>
|
||||
{ amount }{ type !== "calories" && "g" }
|
||||
</IonCardTitle>
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
);
|
||||
}
|
||||
|
||||
export default NutritionalFact;
|
@@ -0,0 +1,18 @@
|
||||
import { IonItem, IonLabel } from "@ionic/react";
|
||||
import { Link } from "react-router-dom";
|
||||
import styles from "./RecipeListItem.module.scss";
|
||||
|
||||
export const RecipeListItem = ({ recipe, fromSearch = false, fromBookmarks = false }) => (
|
||||
|
||||
<Link to={{ pathname: `/recipe/${ recipe.label.replace(" ", "").toLowerCase() }`, state: { recipe, fromSearch, fromBookmarks }}}>
|
||||
<IonItem detail={ true } lines="full" className={ styles.categoryItem }>
|
||||
|
||||
<img src={ recipe.image } alt="cover" className={ styles.categoryImage } />
|
||||
|
||||
<IonLabel className={ styles.categoryDetails }>
|
||||
<h2>{ recipe.label }</h2>
|
||||
<p>{ recipe.dishType && recipe.dishType[0] }</p>
|
||||
</IonLabel>
|
||||
</IonItem>
|
||||
</Link>
|
||||
);
|
@@ -0,0 +1,17 @@
|
||||
.categoryDetails {
|
||||
|
||||
margin-left: 1rem;
|
||||
}
|
||||
|
||||
.categoryImage {
|
||||
|
||||
width: 5rem;
|
||||
height: 5rem;
|
||||
border-radius: 10px;
|
||||
}
|
||||
|
||||
.categoryItem {
|
||||
|
||||
--padding-top: 1.5rem;
|
||||
--padding-bottom: 1.5rem;
|
||||
}
|
65
03_source/mobile/src/pages/DemoRecipeApp/pages/Bookmarks.jsx
Normal file
65
03_source/mobile/src/pages/DemoRecipeApp/pages/Bookmarks.jsx
Normal file
@@ -0,0 +1,65 @@
|
||||
import { IonBackButton, IonButtons, IonCol, IonContent, IonHeader, IonImg, IonList, IonNote, IonPage, IonRow, IonText, IonTitle, IonToolbar } from '@ionic/react';
|
||||
import { RecipeListItem } from '../components/RecipeListItem';
|
||||
|
||||
import { useStoreState } from "pullstate";
|
||||
import { BookmarkStore } from '../store';
|
||||
import { getBookmarks } from '../store/Selectors';
|
||||
|
||||
const Bookmarks = () => {
|
||||
|
||||
const bookmarks = useStoreState(BookmarkStore, getBookmarks);
|
||||
|
||||
return (
|
||||
<IonPage>
|
||||
<IonHeader>
|
||||
<IonToolbar>
|
||||
<IonButtons slot="start">
|
||||
<IonBackButton text="Categories" />
|
||||
</IonButtons>
|
||||
<IonTitle>Bookmarks</IonTitle>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
<IonContent fullscreen>
|
||||
<IonHeader collapse="condense">
|
||||
<IonToolbar>
|
||||
<IonTitle size="large">Bookmarks ({ bookmarks.length })</IonTitle>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
|
||||
<IonList>
|
||||
{ bookmarks.map((bookmark, index) => {
|
||||
|
||||
return (
|
||||
<RecipeListItem recipe={ bookmark } key={ `recipe_${ index }` } fromBookmarks={ true } />
|
||||
);
|
||||
})}
|
||||
</IonList>
|
||||
|
||||
{ bookmarks.length < 1 &&
|
||||
|
||||
<>
|
||||
<IonRow className="ion-justify-content-center ion-text-center ion-margin-top ion-padding-top">
|
||||
<IonCol size="8">
|
||||
<IonText>
|
||||
You don't have any bookmarks yet
|
||||
</IonText>
|
||||
|
||||
<IonNote>
|
||||
<p>When viewing a recipe, press the bookmark icon to add it</p>
|
||||
</IonNote>
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
|
||||
<IonRow className="ion-justify-content-center">
|
||||
<IonCol size="8">
|
||||
<IonImg src="/assets/bookmark.png" />
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
</>
|
||||
}
|
||||
</IonContent>
|
||||
</IonPage>
|
||||
);
|
||||
};
|
||||
|
||||
export default Bookmarks;
|
101
03_source/mobile/src/pages/DemoRecipeApp/pages/Categories.jsx
Normal file
101
03_source/mobile/src/pages/DemoRecipeApp/pages/Categories.jsx
Normal file
@@ -0,0 +1,101 @@
|
||||
import { useEffect, useState } from "react";
|
||||
import { IonButton, IonButtons, IonCardTitle, IonCol, IonContent, IonGrid, IonHeader, IonIcon, IonPage, IonRow, IonSearchbar, IonTitle, IonToolbar, useIonRouter } from '@ionic/react';
|
||||
import styles from "./Categories.module.scss";
|
||||
import { recipes } from "../recipes";
|
||||
import { useRef } from "react";
|
||||
import { Link } from "react-router-dom";
|
||||
import { bookmarkOutline } from "ionicons/icons";
|
||||
import { getBookmarks } from "../store/Selectors";
|
||||
import { useStoreState } from "pullstate";
|
||||
import { BookmarkStore } from "../store";
|
||||
|
||||
const Categories = () => {
|
||||
|
||||
const router = useIonRouter();
|
||||
const pageRef = useRef();
|
||||
const [ recipeCategories, setRecipeCategories ] = useState([]);
|
||||
const bookmarks = useStoreState(BookmarkStore, getBookmarks)
|
||||
|
||||
useEffect(() => {
|
||||
|
||||
const getAllRecipes = async () => {
|
||||
|
||||
const tempRecipeCategories = [
|
||||
{
|
||||
name: "Chicken",
|
||||
data: recipes.chicken.hits[0]
|
||||
},
|
||||
{
|
||||
name: "Beef",
|
||||
data: recipes.beef.hits[0]
|
||||
},
|
||||
{
|
||||
name: "Fish",
|
||||
data: recipes.fish.hits[0]
|
||||
},
|
||||
{
|
||||
name: "Fruit",
|
||||
data: recipes.fruit.hits[0]
|
||||
},
|
||||
{
|
||||
name: "Salad",
|
||||
data: recipes.salad.hits[0]
|
||||
},
|
||||
{
|
||||
name: "Vegan",
|
||||
data: recipes.vegan.hits[0]
|
||||
}
|
||||
];
|
||||
|
||||
setRecipeCategories(tempRecipeCategories);
|
||||
}
|
||||
|
||||
getAllRecipes();
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<IonPage ref={ pageRef }>
|
||||
<IonHeader className="ion-no-border">
|
||||
<IonToolbar className="ion-no-border">
|
||||
<IonTitle>Recipe Categories</IonTitle>
|
||||
|
||||
<IonButtons slot="end">
|
||||
<IonButton routerLink="/bookmarks">
|
||||
<IonIcon icon={ bookmarkOutline } />
|
||||
{ bookmarks.length }
|
||||
</IonButton>
|
||||
</IonButtons>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
<IonContent fullscreen>
|
||||
<div className={ styles.searchArea }>
|
||||
<IonSearchbar className="ion-justify-content-center" placeholder="Try 'Chicken Piccata'" onClick={ () => router.push("/search") } />
|
||||
</div>
|
||||
|
||||
<IonGrid>
|
||||
<IonRow className={ styles.row }>
|
||||
{ recipeCategories.map((category, index) => {
|
||||
|
||||
const { name, data } = category;
|
||||
const { recipe } = data;
|
||||
|
||||
return (
|
||||
|
||||
<IonCol className={ styles.col } size="6" key={ `recipe_${ index }` }>
|
||||
<Link to={ `/category/${ name }`}>
|
||||
<img src={ recipe.image } alt="cover" />
|
||||
<div className={ styles.categoryName }>
|
||||
<IonCardTitle>{ name }</IonCardTitle>
|
||||
</div>
|
||||
</Link>
|
||||
</IonCol>
|
||||
);
|
||||
})}
|
||||
</IonRow>
|
||||
</IonGrid>
|
||||
</IonContent>
|
||||
</IonPage>
|
||||
);
|
||||
};
|
||||
|
||||
export default Categories;
|
@@ -0,0 +1,65 @@
|
||||
.categoryName {
|
||||
|
||||
position: absolute;
|
||||
left: 0;
|
||||
top: 0;
|
||||
margin: 0 auto;
|
||||
z-index: 10;
|
||||
background-color: rgba(0, 0, 0, 0.3);
|
||||
border-radius: 10px;
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
|
||||
ion-card-title {
|
||||
|
||||
color: white;
|
||||
margin-left: 0.5rem;
|
||||
margin-top: 0.5rem;
|
||||
padding: 0.5rem;
|
||||
width: fit-content;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
border-radius: 10px;
|
||||
}
|
||||
}
|
||||
|
||||
.row,
|
||||
.col,
|
||||
.card {
|
||||
|
||||
padding: 0;
|
||||
background-color: white;
|
||||
border-radius: 10px;
|
||||
}
|
||||
|
||||
.col {
|
||||
|
||||
border: 5px solid white;
|
||||
|
||||
img {
|
||||
|
||||
border-radius: 10px;
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
|
||||
.searchArea {
|
||||
|
||||
background-color: var(--ion-toolbar-background);
|
||||
padding-bottom: 0.5rem;
|
||||
|
||||
ion-searchbar {
|
||||
|
||||
color: white;
|
||||
--background: rgb(49, 49, 49);
|
||||
--icon-color: rgb(27, 173, 100);
|
||||
// --border-radius: 0;
|
||||
}
|
||||
}
|
||||
|
||||
.searchButton {
|
||||
|
||||
height: 2.2rem;
|
||||
margin-top: 0.9rem;
|
||||
|
||||
}
|
50
03_source/mobile/src/pages/DemoRecipeApp/pages/Category.jsx
Normal file
50
03_source/mobile/src/pages/DemoRecipeApp/pages/Category.jsx
Normal file
@@ -0,0 +1,50 @@
|
||||
import { IonBackButton, IonButtons, IonContent, IonHeader, IonList, IonPage, IonTitle, IonToolbar } from '@ionic/react';
|
||||
import { useEffect } from 'react';
|
||||
import { useState } from 'react';
|
||||
import { useParams } from 'react-router';
|
||||
import { RecipeListItem } from '../components/RecipeListItem';
|
||||
import { recipes } from '../recipes';
|
||||
|
||||
const Category = () => {
|
||||
|
||||
const { name } = useParams();
|
||||
const [ categoryRecipes, setCategoryRecipes ] = useState([]);
|
||||
|
||||
useEffect(() => {
|
||||
|
||||
setCategoryRecipes(recipes[name.toLowerCase()].hits);
|
||||
}, [ name ]);
|
||||
|
||||
return (
|
||||
<IonPage>
|
||||
<IonHeader>
|
||||
<IonToolbar>
|
||||
<IonButtons slot="start">
|
||||
<IonBackButton text="Categories" />
|
||||
</IonButtons>
|
||||
<IonTitle>{ name } Recipes</IonTitle>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
<IonContent fullscreen>
|
||||
<IonHeader collapse="condense">
|
||||
<IonToolbar>
|
||||
<IonTitle size="large">{ name } Recipes</IonTitle>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
|
||||
<IonList>
|
||||
{ categoryRecipes.map((categoryRecipe, index) => {
|
||||
|
||||
const { recipe } = categoryRecipe;
|
||||
|
||||
return (
|
||||
<RecipeListItem recipe={ recipe } key={ `recipe_${ index }` } />
|
||||
);
|
||||
})}
|
||||
</IonList>
|
||||
</IonContent>
|
||||
</IonPage>
|
||||
);
|
||||
};
|
||||
|
||||
export default Category;
|
165
03_source/mobile/src/pages/DemoRecipeApp/pages/Recipe.jsx
Normal file
165
03_source/mobile/src/pages/DemoRecipeApp/pages/Recipe.jsx
Normal file
@@ -0,0 +1,165 @@
|
||||
import { IonBackButton, IonButton, IonButtons, IonCardSubtitle, IonCardTitle, IonCol, IonContent, IonGrid, IonHeader, IonIcon, IonList, IonListHeader, IonPage, IonRow, IonTitle, IonToolbar, useIonModal, useIonToast } from '@ionic/react';
|
||||
import { bookmark, bookmarkOutline, informationCircleOutline, layersOutline, peopleOutline, timeOutline } from 'ionicons/icons';
|
||||
import { useEffect, useRef } from 'react';
|
||||
import { useState } from 'react';
|
||||
import { useLocation } from 'react-router';
|
||||
import { Ingredient } from '../components/Ingredient';
|
||||
import IngredientsModal from '../components/IngredientsModal';
|
||||
import NutritionModal from '../components/NutritionModal';
|
||||
import BookmarkStore, { addToBookmarks } from '../store/BookmarkStore';
|
||||
import styles from "./Recipe.module.scss";
|
||||
|
||||
import { useStoreState } from "pullstate";
|
||||
import { getBookmarks } from '../store/Selectors';
|
||||
|
||||
const Recipe = () => {
|
||||
|
||||
const pageRef = useRef();
|
||||
const { state } = useLocation();
|
||||
const [ recipe, setRecipe ] = useState([]);
|
||||
const [ fromSearch, setFromSearch ] = useState(false);
|
||||
const [ fromBookmarks, setFromBookmarks ] = useState(false);
|
||||
|
||||
const bookmarks = useStoreState(BookmarkStore, getBookmarks);
|
||||
|
||||
const [ showToast ] = useIonToast();
|
||||
|
||||
const handleDismissIngredientsModal = () => {
|
||||
|
||||
hideIngredientsModal();
|
||||
}
|
||||
|
||||
const handleDismissNutritionModal = () => {
|
||||
|
||||
hideNutritionModal();
|
||||
}
|
||||
|
||||
const [ showIngredientsModal, hideIngredientsModal ] = useIonModal(IngredientsModal, {
|
||||
|
||||
dismiss: handleDismissIngredientsModal,
|
||||
ingredients: recipe.ingredients
|
||||
});
|
||||
|
||||
const [ showNutritionModal, hideNutritionModal ] = useIonModal(NutritionModal, {
|
||||
|
||||
dismiss: handleDismissNutritionModal,
|
||||
recipe
|
||||
});
|
||||
|
||||
useEffect(() => {
|
||||
|
||||
if (state && state.recipe) {
|
||||
|
||||
setRecipe(state.recipe);
|
||||
}
|
||||
|
||||
if (state && state.fromSearch) {
|
||||
|
||||
setFromSearch(state.fromSearch);
|
||||
}
|
||||
|
||||
if (state && state.fromBookmarks) {
|
||||
|
||||
setFromBookmarks(state.fromBookmarks);
|
||||
}
|
||||
}, [ state ]);
|
||||
|
||||
const addBookmark = async () => {
|
||||
|
||||
const added = addToBookmarks(recipe);
|
||||
showToast({
|
||||
|
||||
message: added ? "This recipe has been bookmarked!" : "This recipe has been removed from your bookmarks.",
|
||||
duration: 2000,
|
||||
color: "main"
|
||||
});
|
||||
}
|
||||
|
||||
return (
|
||||
<IonPage ref={ pageRef }>
|
||||
<IonHeader>
|
||||
<IonToolbar>
|
||||
<IonTitle>View Recipe</IonTitle>
|
||||
<IonButtons slot="start">
|
||||
<IonBackButton text={ fromSearch ? "Search" : fromBookmarks ? "Bookmarks" : "Recipes" } color="main" />
|
||||
</IonButtons>
|
||||
|
||||
<IonButtons slot="end">
|
||||
<IonButton onClick={ addBookmark }>
|
||||
<IonIcon icon={ bookmarks.includes(recipe) ? bookmark : bookmarkOutline } />
|
||||
</IonButton>
|
||||
</IonButtons>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
<IonContent fullscreen>
|
||||
|
||||
<div className={ styles.headerImage }>
|
||||
<img src={ recipe.image } alt="main cover" />
|
||||
<div className={ `${ styles.headerInfo } animate__animated animate__slideInLeft` }>
|
||||
<h1>{ recipe.label }</h1>
|
||||
<p>{ recipe.dishType && recipe.dishType[0] }</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<IonGrid>
|
||||
|
||||
<IonRow className="ion-text-center">
|
||||
<IonCol size="4">
|
||||
<IonCardTitle>
|
||||
<IonIcon icon={ peopleOutline } />
|
||||
</IonCardTitle>
|
||||
<IonCardSubtitle>serves { recipe.yield && recipe.yield }</IonCardSubtitle>
|
||||
</IonCol>
|
||||
<IonCol size="4">
|
||||
<IonCardTitle>
|
||||
<IonIcon icon={ timeOutline } />
|
||||
</IonCardTitle>
|
||||
<IonCardSubtitle>{ recipe.totalTime !== 0 ? `${ recipe.totalWeight && recipe.totalWeight.toFixed(0) } mins` : "N/A" }</IonCardSubtitle>
|
||||
</IonCol>
|
||||
<IonCol size="4">
|
||||
<IonCardTitle>
|
||||
<IonIcon icon={ layersOutline } />
|
||||
</IonCardTitle>
|
||||
<IonCardSubtitle>{ recipe.totalWeight && recipe.totalWeight.toFixed(0) }g</IonCardSubtitle>
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
|
||||
<IonRow className="ion-text-center">
|
||||
<IonCol size="6">
|
||||
{/* <IonButton color="main" onClick={ () => showIngredientsModal({
|
||||
|
||||
presentingElement: pageRef.current,
|
||||
cssClass: "customModal"
|
||||
})}>
|
||||
<IonIcon icon={ informationCircleOutline } />
|
||||
View Ingredients
|
||||
</IonButton> */}
|
||||
</IonCol>
|
||||
<IonCol size="12">
|
||||
<IonButton expand="block" color="main" onClick={ () => showNutritionModal({
|
||||
|
||||
presentingElement: pageRef.current,
|
||||
cssClass: "customModal"
|
||||
})}>
|
||||
<IonIcon icon={ informationCircleOutline } />
|
||||
View Nutrition
|
||||
</IonButton>
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
|
||||
{ recipe.ingredients &&
|
||||
<IonList>
|
||||
<IonListHeader>Ingredients ({ recipe.ingredients.length })</IonListHeader>
|
||||
{ recipe.ingredients.map((ingredient, index) => {
|
||||
|
||||
return <Ingredient key={ `ingredient_${ index }` } ingredient={ ingredient } />;
|
||||
})}
|
||||
</IonList>
|
||||
}
|
||||
</IonGrid>
|
||||
</IonContent>
|
||||
</IonPage>
|
||||
);
|
||||
};
|
||||
|
||||
export default Recipe;
|
@@ -0,0 +1,37 @@
|
||||
.headerImage {
|
||||
|
||||
img {
|
||||
|
||||
width: 100%;
|
||||
margin-top: -5rem;
|
||||
border-bottom: 5px solid var(--ion-color-main);
|
||||
}
|
||||
|
||||
.headerInfo {
|
||||
|
||||
position: absolute;
|
||||
top: 10rem;
|
||||
z-index: 10;
|
||||
|
||||
background-color: rgba($color: #000000, $alpha: 0.8);
|
||||
color: white;
|
||||
padding: 1rem;
|
||||
border-top-right-radius: 10px;
|
||||
border-bottom-right-radius: 10px;
|
||||
|
||||
h1 {
|
||||
|
||||
font-size: 1.1rem;
|
||||
padding: 0;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
p {
|
||||
|
||||
font-size: 0.9rem;
|
||||
padding: 0;
|
||||
margin: 0;
|
||||
color: var(--ion-color-main);
|
||||
}
|
||||
}
|
||||
}
|
106
03_source/mobile/src/pages/DemoRecipeApp/pages/Search.jsx
Normal file
106
03_source/mobile/src/pages/DemoRecipeApp/pages/Search.jsx
Normal file
@@ -0,0 +1,106 @@
|
||||
import { IonBackButton, IonButton, IonButtons, IonCol, IonContent, IonGrid, IonHeader, IonImg, IonList, IonNote, IonPage, IonRow, IonSearchbar, IonText, IonTitle, IonToolbar, useIonLoading, useIonViewDidEnter } from "@ionic/react"
|
||||
import { useState } from "react";
|
||||
import { useRef } from "react";
|
||||
import styles from "./Categories.module.scss";
|
||||
import { performSearch } from "../utils";
|
||||
import { RecipeListItem } from "../components/RecipeListItem";
|
||||
|
||||
const Search = () => {
|
||||
|
||||
const searchRef = useRef();
|
||||
const [ searchResults, setSearchResults ] = useState([]);
|
||||
const [ showLoader, hideLoader ] = useIonLoading();
|
||||
|
||||
useIonViewDidEnter(() => {
|
||||
|
||||
searchRef.current.setFocus();
|
||||
});
|
||||
|
||||
const search = async () => {
|
||||
|
||||
showLoader({
|
||||
cssClass: "customLoader",
|
||||
message: "Searching...",
|
||||
duration: 9999999,
|
||||
spinner: "dots"
|
||||
});
|
||||
|
||||
const searchTerm = searchRef.current.value;
|
||||
const data = await performSearch(searchTerm);
|
||||
|
||||
setSearchResults(data.hits);
|
||||
|
||||
setTimeout(() => {
|
||||
|
||||
hideLoader();
|
||||
}, 300);
|
||||
}
|
||||
|
||||
return (
|
||||
|
||||
<IonPage>
|
||||
<IonHeader>
|
||||
<IonToolbar>
|
||||
<IonTitle>Search Recipes</IonTitle>
|
||||
<IonButtons slot="start">
|
||||
<IonBackButton color="main" text="Categories" />
|
||||
</IonButtons>
|
||||
</IonToolbar>
|
||||
</IonHeader>
|
||||
|
||||
<IonContent>
|
||||
<div className={ styles.searchArea }>
|
||||
|
||||
<IonGrid>
|
||||
<IonRow>
|
||||
<IonCol size="9">
|
||||
<IonSearchbar ref={ searchRef } placeholder="Try 'Chicken Piccata'" />
|
||||
</IonCol>
|
||||
<IonCol size="3">
|
||||
<IonButton className={ styles.searchButton } expand="block" color="main" onClick={ search }>Search</IonButton>
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
</IonGrid>
|
||||
</div>
|
||||
|
||||
{ searchResults.length > 0 &&
|
||||
<IonList>
|
||||
{ searchResults.map((result, index) => {
|
||||
|
||||
const { recipe } = result;
|
||||
|
||||
return (
|
||||
<RecipeListItem recipe={ recipe } key={ `result_${ index }` } fromSearch={ true } />
|
||||
);
|
||||
})}
|
||||
</IonList>
|
||||
}
|
||||
|
||||
{ searchResults.length < 1 &&
|
||||
|
||||
<>
|
||||
<IonRow className="ion-justify-content-center ion-text-center ion-margin-top ion-padding-top">
|
||||
<IonCol size="8">
|
||||
<IonText>
|
||||
Search for a recipe then select from the list to view it
|
||||
</IonText>
|
||||
|
||||
<IonNote>
|
||||
<p>For development purposes, only 20 results will be returned</p>
|
||||
</IonNote>
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
|
||||
<IonRow className="ion-justify-content-center">
|
||||
<IonCol size="8">
|
||||
<IonImg src="/assets/placeholder.png" />
|
||||
</IonCol>
|
||||
</IonRow>
|
||||
</>
|
||||
}
|
||||
</IonContent>
|
||||
</IonPage>
|
||||
);
|
||||
}
|
||||
|
||||
export default Search;
|
141
03_source/mobile/src/pages/DemoRecipeApp/theme/variables.css
Normal file
141
03_source/mobile/src/pages/DemoRecipeApp/theme/variables.css
Normal file
@@ -0,0 +1,141 @@
|
||||
/* Ionic Variables and Theming. For more info, please see:
|
||||
http://ionicframework.com/docs/theming/ */
|
||||
|
||||
/** Ionic CSS Variables **/
|
||||
:root {
|
||||
/** primary **/
|
||||
--ion-color-primary: #3880ff;
|
||||
--ion-color-primary-rgb: 56, 128, 255;
|
||||
--ion-color-primary-contrast: #ffffff;
|
||||
--ion-color-primary-contrast-rgb: 255, 255, 255;
|
||||
--ion-color-primary-shade: #3171e0;
|
||||
--ion-color-primary-tint: #4c8dff;
|
||||
|
||||
/** secondary **/
|
||||
--ion-color-secondary: #3dc2ff;
|
||||
--ion-color-secondary-rgb: 61, 194, 255;
|
||||
--ion-color-secondary-contrast: #ffffff;
|
||||
--ion-color-secondary-contrast-rgb: 255, 255, 255;
|
||||
--ion-color-secondary-shade: #36abe0;
|
||||
--ion-color-secondary-tint: #50c8ff;
|
||||
|
||||
/** tertiary **/
|
||||
--ion-color-tertiary: #5260ff;
|
||||
--ion-color-tertiary-rgb: 82, 96, 255;
|
||||
--ion-color-tertiary-contrast: #ffffff;
|
||||
--ion-color-tertiary-contrast-rgb: 255, 255, 255;
|
||||
--ion-color-tertiary-shade: #4854e0;
|
||||
--ion-color-tertiary-tint: #6370ff;
|
||||
|
||||
/** success **/
|
||||
--ion-color-success: #2dd36f;
|
||||
--ion-color-success-rgb: 45, 211, 111;
|
||||
--ion-color-success-contrast: #ffffff;
|
||||
--ion-color-success-contrast-rgb: 255, 255, 255;
|
||||
--ion-color-success-shade: #28ba62;
|
||||
--ion-color-success-tint: #42d77d;
|
||||
|
||||
/** warning **/
|
||||
--ion-color-warning: #ffc409;
|
||||
--ion-color-warning-rgb: 255, 196, 9;
|
||||
--ion-color-warning-contrast: #000000;
|
||||
--ion-color-warning-contrast-rgb: 0, 0, 0;
|
||||
--ion-color-warning-shade: #e0ac08;
|
||||
--ion-color-warning-tint: #ffca22;
|
||||
|
||||
/** danger **/
|
||||
--ion-color-danger: #eb445a;
|
||||
--ion-color-danger-rgb: 235, 68, 90;
|
||||
--ion-color-danger-contrast: #ffffff;
|
||||
--ion-color-danger-contrast-rgb: 255, 255, 255;
|
||||
--ion-color-danger-shade: #cf3c4f;
|
||||
--ion-color-danger-tint: #ed576b;
|
||||
|
||||
/** dark **/
|
||||
--ion-color-dark: #222428;
|
||||
--ion-color-dark-rgb: 34, 36, 40;
|
||||
--ion-color-dark-contrast: #ffffff;
|
||||
--ion-color-dark-contrast-rgb: 255, 255, 255;
|
||||
--ion-color-dark-shade: #1e2023;
|
||||
--ion-color-dark-tint: #383a3e;
|
||||
|
||||
/** medium **/
|
||||
--ion-color-medium: #92949c;
|
||||
--ion-color-medium-rgb: 146, 148, 156;
|
||||
--ion-color-medium-contrast: #ffffff;
|
||||
--ion-color-medium-contrast-rgb: 255, 255, 255;
|
||||
--ion-color-medium-shade: #808289;
|
||||
--ion-color-medium-tint: #9d9fa6;
|
||||
|
||||
/** light **/
|
||||
--ion-color-light: #f4f5f8;
|
||||
--ion-color-light-rgb: 244, 245, 248;
|
||||
--ion-color-light-contrast: #000000;
|
||||
--ion-color-light-contrast-rgb: 0, 0, 0;
|
||||
--ion-color-light-shade: #d7d8da;
|
||||
--ion-color-light-tint: #f5f6f9;
|
||||
|
||||
--ion-color-main: #1BAD64;
|
||||
--ion-color-main-rgb: 27,173,100;
|
||||
--ion-color-main-contrast: #ffffff;
|
||||
--ion-color-main-contrast-rgb: 255,255,255;
|
||||
--ion-color-main-shade: #189858;
|
||||
--ion-color-main-tint: #32b574;
|
||||
|
||||
--ion-background-color: white;
|
||||
--ion-tab-bar-background: rgb(36, 36, 36);
|
||||
--ion-toolbar-background: rgb(36, 36, 36);
|
||||
--ion-tab-bar-color-selected: rgb(27, 173, 100);
|
||||
--ion-tab-bar-color: rgb(97, 97, 97);
|
||||
}
|
||||
|
||||
.ion-color-main {
|
||||
--ion-color-base: var(--ion-color-main);
|
||||
--ion-color-base-rgb: var(--ion-color-main-rgb);
|
||||
--ion-color-contrast: var(--ion-color-main-contrast);
|
||||
--ion-color-contrast-rgb: var(--ion-color-main-contrast-rgb);
|
||||
--ion-color-shade: var(--ion-color-main-shade);
|
||||
--ion-color-tint: var(--ion-color-main-tint);
|
||||
}
|
||||
|
||||
ion-toolbar ion-title {
|
||||
|
||||
--color: white;
|
||||
}
|
||||
|
||||
ion-toolbar ion-back-button,
|
||||
ion-toolbar ion-button {
|
||||
|
||||
--color: rgb(27, 173, 100);
|
||||
}
|
||||
|
||||
ion-content ion-toolbar ion-title {
|
||||
|
||||
color: rgb(36, 36, 36);
|
||||
}
|
||||
|
||||
* {
|
||||
|
||||
font-family: 'Ubuntu', sans-serif;
|
||||
}
|
||||
|
||||
ion-toolbar {
|
||||
--border-style: none;
|
||||
}
|
||||
|
||||
.customLoader {
|
||||
|
||||
--background: var(--ion-toolbar-background);
|
||||
--spinner-color: var(--ion-tab-bar-color-selected);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.customModal {
|
||||
|
||||
--background: var(--ion-toolbar-background);
|
||||
}
|
||||
|
||||
a {
|
||||
|
||||
text-decoration: none;
|
||||
}
|
Reference in New Issue
Block a user