Files
agent-skills/skills/vue-best-practices/reference/attrs-hyphenated-property-access.md
Jason Woltje f5792c40be feat: Complete fleet — 94 skills across 10+ domains
Pulled ALL skills from 15 source repositories:
- anthropics/skills: 16 (docs, design, MCP, testing)
- obra/superpowers: 14 (TDD, debugging, agents, planning)
- coreyhaines31/marketingskills: 25 (marketing, CRO, SEO, growth)
- better-auth/skills: 5 (auth patterns)
- vercel-labs/agent-skills: 5 (React, design, Vercel)
- antfu/skills: 16 (Vue, Vite, Vitest, pnpm, Turborepo)
- Plus 13 individual skills from various repos

Mosaic Stack is not limited to coding — the Orchestrator and
subagents serve coding, business, design, marketing, writing,
logistics, analysis, and more.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-16 16:27:42 -06:00

187 lines
4.4 KiB
Markdown

# Accessing Hyphenated Attributes in $attrs
## Rule
Fallthrough attributes preserve their original casing in JavaScript. Hyphenated attribute names (like `data-testid` or `aria-label`) must be accessed using bracket notation. Event listeners are exposed as camelCase functions (e.g., `@click` becomes `$attrs.onClick`).
## Why This Matters
- JavaScript identifiers cannot contain hyphens
- Using dot notation with hyphenated names causes syntax errors or undefined values
- Event listener naming follows a different convention than attribute naming
- Common source of "undefined" errors when working with attrs programmatically
## Bad Code
```vue
<script setup>
import { useAttrs } from 'vue'
const attrs = useAttrs()
// WRONG: Syntax error - hyphen interpreted as minus
console.log(attrs.data-testid) // Error!
// WRONG: This accesses a different property
console.log(attrs.dataTestid) // undefined (camelCase doesn't work for attrs)
// WRONG: Expecting hyphenated event name
console.log(attrs['on-click']) // undefined
console.log(attrs['@click']) // undefined
</script>
```
## Good Code
```vue
<script setup>
import { useAttrs } from 'vue'
const attrs = useAttrs()
// CORRECT: Use bracket notation for hyphenated attributes
console.log(attrs['data-testid']) // "my-button"
console.log(attrs['aria-label']) // "Submit form"
console.log(attrs['foo-bar']) // "baz"
// CORRECT: Event listeners use camelCase with 'on' prefix
console.log(attrs.onClick) // function
console.log(attrs.onCustomEvent) // function (from @custom-event)
console.log(attrs.onMouseEnter) // function (from @mouseenter or @mouse-enter)
</script>
```
## Attribute vs Event Naming Reference
| Parent Usage | $attrs Access |
|--------------|---------------|
| `class="foo"` | `attrs.class` |
| `data-id="123"` | `attrs['data-id']` |
| `aria-label="..."` | `attrs['aria-label']` |
| `foo-bar="baz"` | `attrs['foo-bar']` |
| `@click="fn"` | `attrs.onClick` |
| `@custom-event="fn"` | `attrs.onCustomEvent` |
| `@update:modelValue="fn"` | `attrs['onUpdate:modelValue']` |
## Common Patterns
### Checking for specific attributes
```vue
<script setup>
import { useAttrs, computed } from 'vue'
const attrs = useAttrs()
// Check if data attribute exists
const hasTestId = computed(() => 'data-testid' in attrs)
// Get aria attribute with default
const ariaLabel = computed(() => attrs['aria-label'] ?? 'Default label')
</script>
```
### Filtering attributes by type
```vue
<script setup>
import { useAttrs, computed } from 'vue'
const attrs = useAttrs()
// Separate event listeners from other attributes
const { listeners, otherAttrs } = computed(() => {
const listeners = {}
const otherAttrs = {}
for (const [key, value] of Object.entries(attrs)) {
if (key.startsWith('on') && typeof value === 'function') {
listeners[key] = value
} else {
otherAttrs[key] = value
}
}
return { listeners, otherAttrs }
}).value
</script>
```
### Extracting data attributes
```vue
<script setup>
import { useAttrs, computed } from 'vue'
const attrs = useAttrs()
// Get all data-* attributes
const dataAttrs = computed(() => {
const result = {}
for (const [key, value] of Object.entries(attrs)) {
if (key.startsWith('data-')) {
result[key] = value
}
}
return result
})
</script>
<template>
<div v-bind="dataAttrs">
<!-- Only data attributes are bound -->
</div>
</template>
```
### Forwarding specific events
```vue
<script setup>
import { useAttrs } from 'vue'
defineOptions({
inheritAttrs: false
})
const attrs = useAttrs()
// Call parent's click handler with custom logic
function handleClick(event) {
console.log('Internal handling first')
// Then forward to parent if handler exists
if (attrs.onClick) {
attrs.onClick(event)
}
}
</script>
<template>
<button @click="handleClick">
<slot />
</button>
</template>
```
## TypeScript Considerations
```vue
<script setup lang="ts">
import { useAttrs } from 'vue'
const attrs = useAttrs()
// attrs is typed as Record<string, unknown>
// You may need to cast for specific usage
const testId = attrs['data-testid'] as string | undefined
const onClick = attrs.onClick as ((e: MouseEvent) => void) | undefined
</script>
```
## References
- [Fallthrough Attributes - Accessing in JavaScript](https://vuejs.org/guide/components/attrs.html#accessing-fallthrough-attributes-in-javascript)
- [Vue 3 $attrs Documentation](https://vuejs.org/api/component-instance.html#attrs)