mirror of
				https://github.com/mealie-recipes/mealie.git
				synced 2025-10-25 15:53:26 -04:00 
			
		
		
		
	* add vitest * initialize lib w/ tests * move to dev dep * run tests in CI * update file names * move api folder to lib * move api and api types to same folder * update generator outpath * rm husky * i guess i _did_ need those types * reorg types * extract validators into testable components * (WIP) start composable testing * fix import type * fix linter complaint * simplify icon type def * fix linter errors (maybe?) * rename client file for sorting
		
			
				
	
	
		
			73 lines
		
	
	
		
			1.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			73 lines
		
	
	
		
			1.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|   <div v-if="items.length > 0">
 | |
|     <h2 v-if="title" class="mt-4">{{ title }}</h2>
 | |
|     <v-chip
 | |
|       v-for="category in items.slice(0, limit)"
 | |
|       :key="category.name"
 | |
|       label
 | |
|       class="ma-1"
 | |
|       color="accent"
 | |
|       :small="small"
 | |
|       dark
 | |
|       :to="`/recipes/${urlPrefix}/${category.slug}`"
 | |
|     >
 | |
|       {{ truncateText(category.name) }}
 | |
|     </v-chip>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script lang="ts">
 | |
| import { defineComponent } from "@nuxtjs/composition-api";
 | |
| import { RecipeCategory, RecipeTag, RecipeTool } from "~/lib/api/types/user";
 | |
| 
 | |
| export type UrlPrefixParam = "tags" | "categories" | "tools";
 | |
| 
 | |
| export default defineComponent({
 | |
|   props: {
 | |
|     truncate: {
 | |
|       type: Boolean,
 | |
|       default: false,
 | |
|     },
 | |
|     items: {
 | |
|       type: Array as () => RecipeCategory[] | RecipeTag[] | RecipeTool[],
 | |
|       default: () => [],
 | |
|     },
 | |
|     title: {
 | |
|       type: Boolean,
 | |
|       default: false,
 | |
|     },
 | |
|     urlPrefix: {
 | |
|       type: String as () => UrlPrefixParam,
 | |
|       default: "categories",
 | |
|     },
 | |
|     limit: {
 | |
|       type: Number,
 | |
|       default: 999,
 | |
|     },
 | |
|     small: {
 | |
|       type: Boolean,
 | |
|       default: false,
 | |
|     },
 | |
|     maxWidth: {
 | |
|       type: String,
 | |
|       default: null,
 | |
|     },
 | |
|   },
 | |
|   setup(props) {
 | |
|     function truncateText(text: string, length = 20, clamp = "...") {
 | |
|       if (!props.truncate) return text;
 | |
|       const node = document.createElement("div");
 | |
|       node.innerHTML = text;
 | |
|       const content = node.textContent || "";
 | |
|       return content.length > length ? content.slice(0, length) + clamp : content;
 | |
|     }
 | |
| 
 | |
|     return {
 | |
|       truncateText,
 | |
|     };
 | |
|   },
 | |
| });
 | |
| </script>
 | |
| 
 | |
| <style></style>
 |