| 
									
										
										
										
											2022-01-16 15:24:24 -09:00
										 |  |  | <template> | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  |   <v-list :class="tile ? 'd-flex flex-wrap background' : 'background'"> | 
					
						
							| 
									
										
										
										
											2023-10-07 16:06:00 -05:00
										 |  |  |     <v-sheet | 
					
						
							|  |  |  |       v-for="recipe, index in recipes" | 
					
						
							|  |  |  |       :key="recipe.id" | 
					
						
							|  |  |  |       :elevation="2" | 
					
						
							|  |  |  |       :class="attrs.class.sheet" | 
					
						
							|  |  |  |       :style="tile ? 'max-width: 100%; width: fit-content;' : 'width: 100%;'" | 
					
						
							|  |  |  |     > | 
					
						
							| 
									
										
										
										
											2023-11-20 18:43:44 +00:00
										 |  |  |       <v-list-item :to="'/g/' + groupSlug + '/r/' + recipe.slug" :class="attrs.class.listItem"> | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  |         <v-list-item-avatar :class="attrs.class.avatar"> | 
					
						
							|  |  |  |           <v-icon :class="attrs.class.icon" dark :small="small"> {{ $globals.icons.primary }} </v-icon> | 
					
						
							|  |  |  |         </v-list-item-avatar> | 
					
						
							|  |  |  |         <v-list-item-content :class="attrs.class.text"> | 
					
						
							|  |  |  |           <v-list-item-title :class="listItem && listItemDescriptions[index] ? '' : 'pr-4'" :style="attrs.style.text.title"> | 
					
						
							|  |  |  |             {{ recipe.name }} | 
					
						
							|  |  |  |           </v-list-item-title> | 
					
						
							|  |  |  |           <v-list-item-subtitle v-if="showDescription">{{ recipe.description }}</v-list-item-subtitle> | 
					
						
							| 
									
										
										
										
											2023-10-07 14:23:47 -05:00
										 |  |  |           <v-list-item-subtitle v-if="listItem && listItemDescriptions[index]" :style="attrs.style.text.subTitle"> | 
					
						
							|  |  |  |             <!-- eslint-disable-next-line vue/no-v-html --> | 
					
						
							|  |  |  |             <div v-html="listItemDescriptions[index]"></div> | 
					
						
							|  |  |  |           </v-list-item-subtitle> | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  |         </v-list-item-content> | 
					
						
							|  |  |  |         <slot :name="'actions-' + recipe.id" :v-bind="{ item: recipe }"> </slot> | 
					
						
							|  |  |  |       </v-list-item> | 
					
						
							|  |  |  |     </v-sheet> | 
					
						
							| 
									
										
										
										
											2022-01-16 15:24:24 -09:00
										 |  |  |   </v-list> | 
					
						
							|  |  |  | </template> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <script lang="ts"> | 
					
						
							| 
									
										
										
										
											2023-11-05 19:07:02 -06:00
										 |  |  | import { computed, defineComponent, useContext, useRoute } from "@nuxtjs/composition-api"; | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  | import DOMPurify from "dompurify"; | 
					
						
							|  |  |  | import { useFraction } from "~/composables/recipes/use-fraction"; | 
					
						
							|  |  |  | import { ShoppingListItemOut } from "~/lib/api/types/group"; | 
					
						
							| 
									
										
										
										
											2022-10-22 11:51:07 -08:00
										 |  |  | import { RecipeSummary } from "~/lib/api/types/recipe"; | 
					
						
							| 
									
										
										
										
											2022-01-16 15:24:24 -09:00
										 |  |  | 
 | 
					
						
							|  |  |  | export default defineComponent({ | 
					
						
							|  |  |  |   props: { | 
					
						
							|  |  |  |     recipes: { | 
					
						
							|  |  |  |       type: Array as () => RecipeSummary[], | 
					
						
							|  |  |  |       required: true, | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  |     listItem: { | 
					
						
							|  |  |  |       type: Object as () => ShoppingListItemOut | undefined, | 
					
						
							|  |  |  |       default: undefined, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     small: { | 
					
						
							|  |  |  |       type: Boolean, | 
					
						
							|  |  |  |       default: false, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     tile: { | 
					
						
							|  |  |  |       type: Boolean, | 
					
						
							|  |  |  |       default: false, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     showDescription: { | 
					
						
							|  |  |  |       type: Boolean, | 
					
						
							|  |  |  |       default: false, | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2022-01-16 15:24:24 -09:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  |   setup(props) { | 
					
						
							| 
									
										
										
										
											2023-11-05 19:07:02 -06:00
										 |  |  |     const { $auth } = useContext(); | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  |     const { frac } = useFraction(); | 
					
						
							| 
									
										
										
										
											2023-11-05 19:07:02 -06:00
										 |  |  |     const route = useRoute(); | 
					
						
							|  |  |  |     const groupSlug = computed(() => route.value.params.groupSlug || $auth.user?.groupSlug || ""); | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const attrs = computed(() => { | 
					
						
							|  |  |  |       return props.small ? { | 
					
						
							|  |  |  |         class: { | 
					
						
							|  |  |  |           sheet: props.tile ? "mb-1 me-1 justify-center align-center" : "mb-1 justify-center align-center", | 
					
						
							|  |  |  |           listItem: "px-0", | 
					
						
							|  |  |  |           avatar: "ma-0", | 
					
						
							|  |  |  |           icon: "ma-0 pa-0 primary", | 
					
						
							|  |  |  |           text: "pa-0", | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         style: { | 
					
						
							|  |  |  |           text: { | 
					
						
							|  |  |  |             title: "font-size: small;", | 
					
						
							|  |  |  |             subTitle: "font-size: x-small;", | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       } : { | 
					
						
							|  |  |  |         class: { | 
					
						
							|  |  |  |           sheet: props.tile ? "mx-1 justify-center align-center" : "mb-1 justify-center align-center", | 
					
						
							|  |  |  |           listItem: "px-4", | 
					
						
							|  |  |  |           avatar: "", | 
					
						
							|  |  |  |           icon: "pa-1 primary", | 
					
						
							|  |  |  |           text: "", | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         style: { | 
					
						
							|  |  |  |           text: { | 
					
						
							|  |  |  |             title: "", | 
					
						
							|  |  |  |             subTitle: "", | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function sanitizeHTML(rawHtml: string) { | 
					
						
							|  |  |  |       return DOMPurify.sanitize(rawHtml, { | 
					
						
							|  |  |  |         USE_PROFILES: { html: true }, | 
					
						
							|  |  |  |         ALLOWED_TAGS: ["strong", "sup"], | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const listItemDescriptions = computed<string[]>(() => { | 
					
						
							|  |  |  |       if ( | 
					
						
							|  |  |  |           props.recipes.length === 1  // we don't need to specify details if there's only one recipe ref
 | 
					
						
							|  |  |  |           || !props.listItem?.recipeReferences | 
					
						
							|  |  |  |           || props.listItem.recipeReferences.length !== props.recipes.length | 
					
						
							|  |  |  |         ) { | 
					
						
							|  |  |  |         return props.recipes.map((_) => "") | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const listItemDescriptions: string[] = []; | 
					
						
							|  |  |  |       for (let i = 0; i < props.recipes.length; i++) { | 
					
						
							|  |  |  |         const itemRef = props.listItem?.recipeReferences[i]; | 
					
						
							|  |  |  |         const quantity = (itemRef.recipeQuantity || 1) * (itemRef.recipeScale || 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let listItemDescription = "" | 
					
						
							|  |  |  |         if (props.listItem.unit?.fraction) { | 
					
						
							|  |  |  |             const fraction = frac(quantity, 10, true); | 
					
						
							|  |  |  |             if (fraction[0] !== undefined && fraction[0] > 0) { | 
					
						
							|  |  |  |               listItemDescription += fraction[0]; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (fraction[1] > 0) { | 
					
						
							|  |  |  |               listItemDescription += ` <sup>${fraction[1]}</sup>⁄<sub>${fraction[2]}</sub>`; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |               listItemDescription = (quantity).toString(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           else { | 
					
						
							|  |  |  |             listItemDescription = (Math.round(quantity*100)/100).toString(); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (props.listItem.unit) { | 
					
						
							|  |  |  |             const unitDisplay = props.listItem.unit.useAbbreviation && props.listItem.unit.abbreviation | 
					
						
							|  |  |  |               ? props.listItem.unit.abbreviation : props.listItem.unit.name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             listItemDescription += ` ${unitDisplay}` | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (itemRef.recipeNote) { | 
					
						
							|  |  |  |             listItemDescription += `, ${itemRef.recipeNote}` | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           listItemDescriptions.push(sanitizeHTML(listItemDescription)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return listItemDescriptions; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       attrs, | 
					
						
							| 
									
										
										
										
											2023-11-05 19:07:02 -06:00
										 |  |  |       groupSlug, | 
					
						
							| 
									
										
										
										
											2023-08-21 12:18:37 -05:00
										 |  |  |       listItemDescriptions, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2022-01-16 15:24:24 -09:00
										 |  |  |   }, | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2022-10-22 11:51:07 -08:00
										 |  |  | </script> |