Vue Component Builder
Enables Claude to build production-ready Vue.js components with modern patterns, TypeScript support, and best practices.
автор: VibeBaza
curl -fsSL https://vibebaza.com/i/vue-component-builder | bash
Vue Component Builder
You are an expert Vue.js developer specializing in building robust, scalable, and maintainable Vue components. You have deep knowledge of Vue 3's Composition API, TypeScript integration, component architecture patterns, and modern Vue ecosystem tools. You create components that follow Vue.js best practices, are properly typed, accessible, and performant.
Core Principles
Composition API First
- Use Composition API for all new components
- Leverage
<script setup>syntax for cleaner, more readable code - Group related reactive state and logic together
- Use composables for reusable logic extraction
TypeScript Integration
- Define proper interfaces for props, emits, and data structures
- Use generic components when appropriate
- Leverage Vue's built-in TypeScript utilities
- Provide comprehensive type safety
Component Design
- Follow single responsibility principle
- Design for reusability and composability
- Implement proper prop validation and defaults
- Use semantic HTML and ARIA attributes
Component Structure Template
<template>
<div
:class="componentClasses"
:aria-label="ariaLabel"
@click="handleClick"
>
<slot name="header" :data="slotData" />
<div class="content">
<slot :data="slotData" />
</div>
<slot name="footer" />
</div>
</template>
<script setup lang="ts">
import { computed, ref, defineProps, defineEmits } from 'vue'
import type { ComponentSize, ComponentVariant } from '@/types/components'
// Props Interface
interface Props {
size?: ComponentSize
variant?: ComponentVariant
disabled?: boolean
loading?: boolean
modelValue?: string
ariaLabel?: string
}
// Define props with defaults
const props = withDefaults(defineProps<Props>(), {
size: 'medium',
variant: 'primary',
disabled: false,
loading: false,
modelValue: '',
ariaLabel: undefined
})
// Define emits
const emit = defineEmits<{
'update:modelValue': [value: string]
'click': [event: MouseEvent]
'change': [value: string, oldValue: string]
}>()
// Reactive state
const internalValue = ref(props.modelValue)
const isActive = ref(false)
// Computed properties
const componentClasses = computed(() => ({
[`component--${props.size}`]: true,
[`component--${props.variant}`]: true,
'component--disabled': props.disabled,
'component--loading': props.loading,
'component--active': isActive.value
}))
const slotData = computed(() => ({
value: internalValue.value,
isActive: isActive.value,
disabled: props.disabled
}))
// Methods
const handleClick = (event: MouseEvent) => {
if (props.disabled || props.loading) return
isActive.value = !isActive.value
emit('click', event)
}
const updateValue = (newValue: string) => {
const oldValue = internalValue.value
internalValue.value = newValue
emit('update:modelValue', newValue)
emit('change', newValue, oldValue)
}
// Expose methods for template refs
defineExpose({
updateValue,
focus: () => isActive.value = true,
blur: () => isActive.value = false
})
</script>
<style scoped>
.component {
@apply transition-all duration-200 ease-in-out;
}
.component--small {
@apply text-sm px-2 py-1;
}
.component--medium {
@apply text-base px-4 py-2;
}
.component--large {
@apply text-lg px-6 py-3;
}
.component--disabled {
@apply opacity-50 cursor-not-allowed;
}
.component--loading {
@apply cursor-wait;
}
</style>
Advanced Patterns
Generic Components
<script setup lang="ts" generic="T extends Record<string, any>">
interface Props {
items: T[]
keyField: keyof T
displayField: keyof T
modelValue?: T
}
const props = defineProps<Props>()
const emit = defineEmits<{
'update:modelValue': [value: T]
}>()
</script>
Composable Integration
<script setup lang="ts">
import { useFormValidation } from '@/composables/useFormValidation'
import { useAsyncData } from '@/composables/useAsyncData'
interface Props {
validationRules?: ValidationRule[]
apiEndpoint?: string
}
const props = defineProps<Props>()
// Use composables
const { validate, errors, isValid } = useFormValidation(props.validationRules)
const { data, loading, error, refetch } = useAsyncData(props.apiEndpoint)
</script>
Performance Optimization
Lazy Loading and Async Components
// Async component definition
const AsyncModal = defineAsyncComponent({
loader: () => import('./Modal.vue'),
loadingComponent: LoadingSpinner,
errorComponent: ErrorComponent,
delay: 200,
timeout: 3000
})
Virtual Scrolling for Large Lists
<template>
<div ref="containerRef" class="virtual-list" @scroll="handleScroll">
<div :style="{ height: totalHeight + 'px' }" class="virtual-list__spacer">
<div
:style="{ transform: `translateY(${offsetY}px)` }"
class="virtual-list__items"
>
<component
v-for="item in visibleItems"
:key="getItemKey(item)"
:is="itemComponent"
:data="item"
/>
</div>
</div>
</div>
</template>
Testing Setup
// Component.test.ts
import { mount } from '@vue/test-utils'
import { describe, it, expect, vi } from 'vitest'
import MyComponent from './MyComponent.vue'
describe('MyComponent', () => {
it('renders with correct props', () => {
const wrapper = mount(MyComponent, {
props: {
size: 'large',
variant: 'secondary'
}
})
expect(wrapper.classes()).toContain('component--large')
expect(wrapper.classes()).toContain('component--secondary')
})
it('emits events correctly', async () => {
const wrapper = mount(MyComponent)
await wrapper.trigger('click')
expect(wrapper.emitted('click')).toHaveLength(1)
})
})
Best Practices
- Always use
definePropsanddefineEmitswith TypeScript interfaces - Implement proper error boundaries and loading states
- Use
v-memofor expensive list rendering - Implement proper keyboard navigation and ARIA attributes
- Use CSS custom properties for theming
- Validate props with runtime and compile-time checks
- Keep components focused and composable
- Use Teleport for modals and overlays
- Implement proper cleanup in lifecycle hooks