Files
agent-skills/skills/vue-best-practices/reference/definemodel-hidden-modifier-props.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

4.2 KiB

title, impact, impactDescription, type, tags
title impact impactDescription type tags
defineModel Creates Hidden Modifier Props - Avoid Naming Conflicts MEDIUM defineModel automatically adds hidden *Modifiers props that can conflict with your prop names gotcha
vue3
v-model
defineModel
modifiers
props
naming

defineModel Creates Hidden Modifier Props - Avoid Naming Conflicts

Impact: MEDIUM - When using defineModel(), Vue automatically creates hidden props with the suffix Modifiers for each model. For example, a model named title will create both a title prop AND a hidden titleModifiers prop. This can cause unexpected conflicts if you have other props ending in "Modifiers".

Task Checklist

  • Don't create props that end with "Modifiers" when using defineModel
  • Be aware that each defineModel creates an associated *Modifiers prop
  • When using multiple models, avoid names where one model could conflict with another's modifier prop
  • Document custom modifiers to help consumers understand available options

Problem - Hidden props created automatically:

<script setup>
// This creates TWO props: modelValue and modelValueModifiers
const model = defineModel()

// This creates TWO props: title and titleModifiers
const title = defineModel('title')

// CONFLICT: This prop name collides with the hidden titleModifiers
const props = defineProps({
  titleModifiers: Object  // WRONG: Conflicts with defineModel's hidden prop
})
</script>

Parent using modifiers:

<template>
  <!-- Vue passes modifiers via the hidden *Modifiers prop -->
  <MyComponent v-model:title.capitalize.trim="text" />

  <!-- Child receives titleModifiers = { capitalize: true, trim: true } -->
</template>

Correct - Accessing modifiers in child:

<script setup>
// Access modifiers via destructuring
const [title, titleModifiers] = defineModel('title', {
  set(value) {
    // Apply modifiers to the value
    if (titleModifiers.capitalize) {
      value = value.charAt(0).toUpperCase() + value.slice(1)
    }
    if (titleModifiers.trim) {
      value = value.trim()
    }
    return value
  }
})
</script>

Multiple Models and Potential Conflicts

<script setup>
// These are OK - no conflicts
const name = defineModel('name')      // Creates: name, nameModifiers
const age = defineModel('age')        // Creates: age, ageModifiers

// PROBLEM: If you had a model named 'model', it creates:
// - 'model' prop
// - 'modelModifiers' prop
// But 'modelValue' also creates 'modelValueModifiers'!
// The names are similar and can cause confusion

// AVOID: Don't name a model something that would conflict
const model = defineModel('model')  // Creates 'model' and 'modelModifiers'
// This is confusing alongside the default modelValue/modelValueModifiers
</script>

Best Practice - Clear, distinct model names:

<script setup>
// Good: Clear, distinct names that won't conflict
const firstName = defineModel('firstName')   // firstNameModifiers
const lastName = defineModel('lastName')     // lastNameModifiers
const email = defineModel('email')           // emailModifiers

// Avoid ambiguous names
// Bad: const value = defineModel('value')  // valueModifiers - too generic
// Bad: const data = defineModel('data')    // dataModifiers - too generic
</script>

Documenting Custom Modifiers

When creating components with custom modifier support, document them clearly:

<script setup>
/**
 * @prop {string} title - The title text
 * @modifiers
 *   - capitalize: Capitalizes first letter
 *   - uppercase: Converts entire string to uppercase
 *   - trim: Removes leading/trailing whitespace
 */
const [title, modifiers] = defineModel('title', {
  set(value: string) {
    let result = value
    if (modifiers.trim) result = result.trim()
    if (modifiers.capitalize) {
      result = result.charAt(0).toUpperCase() + result.slice(1)
    }
    if (modifiers.uppercase) {
      result = result.toUpperCase()
    }
    return result
  }
})
</script>

<template>
  <input v-model="title" />
</template>

Reference