mirror of
				https://github.com/mealie-recipes/mealie.git
				synced 2025-10-31 10:13:32 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			89 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			89 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import DOMPurify from "isomorphic-dompurify";
 | |
| import { useFraction } from "./use-fraction";
 | |
| import { CreateIngredientFood, CreateIngredientUnit, IngredientFood, IngredientUnit, RecipeIngredient } from "~/lib/api/types/recipe";
 | |
| const { frac } = useFraction();
 | |
| 
 | |
| export function sanitizeIngredientHTML(rawHtml: string) {
 | |
|   return DOMPurify.sanitize(rawHtml, {
 | |
|     USE_PROFILES: { html: true },
 | |
|     ALLOWED_TAGS: ["b", "q", "i", "strong", "sup"],
 | |
|   });
 | |
| }
 | |
| 
 | |
| function useFoodName(food: CreateIngredientFood | IngredientFood | undefined, usePlural: boolean) {
 | |
|   if (!food) {
 | |
|     return "";
 | |
|   }
 | |
| 
 | |
|   return (usePlural ? food.pluralName || food.name : food.name) || "";
 | |
| }
 | |
| 
 | |
| function useUnitName(unit: CreateIngredientUnit | IngredientUnit | undefined, usePlural: boolean) {
 | |
|   if (!unit) {
 | |
|     return "";
 | |
|   }
 | |
| 
 | |
|   let returnVal = "";
 | |
|   if (unit.useAbbreviation) {
 | |
|     returnVal = (usePlural ? unit.pluralAbbreviation || unit.abbreviation : unit.abbreviation) || "";
 | |
|   }
 | |
| 
 | |
|   if (!returnVal) {
 | |
|     returnVal = (usePlural ? unit.pluralName || unit.name : unit.name) || "";
 | |
|   }
 | |
| 
 | |
|   return returnVal;
 | |
| }
 | |
| 
 | |
| export function useParsedIngredientText(ingredient: RecipeIngredient, disableAmount: boolean, scale = 1, includeFormating = true) {
 | |
|   if (disableAmount) {
 | |
|     return {
 | |
|       name: ingredient.note ? sanitizeIngredientHTML(ingredient.note) : undefined,
 | |
|       quantity: undefined,
 | |
|       unit: undefined,
 | |
|       note: undefined,
 | |
|     };
 | |
|   }
 | |
| 
 | |
|   const { quantity, food, unit, note } = ingredient;
 | |
|   const usePluralUnit = quantity !== undefined && (quantity * scale > 1 || quantity * scale === 0);
 | |
|   const usePluralFood = (!quantity) || quantity * scale > 1
 | |
| 
 | |
|   let returnQty = "";
 | |
| 
 | |
|   // casting to number is required as sometimes quantity is a string
 | |
|   if (quantity && Number(quantity) !== 0) {
 | |
|     if (unit && !unit.fraction) {
 | |
|       returnQty = (quantity * scale).toString();
 | |
|     } else {
 | |
|       const fraction = frac(quantity * scale, 10, true);
 | |
|       if (fraction[0] !== undefined && fraction[0] > 0) {
 | |
|         returnQty += fraction[0];
 | |
|       }
 | |
| 
 | |
|       if (fraction[1] > 0) {
 | |
|         returnQty += includeFormating ?
 | |
|           `<sup>${fraction[1]}</sup><span>⁄</span><sub>${fraction[2]}</sub>` :
 | |
|           ` ${fraction[1]}/${fraction[2]}`;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const unitName = useUnitName(unit, usePluralUnit);
 | |
|   const foodName = useFoodName(food, usePluralFood);
 | |
| 
 | |
|   return {
 | |
|     quantity: returnQty ? sanitizeIngredientHTML(returnQty) : undefined,
 | |
|     unit: unitName && quantity ? sanitizeIngredientHTML(unitName) : undefined,
 | |
|     name: foodName ? sanitizeIngredientHTML(foodName) : undefined,
 | |
|     note: note ? sanitizeIngredientHTML(note) : undefined,
 | |
|   };
 | |
| }
 | |
| 
 | |
| export function parseIngredientText(ingredient: RecipeIngredient, disableAmount: boolean, scale = 1, includeFormating = true): string {
 | |
|   const { quantity, unit, name, note } = useParsedIngredientText(ingredient, disableAmount, scale, includeFormating);
 | |
| 
 | |
|   const text = `${quantity || ""} ${unit || ""} ${name || ""} ${note || ""}`.replace(/ {2,}/g, " ").trim();
 | |
|   return sanitizeIngredientHTML(text);
 | |
| }
 |